# Lua与C++交互(二)———— 交互
C++ 调用lua
基础调用
再来温习一下
myName = “beauty girl”
- C++想要获取myName的值,根据规则,它需要把myName压入栈中,这样lua就能看到;
- lua从堆栈中获取myName的值,此时栈顶为空;
- lua拿着myName去全局表中查找与之对应的字符串;
- 全局表找到,并返回"beauty girl";
- lua把"beauty girl"压入栈中;
- C++从栈中获取"beauty girl"
用代码来实现
//创建一个statelua_State *L = luaL_newstate();// 重置栈顶索引 为了确认让栈顶的索引置为0 置0之后,我们入栈的第一个元素的索引就是1lua_settop(L, 0);// 把myName放入栈中,然后lua去全局表中查找,找到对应的字段,再放回栈中lua_getglobal(L, "myName");// 判断栈顶的值的类型是否为String, 返回非0值代表成功int isstr = lua_isstring(L, 1);//获取栈顶的值const char* str = lua_tostring(L, 1);lua_close(L);
C++获取Lua的table
helloTable = {name = “xxx”, sex = “man”}
和上面一样,要获取就先把helloTable放到栈顶,让Lua知道C++要获取这个值
//创建一个statelua_State *L = luaL_newstate();// 重置栈顶索引 为了确认让栈顶的索引置为0 置0之后,我们入栈的第一个元素的索引就是1lua_settop(L, 0);//获取helloTable的值 这个时候栈底 是 helloTablelua_getglobal(L, "helloTable");//我们想要获取table中name的值,那么就把name压入栈//这个时候栈中是 name,helloTable,其中name是栈顶lua_pushstring(L, "name");//lua api提供了一个获取table的函数 lua_gettable//该函数会从栈顶取得一个值,然后根据这个值去table中寻找对应的值,最后把找到的值放到栈顶 第二个参数是指table变量所在栈中的位置lua_gettable(L, -2); // -1永远是栈顶,那么helloTable就是-2,这里也可以用1//lua_gettable 会把值放到 栈顶 const char* sName = lua_tostring(pL, -1);
C++调用Lua函数
function helloAdd(num1, num2)return (num1 + num2)
end
这里有个新的函数 lua_call
第一个参数表示函数的参数个数,第二个参数表示函数返回值个数
Lua会先去堆栈取出参数,然后再取出函数对象,开始执行函数
//创建一个statelua_State *L = luaL_newstate();// 重置栈顶索引 为了确认让栈顶的索引置为0 置0之后,我们入栈的第一个元素的索引就是1lua_settop(L, 0);//把helloAdd函数对象放到栈中lua_getglobal(L, "helloAdd");//把函数所需要的参数入栈 lua_pushnumber(L, 10);lua_pushnumber(L, 5);//调用lua_calllua_call(L, 2, 1);int iResult = lua_tonumber(L, -1);
C++调用Lua table的函数
lua中table有两种函数
mytable={}
function mytable.StaticFunc()print("mytable.StaticFunc called.")
end
function mytable:Func()print("mytable:Func self:", self)
end
其中StaticFunc可以理解成table的静态函数,Func为table的成员函数
// 调用mytable表的静态函数
lua_getglobal(L, "mytable"); // 将名为mytable的全局table变量的值压栈
lua_pushstring(L, "StaticFunc"); // 将函数名为StaticFunc压栈
lua_gettable(L, -2); // 从索引为-2处的表中,读取key(在栈顶处)为StaticFunc的函数名 读取成功后,将key出栈,并将读取到的函数名入栈
lua_call(L, 0, 0); // 执行完后将StaticFunc弹出栈 注: 第一个0表示参数个数为0,第二个0表示无返回值// 调用mytable表的成员函数 采用新方法获取函数名
lua_getfield(L, -1, "Func");// 从索引为-1处的表中,读取key为Func的函数名 成功后将读取到的函数名入栈
lua_pushvalue(L, -2); // 将索引为-2处的表复制一份并压入栈顶
lua_call(L, 1, 0); // 执行完后将Func弹出栈 注: 1表示参数个数,即self指针,为当前table,第二个0表示无返回值
唯一不同的是lua_call的时候,需要注意第二个的值,成员函数默认需要传递self。
这里获取的时候,用到了函数lua_getfield
函数原型如下
void lua_getfield (lua_State *L, int index, const char *k);
Pushes onto the stack the value t[k], where t is the value at the given valid index. As in Lua, this function may trigger a metamethod for the “index” event
大概意思,将t[k]压入堆栈,t由参数index指定在栈中的位置
Lua 调用C++
Lua调用C++ 函数
大概的步骤如下:
- 将C++的函数包装成Lua环境认可的Lua_CFunction格式
- 将包装好的函数注册到Lua环境中
- 像使用普通Lua函数那样使用注册函数
简单的C++函数
int add(int a,int b)
{return a+b;
}
包装C++函数
int add(lua_state *L)
{int a = lua_tonumber(-1);int b = lua_tonumber(-2);int sum = a+b;// 将返回值压入栈中lua_pushnumber(L,sum);// 返回返回值个数return 1;
}
- Lua脚本里会调用add(lua_state *L)
- 调用add(lua_state *L)函数的时候,会反过来进行之前的C++调用lua
- Lua调用add(lua_state *L)函数之后,有一个返回值,需要压入栈中
- 最后return表示有多少个返回值,Lua支持多个返回值
最关键的一步,需要注册C++的函数,Lua才能调用
lua_register(L, "add", add);
Lua调用C++类
这里有两种方式,一个是用luaL_newlib方式
luaL_newlib方式
大概步骤如下:
- 新建创建对象函数,调用lua_newuserdata,创建一个对象指针,指向new出来的新的对象。
- 新建成员方法,调用lua_touserdata,得到从lua中传入的对象指针,调用成员方法。
- 调用luaL_newlib,将需要封装的C++函数放入到一个lua表中压入栈里。
- 将自定义模块,注册到Lua环境中。
- 在lua中,会首先调用创建对象函数,获得Student对象指针。通过Student对象指针,调用成员方法
Student.h
#pragma once#include <iostream>
#include <string>
using namespace std;class Student
{
public://构造/析构函数Student();~Student();//get/set函数string get_name();void set_name(string name);unsigned get_age();void set_age(unsigned age);//打印函数void print();private:string _name;unsigned _age;
};
Student.cpp
#include "Student.h"
using namespace std;Student::Student():_name("Empty"),_age(0)
{cout << "Student Constructor" << endl;
}Student::~Student()
{cout << "Student Destructor" << endl;
}string Student::get_name()
{return _name;
}void Student::set_name(string name)
{_name = name;
}unsigned Student::get_age()
{return _age;
}void Student::set_age(unsigned age)
{_age = age;
}void Student::print()
{cout << "name :" << _name << " age : " << _age << endl;
}
StudentRegFunc.h
#pragma once#include "Student.h"
extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}//------定义相关的全局函数------
//创建对象
int lua_create_new_student(lua_State* L);//get/set函数
int lua_get_name(lua_State* L);
int lua_set_name(lua_State* L);
int lua_get_age(lua_State* L);
int lua_set_age(lua_State* L);//打印函数
int lua_print(lua_State* L);//------注册全局函数供Lua使用------
static const luaL_Reg lua_reg_student_funcs[] = {{ "create", lua_create_new_student },{ "get_name", lua_get_name },{ "set_name", lua_set_name },{ "get_age", lua_get_age },{ "set_age", lua_set_age },{ "print", lua_print },{ NULL, NULL },
};int luaopen_student_libs(lua_State* L);
StudentRegFunc.cpp
#include "StudentRegFunc.h"int lua_create_new_student(lua_State* L)
{//创建一个对象指针放到stack里,返回给Lua中使用Student** s = (Student**)lua_newuserdata(L, sizeof(Student*));*s = new Student();return 1;
}int lua_get_name(lua_State* L)
{//得到第一个传入的对象参数(在stack最底部)Student** s = (Student**)lua_touserdata(L, 1);luaL_argcheck(L, s != NULL, 1, "invalid user data");//清空stacklua_settop(L, 0);//将数据放入stack中,供Lua使用lua_pushstring(L, (*s)->get_name().c_str());return 1;
}int lua_set_name(lua_State* L)
{//得到第一个传入的对象参数Student** s = (Student**)lua_touserdata(L, 1);luaL_argcheck(L, s != NULL, 1, "invalid user data");luaL_checktype(L, -1, LUA_TSTRING);std::string name = lua_tostring(L, -1);(*s)->set_name(name);return 0;
}int lua_get_age(lua_State* L)
{Student** s = (Student**)lua_touserdata(L, 1);luaL_argcheck(L, s != NULL, 1, "invalid user data");lua_pushinteger(L, (*s)->get_age());return 1;
}int lua_set_age(lua_State* L)
{Student** s = (Student**)lua_touserdata(L, 1);luaL_argcheck(L, s != NULL, 1, "invalid user data");luaL_checktype(L, -1, LUA_TNUMBER);(*s)->set_age((unsigned)lua_tointeger(L, -1));return 0;
}int lua_print(lua_State* L)
{Student** s = (Student**)lua_touserdata(L, 1);luaL_argcheck(L, s != NULL, 1, "invalid user data");(*s)->print();return 0;
}int luaopen_student_libs(lua_State* L)
{// 创建一张新的表,并把列表的函数注册进去luaL_newlib(L, lua_reg_student_funcs);return 1;
}
main.cpp
#include <iostream>
using namespace std;extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}#include "Student.h"
#include "StudentRegFunc.h"static const luaL_Reg lua_reg_libs[] = {{ "base", luaopen_base }, //系统模块{ "Student", luaopen_student_libs}, //模块名字Student,注册函数luaopen_student_libs{ NULL, NULL }
};int main(int argc, char* argv[])
{if (lua_State* L = luaL_newstate()) {//注册让lua使用的库const luaL_Reg* lua_reg = lua_reg_libs;for (; lua_reg->func; ++lua_reg) {luaL_requiref(L, lua_reg->name, lua_reg->func, 1);lua_pop(L, 1);}//加载脚本,如果出错,则打印错误if (luaL_dofile(L, "lua4.lua")) {cout << lua_tostring(L, -1) << endl;}lua_close(L);}else {cout << "luaL_newstate error !" << endl;}system("pause");return 0;
}
tolua
第二种方式是tolua,也就是tolua++
在cocos2dx中,基本都是用这种方式
//.h
class CMD_Data : public cocos2d::Ref
{
public:CMD_Data(unsigned short nlength);virtual ~CMD_Data();
public:void setMainCmdAndSubCmd(const unsigned short mainCmd, const unsigned short subCmd);unsigned short getMainCmd();unsigned short getSubCmd();
public:static CMD_Data *create(const int nLenth);//...
}//.cppvoid CMD_Data::setMainCmdAndSubCmd(const unsigned short mainCmd, const unsigned short subCmd)
{m_wMainCmd = mainCmd;m_wSubCmd = subCmd;
}CMD_Data * CMD_Data::create(const int nLenth)
{CMD_Data * pData = new(std::nothrow) CMD_Data(nLenth);if (pData){//pData->autorelease();return pData;}CC_SAFE_DELETE(pData);return nullptr;
}unsigned short CMD_Data::getMainCmd()
{return m_wMainCmd;
}unsigned short CMD_Data::getSubCmd()
{return m_wSubCmd;
}
注册
.h
#pragma once
#include "base/ccConfig.h"
#ifdef __cplusplus
extern "C" {
#endif
#include "tolua++.h"
#ifdef __cplusplus
}
#endif
int register_all_Cmd_Data();
.cpp
int register_all_Cmd_Data()
{auto engine = LuaEngine::getInstance();ScriptEngineManager::getInstance()->setScriptEngine(engine);lua_State* tolua_S = engine->getLuaStack()->getLuaState();tolua_usertype(tolua_S, "CMD_Data");tolua_cclass(tolua_S, "CMD_Data", "CMD_Data", "cc.Node", nullptr);tolua_beginmodule(tolua_S, "CMD_Data");tolua_function(tolua_S, "create", tolua_Cmd_Data_create);tolua_function(tolua_S, "setCmdInfo", tolua_Cmd_Data_setMainCmdAndSubCmd);tolua_function(tolua_S, "getMainCmd", tolua_Cmd_Data_getMainCmd);tolua_function(tolua_S, "getSubCmd", tolua_Cmd_Data_getSubCmd);tolua_function(tolua_S, "getBufferLength", tolua_Cmd_Data_getBufferLength);tolua_function(tolua_S, "getCurIndex", tolua_Cmd_Data_getCurIndex);tolua_function(tolua_S, "setSendMessageLength", tolua_Cmd_Data_setSendMessageLength);tolua_endmodule(tolua_S);std::string typeName = typeid(CMD_Data).name();g_luaType[typeName] = "CMD_Data";g_typeCast["CMD_Data"] = "CMD_Data";return 1;
}int tolua_Cmd_Data_create(lua_State* tolua_S)
{int argc = lua_gettop(tolua_S);CMD_Data *data = nullptr;if (argc == 2){int nLength = lua_tointeger(tolua_S, 2);data = CMD_Data::create(nLength);}else{CCLOG("error CMD_Data create");}int nID = (data) ? data->_ID : -1;int *pLuaID = (data) ? &data->_luaID : nullptr;toluafix_pushusertype_ccobject(tolua_S, nID, pLuaID, (void*)data, "Cmd_Data");return 1;
}int tolua_Cmd_Data_setMainCmdAndSubCmd(lua_State* tolua_S)
{CMD_Data *cobj = (CMD_Data*)tolua_tousertype(tolua_S, 1, nullptr);if (cobj){int argc = lua_gettop(tolua_S);if (argc == 3){unsigned short mainCmd = (unsigned short)lua_tointeger(tolua_S, 2);unsigned short subCmd = (unsigned short)lua_tointeger(tolua_S, 3);cobj->setMainCmdAndSubCmd(mainCmd, subCmd);}}return 0;
}int tolua_Cmd_Data_getMainCmd(lua_State* tolua_S)
{CMD_Data *cobj = (CMD_Data*)tolua_tousertype(tolua_S, 1, nullptr);if (cobj){lua_pushinteger(tolua_S, cobj->getMainCmd());return 1;}return 0;
}int tolua_Cmd_Data_getSubCmd(lua_State* tolua_S)
{CMD_Data *cobj = (CMD_Data*)tolua_tousertype(tolua_S, 1, nullptr);if (cobj){lua_pushinteger(tolua_S, cobj->getSubCmd());return 1;}return 0;
}
cocos2dx 生成工具
方便的是,cocos2dx提供了一套转换的工具。
在cocos2dx引擎目录下有个tools的目录,里面有tolua的文件夹。
它里面的大概结构如下
可以看得出来,cocos2dx本身的类都是用这个方法去实现的。
随便打开一个ini文件
[cocos2dx_ui]
# the prefix to be added to the generated functions. You might or might not use this in your own
# templates
prefix = cocos2dx_ui# create a target namespace (in javascript, this would create some code like the equiv. to `ns = ns || {}`)
# all classes will be embedded in that namespace
target_namespace = ccui# the native namespace in which this module locates, this parameter is used for avoid conflict of the same class name in different modules, as "cocos2d::Label" <-> "cocos2d::ui::Label".
cpp_namespace = cocos2d::uiandroid_headers = -I%(androidndkdir)s/platforms/android-14/arch-arm/usr/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/%(gnu_libstdc_version)s/libs/armeabi-v7a/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/%(gnu_libstdc_version)s/include
android_flags = -D_SIZE_T_DEFINED_ clang_headers = -I%(clangllvmdir)s/%(clang_lib_version)s/clang/%(clang_version)s/include
clang_flags = -nostdinc -x c++ -std=c++11 -U __SSE__cocos_headers = -I%(cocosdir)s/cocos -I%(cocosdir)s/cocos/editor-support -I%(cocosdir)s/cocos/platform/androidcocos_flags = -DANDROIDcxxgenerator_headers = # extra arguments for clang
extra_arguments = %(android_headers)s %(clang_headers)s %(cxxgenerator_headers)s %(cocos_headers)s %(android_flags)s %(clang_flags)s %(cocos_flags)s %(extra_flags)s # what headers to parse
headers = %(cocosdir)s/cocos/ui/CocosGUI.h# what classes to produce code for. You can use regular expressions here. When testing the regular
# expression, it will be enclosed in "^$", like this: "^Menu*$".
classes = Helper Widget Layout Button CheckBox ImageView Text TextAtlas TextBMFont LoadingBar Slider TextField ScrollView ListView PageView LayoutParameter LinearLayoutParameter RelativeLayoutParameter Rich.* HBox VBox RelativeBox Scale9Sprite EditBox LayoutComponent AbstractCheckButton RadioButton RadioButtonGroup TabControl TabHeader# what should we skip? in the format ClassName::[function function]
# ClassName is a regular expression, but will be used like this: "^ClassName$" functions are also
# regular expressions, they will not be surrounded by "^$". If you want to skip a whole class, just
# add a single "*" as functions. See bellow for several examples. A special class name is "*", which
# will apply to all class names. This is a convenience wildcard to be able to skip similar named
# functions from all classes.skip = *::[^visit$ copyWith.* onEnter.* onExit.* ^description$ getObjectType .*HSV onTouch.* onAcc.* onKey.* onRegisterTouchListener ccTouch.* (g|s)etDelegate],Widget::[addTouchEventListener addClickEventListener addCCSEventListener],LayoutParameter::[(s|g)etMargin],RichText::[setTagDescription removeTagDescription setOpenUrlHandler]rename_functions = rename_classes =# for all class names, should we remove something when registering in the target VM?
remove_prefix = # classes for which there will be no "parent" lookup
classes_have_no_parents = Helper# base classes which will be skipped when their sub-classes found them.
base_classes_to_skip =# classes that create no constructor
# Set is special and we will use a hand-written constructor
abstract_classes = Helper AbstractCheckButton# Determining whether to use script object(js object) to control the lifecycle of native(cpp) object or the other way around. Supported values are 'yes' or 'no'.
script_control_cpp = no
prefix : 输出的前缀,会连接到tolua类型的函数名之前,例如 int cocos2dx_ui_xxx(lua_State* tolua_S);
target_namespace : 所有生成的类都属于这个表下 例如这个里面的 ccui.xxx
cpp_namespace : C++中的命名空间,自动生成的代码中会加上这个命名空间
android_headers : 这是安卓编译的一些指令,不需要修改,照抄就行
android_flags : 这是安卓编译的一些指令,不需要修改,照抄就行
clang_headers : 这是Clang的编译指令,不需要修改,照抄就行
cocos_flags : 这是Clang的编译指令,不需要修改,照抄就行
cocos_headers: cocos的头文件搜索目录
cocos_flags:照抄
cxxgenerator_headers : 不管
extra_arguments : 所有文件的搜索路径
headers:这是需要解析的头文件 会从这个文件中识别所有include的头文件,并解析其中的类, 可以填多个文件 比如自己的文件的需要引用的头文件
classes:需要生成哪些类的接口,这里支持正则表达式
skip:跳过哪些类和函数,支持正则表达式,可以借鉴cocos的配置文件
rename_functions:重命名函数
rename_classes:重命名类
abstract_classes:哪些类没有创建构造函数,需要手动重写构造函数
script_control_cpp:不管,一般都是no
可以生成自己的一个配置文件。
然后看下同样目录下的genbindings.py文件
需要将自己的ini文件填入进去。
然后运行这个python文件,便会自动生成。
一般如下
手写调用
有人会说,为啥cocos里面还有类似
首先,工具脚本不是万能的,有些无法导出,例如Lambda表达式,例如某些回调函数。
void GameNetWorkManager::resumSocketByIp(const int socketMark, const std::string &serverIp, const int serverPort, const std::function<void(bool)> &callback)
{callback(m_socketManage->resumSocket(socketMark,serverIp,serverPort));
}
比如这个方法。
无法生成,我们就需要手写
int tolua_GameNetWorkManager_resumSocket(lua_State* tolua_S)
{int argc = 0;GameNetWorkManager* cobj = nullptr;bool ok = true;#if COCOS2D_DEBUG >= 1tolua_Error tolua_err;
#endif#if COCOS2D_DEBUG >= 1if (!tolua_isusertype(tolua_S, 1, "GameNetWorkManager", 0, &tolua_err)) goto tolua_lerror;
#endifcobj = (GameNetWorkManager*)tolua_tousertype(tolua_S, 1, 0);#if COCOS2D_DEBUG >= 1if (!cobj){tolua_error(tolua_S, "invalid 'cobj' in function 'tolua_GameNetWorkManager_resumSocket'", nullptr);return 0;}
#endifargc = lua_gettop(tolua_S) - 1;if (argc == 2){int arg1;ok &= luaval_to_int32(tolua_S, 2, (int *)&arg1, "GameNetWorkManager:resumSocket");
#if COCOS2D_DEBUG >= 1if (!toluafix_isfunction(tolua_S, 3, "LUA_FUNCTION", 0, &tolua_err)){goto tolua_lerror;}
#endifint handler = (toluafix_ref_function(tolua_S, 3, 0));if (!ok){tolua_error(tolua_S, "invalid arguments in function 'tolua_GameNetWorkManager_resumSocket'", nullptr);return 0;}cobj->resumSocket(arg1,[=](bool resum){LuaStack* stack = LuaEngine::getInstance()->getLuaStack();stack->pushBoolean(resum);stack->executeFunctionByHandler(handler, 1);stack->clean();return 0;});ScriptHandlerMgr::getInstance()->addCustomHandler((void*)cobj, handler);lua_settop(tolua_S, 1);return 1;}else if (argc == 4){int arg1;std::string arg2;int arg3;ok &= luaval_to_int32(tolua_S, 2, (int *)&arg1, "GameNetWorkManager:resumSocket");ok &= luaval_to_std_string(tolua_S, 3, &arg2, "GameNetWorkManager:resumSocket");ok &= luaval_to_int32(tolua_S, 4, (int *)&arg3, "GameNetWorkManager:resumSocket");
#if COCOS2D_DEBUG >= 1if (!toluafix_isfunction(tolua_S, 5, "LUA_FUNCTION", 0, &tolua_err)){goto tolua_lerror;}
#endifint handler = (toluafix_ref_function(tolua_S, 5, 0));if (!ok){tolua_error(tolua_S, "invalid arguments in function 'tolua_GameNetWorkManager_resumSocket'", nullptr);return 0;}cobj->resumSocketByIp(arg1, arg2, arg3, [=](bool resum){LuaStack* stack = LuaEngine::getInstance()->getLuaStack();stack->pushBoolean(resum);stack->executeFunctionByHandler(handler, 1);stack->clean();return 0;});ScriptHandlerMgr::getInstance()->addCustomHandler((void*)cobj, handler);lua_settop(tolua_S, 1);return 1;}return 0;
#if COCOS2D_DEBUG >= 1tolua_lerror:tolua_error(tolua_S, "#ferror in function 'tolua_GameNetWorkManager_resumSocket'.", &tolua_err);
#endifreturn 0;
}
最后
如果需要深入了解Lua,强烈建议阅读《lua设计与实现》。
相关文章:
# Lua与C++交互(二)———— 交互
C 调用lua 基础调用 再来温习一下 myName “beauty girl” C想要获取myName的值,根据规则,它需要把myName压入栈中,这样lua就能看到;lua从堆栈中获取myName的值,此时栈顶为空;lua拿着myName去全局表中查…...
机器人焊接生产线参数监控系统理解需求
机器人焊接生产线参数监控系统是以参数来反映系统状态并以直观的方式表现 出来,及时了解被监视对象的状态和状态的变化情况。其主要目标是为了达到减少 生产线的处理时间,降低故障率,缩短故障排除时间,从而提高生产线的生产效率 …...
前端基础(ES6 模块化)
目录 前言 复习 ES6 模块化导出导入 解构赋值 导入js文件 export default 全局注册 局部注册 前言 前面学习了js,引入方式使用的是<script s"XXX.js">,今天来学习引入文件的其他方式,使用ES6 模块化编程,…...
第七章,文章界面
7.1添加个人专栏 <template><div class="blog-container"><div class="blog-pages"><!-- 用于渲染『文章列表』和『文章内容』 --><router-view/><div class="col-md-3 main-col pull-left"><div cla…...
HJ102 字符统计
描述 输入一个只包含小写英文字母和数字的字符串,按照不同字符统计个数由多到少输出统计结果,如果统计的个数相同,则按照ASCII码由小到大排序输出。 数据范围:字符串长度满足 1≤len(str)≤1000 1≤len(str)≤1000 输入描述&a…...
Maven聚合项目(微服务项目)创建流程,以及pom详解
1、首先创建springboot项目作为父项目 只留下pom.xml 文件,删除src目录及其他无用文件 2、创建子项目 子项目可以是maven项目,也可以是springboot项目 3、父子项目关联 4、父项目中依赖管理 <?xml version"1.0" encoding"UTF-8&qu…...
Android OkHttp 源码浅析一
演进之路:原生Android框架不好用 ---- HttpUrlConnect 和 Apache HTTPClient 第一版 底层使用HTTPURLConnect 第二版 Square构建 从Android4.4开始 基本使用: val okhttp OkHttpClient()val request Request.Builder().url("http://www.baidu.com").buil…...
【Redis】——Redis基础的数据结构以及应用场景
什么是redis数据库 Redis 是一种基于内存的数据库,对数据的读写操作都是在内存中完成,因此读写速度非常快,常用于缓存,消息队列、分布式锁等场景。,Redis 还支持 事务 、持久化、Lua 脚本、多种集群方案(主…...
SpringBoot+WebSocket搭建多人在线聊天环境
一、WebSocket是什么? WebSocket是在单个TCP连接上进行全双工通信的协议,可以在服务器和客户端之间建立双向通信通道。 WebSocket 首先与服务器建立常规 HTTP 连接,然后通过发送Upgrade标头将其升级为双向 WebSocket 连接。 WebSocket使得…...
推荐适用于不同规模企业的会计软件:选择最适合您企业的解决方案
高效的会计软件不仅可以协助企业进行财务管理,做出科学的财务决策,还可以对企业数字化转型提供助力。不同规模的企业需要根据其特定需求选择适合的会计软件。那么有什么适合不同规模企业的会计软件推荐吗? 小型企业的选择 对于小型企业而言&…...
Apache Zookeeper架构和选举机制
ZooKeeper是一个开源的分布式协调服务,旨在解决分布式系统中的一致性、配置管理、领导者选举等问题。它由Apache软件基金会维护,是Hadoop生态系统的一部分,被广泛用于构建高可用、可靠和具有一致性的分布式应用程序和服务。 ZooKeeper提供了一个层次化的命名空间,类似于文…...
车联网TCU USB的配置和使用
1 usb_composition命令 # cat /sbin/usb/target # cd /sys/class/android_usb/android0 # cat functions console shows that QCOM’s default configuration Usage: usb_composition [Pid] [HSIC] [PERSISTENT] [IMMEDIATE] [FROM_ADBD] usb_composition 9025 n y y Then this…...
Linux系统USB摄像头测试程序(三)_视频预览
这是在linux上usb摄像头视频预览程序,此程序用到了ffmpeg、sdl2、gtk3组件,程序编译之前应先安装他们。 #include <sys/ioctl.h> #include <sys/stat.h> #include <sys/types.h> #include <fcntl.h> #include <zconf.h> …...
目标检测任务数据集的数据增强中,图像水平翻转和xml标注文件坐标调整
需求: 数据集的数据增强中,有时需要用到图像水平翻转的操作,图像水平翻转后,对应的xml标注文件也需要做坐标的调整。 解决方法: 使用pythonopencvimport xml.etree.ElementTree对图像水平翻转和xml标注…...
系统架构的演变
随着互联网的发展,网站应用的规模不断扩大,常规的应用架构已无法应对,分布式服务架构以及微服务架构势在必行,必需一个治理系统确保架构有条不紊的演进。 单体应用架构 Web应用程序发展的早期,大部分web工程(包含前端…...
IDC发布《亚太决策支持型分析数据平台评估》报告,亚马逊云科技位列“领导者”类别
日前,领先的IT市场研究和咨询公司IDC发布《2023年亚太地区(不含日本)决策支持型分析数据平台供应商评估》1报告,亚马逊云科技位列“领导者”类别。IDC认为,亚马逊云科技在解决方案的协同性、敏捷性、完整性、及时性、经…...
C#之OpenFileDialog创建和管理文件选择对话框
OpenFileDialog 是用于图形用户界面(GUI)编程的一个类,它用于显示一个对话框,允许用户选择要打开的文件。在需要用户加载或打开文件的应用程序中(如文本编辑器、图像查看器或文档处理器),这是一…...
Java中使用MongoTemplate 简单操作MongoDB
Autowired private MongoTemplate mongoTemplate; User:封装的对象 插入:mongoTemplate.insert(user); 根据id查询:mongoTemplate.findById(id, User.class); 查询所有:mongoTemplate.findAll(User.class); 条件查询&#…...
[Mac软件]Pixelmator Pro 3.3.12 专业图像编辑中文版
Pixelmator Pro是专为Mac设计的功能强大,美观且易于使用的图像编辑器。借助广泛的专业级无损图像编辑工具,Pixelmator Pro可使您发挥出最佳的照片效果,创建华丽的构图和设计,绘制,绘画,应用令人惊叹的效果&…...
吴恩达 GPT Prompting 课程
Prompting Guidelines 目录 Guidelines for PromptingPrompting Principles Principle 1: Write clear and specific instructions1.1: Use delimiters to clearly indicate distinct parts of the input1.2: Ask for a structured output1.3: Ask the model to check whether …...
gpt3.5写MATLAB代码剪辑视频,使之保留画面ROI区域
% 输入和输出文件名 inputVideoFile input_video.mp4; outputVideoFile output_video.mp4;% 创建 VideoReader 和 VideoWriter 对象 videoReader VideoReader(inputVideoFile); outputVideo VideoWriter(outputVideoFile, MPEG-4); outputVideo.FrameRate videoReader.Fra…...
设计模式二十一:状态模式(State Pattern)
一个对象的内部状态发生变化时,允许对象改变其行为。这种模式使得一个对象看起来好像在运行时改变了它的类,主要目的是将状态的行为从主要对象中分离出来,使得主要对象不必包含所有状态的逻辑,而是将每个状态的行为封装在独立的类…...
【校招VIP】产品思维能力之产品设计
考点介绍: 对于产品经理来说最重要的就是产品思维,而拥有一个好的产品思维需要做到以下三点: 1、把握关键点的能力 2、出方案,协调资源,说服团队把资源倾斜到关键点上的能力 3、评估关键点进展程度的能力 『产品思维能…...
微信小程序卡片横向滚动竖图
滚动并不是使用swiper,该方式使用的是scroll-view实现 Swiper局限性太多了,对竖图并不合适 从左往右滚动图片示例 wxml代码: <view class"img-x" style"margin-top: 10px;"><view style"margin: 20rpx;…...
SpringBoot项目(支付宝整合)——springboot整合支付宝沙箱支付 从极简实现到IOC改进
目录 引出git代码仓库准备工作支付宝沙箱api内网穿透 [natapp.cn](https://natapp.cn/#download) springboot整合—极简实现版1.导包配置文件2.controller层代码3.进行支付流程4.支付成功回调 依赖注入的改进1.整体结构2.pom.xml文件依赖3.配置文件4.配置类,依赖注入…...
【AIGC】一款离线版的AI智能换脸工具V2.0分享(支持图片、视频、直播)
随着人工智能技术的爆发,AI不再局限于大语言模型,在图片处理方面也有非常大的进步,其中AI换脸也是大家一直比较感兴趣的,但这个技术的应用一直有很大的争议。 今天给大家分享一个开源你的AI换脸工具2.0,只需要一张所需…...
管理类联考——逻辑——真题篇——按知识分类——汇总篇——一、形式逻辑——选言——相容选言——或
文章目录 第五章 选言命题:相容选言-或;不相容选言-要么要么第一节 选言-相容选言-或-推结论-A或B为真,则非A→B,非B→A(否一则肯一)真题(2010-28)-选言-相容选言-或-推结论-(1)A或B为真,A为假:得B为真(否一则肯一);真题(2012-29)-选言-相容选言-或-推结论-(1)…...
Git如何操作本地分支仓库?
基本使用TortoiseGit 操作本地仓库(分支) 分支的概念 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着你可以把你的工作从开发主线上分离开来,避免影响开发主线。多线程开发,可以同时开启多个任务的开发,多个任务之间互不影响。 为何要…...
WPS右键新建没有docx pptx xlsx 修复
解决wps右键没有新建文档的问题 右键没有新建PPT和Excel 1 wps自带的修复直接修复没有用 以上不管咋修复都没用 2 先编辑注册表 找到 HKEY_CLASSES_ROOT CTRLF搜文件扩展名 pptx docx xlsx 新建字符串 三种扩展名都一样操作 注册表编辑之后再次使用wps修复 注册组件&am…...
【巧学C++之西游篇】No.2 --- C++闹天宫,带着“重载“和“引用“
文章目录 前言🌟一、函数重载🌏1.1.函数重载概念🌏1.2.C支持函数重载的原理 -- 名字修饰 🌟二、引用🌏2.1.引用的概念🌏2.2.引用特性🌏2.3.常引用🌏2.4.使用场景🌏2.5.传…...
网站建设策划 流程/淘宝店铺推广方法
作为团队的领导,要建立领导力,发挥影响力的关键,我认为是和下属建立真正的信任关系 1,要有服务意识。你的团队的目标要通过下属的工作和努力来完成,这个过程应该是你服务于下属,支持他工作的过程 2…...
临沂网站推广排名/免费推广链接
1.onCreate( )方法是android应用程序中最常见的方法之一:翻译过来就是说,onCreate()函数是在activity初始化的时候调用的,通常情况下,我们需要在onCreate()中调用setContentView(int)函数填充屏幕的UI,一般通过findVie…...
重庆网站制作特点优势/网站营销策略
我想在Java中使用xz压缩。使用xz 1.5压缩库,commons io 2.4库和commons压缩1.8.1库。我试图运行下面的代码给我非常不一致的结果。文本超过70%,音频和视频文件低于0.1%(1-compressed/original * 100)。我在每次压缩之前使用tarbal…...
怎么用webstorm做网站/没经验可以做电商运营吗
python标记语句块使用方法,python语言和其它的编程语言有着显著的区别,那就是python对格式的要求非常苛刻,好处就是书写上简易命令,写错了就会自动报错,但实际上标记语块非常容易。 这里打开编辑器,新建一个…...
网站如何做淘宝推广/荆州网站seo
摘要 按照产品在电磁兼容设计时所采取的各项措施的重要性为先后,分为若干层次进行设计,并加以综合分析进行适当调整直到完善,这就是本文提出的” 电磁兼容分层与综合设计法”。可以做到电磁兼容试验一次成功。 人们在研发新产品时࿰…...
惠州住房和建设局网站/亚马逊市场营销案例分析
单片机的ADC接口属于模数转换接口,将外部的模拟量信号转化为数字信号,单片机属于数字器件,需将模拟信号转化为数字信号才能够为单片机处理。目前市场的很多单片机都自带ADC转换接口,若无ADC转换接口,可以使用ADC数模转…...