欢迎访问 Lu程序设计
C/C++模块化编译Lu脚本函数
1 说明
要演示本文的例子,你必须下载Lu64脚本系统。本文的例子需要lu64.dll、lu64.lib、C格式的头文件lu64.h,相信你会找到并正确使用这几个文件。
用C/C++编译器创建一个控制台应用程序,复制本文的例子代码直接编译运行即可。
2 Lu脚本的模块化编译功能
Lu支持表达式(函数)的模块化编译。参考Lu编程指南中的函数LuCom,该函数有一个参数nModule,要求给该表达式指定一个模块号,模块号用整数进行标识。如果用模块加锁函数LockModule对一个模块号进行了加锁,则编译表达式时必须提供加锁函数 ,本文暂不考虑加锁函数LockModule。
在Lu中,一个模块由一个或多个表达式组成。模块用一个整数标识,整数可正可负,只要绝对值相等,就属于同一个模块。一般用正整数表示该模块名。模块共有两类,即主模块(0#模块)和普通模块(其他标号的模块)。
同一模块中,模块号为负的表达式称私有表达式,只能被本模块的表达式所访问(即调用),在其他模块中是不可见的;模块号为正的表达式称公有表达式或全局表达式,能被任何一个表达式所访问。主模块(0#模块)中的表达式都是私有表达式。任何一个表达式,既可以访问本模块中的表达式,也可以访问其他模块中的全局表达式,如果本模块中的一个私有表达式与其他模块的一个全局表达式重名,将优先调用本模块中的私有表达式。
由以上规定可以看出,主模块可以访问本模块中的表达式,也可以访问其他模块中的全局表达式。因此,主模块常常用在主程序中。
3 代码
#include <stdio.h> #include "lu64.h" #pragma comment( lib, "lu64.lib" ) void main(void) { void *hFor; //存放表达式(脚本函数)句柄 luINT nPara; //存放表达式的自变量个数 LuData *pPara; //存放输入自变量的数组指针 LuData Val; //存放表达式的值 luINT ErrBegin,ErrEnd; //表达式编译出错的初始位置和结束位置 int ErrCode; //错误代码 wchar_t M1For1[]=L"aa(x,y) = x+y"; //将编译为模块1的私有函数,注意模块2中有同名私有函数 wchar_t M1For2[]=L"m1(x,y) = x+y+aa(x,y)"; //将编译为模块1的全局函数,函数名全局唯一 wchar_t M2For1[]=L"aa(x,y) = x-y"; //将编译为模块2的私有函数,注意模块1中有同名私有函数 wchar_t M2For2[]=L"m2(x,y) = x-y+aa(x,y)"; //将编译为模块2的全局函数,函数名全局唯一 wchar_t M0For1[]=L"aa(x,y : a,b) = a=x,b=y, x=m1(a,b), y=m2(a,b), x+y"; //将编译为模块0的私有函数,只能是私有函数,仍特意命名为aa if(!InitLu()) return; //初始化Lu //编译表达式M1For1、M1For2、M2For1、M2For2,我们不会直接调用它们,故所有返回参数均忽略。由于表达式是正确的,为了简单,ErrCode也被忽略。 ErrCode=LuCom(M1For1,-1,0,0,&hFor,&nPara,&pPara,&ErrBegin,&ErrEnd); //编译为模块1的私有函数 ErrCode=LuCom(M1For2, 1,0,0,&hFor,&nPara,&pPara,&ErrBegin,&ErrEnd); //编译为模块1的全局函数 ErrCode=LuCom(M2For1,-2,0,0,&hFor,&nPara,&pPara,&ErrBegin,&ErrEnd); //编译为模块2的私有函数 ErrCode=LuCom(M2For2, 2,0,0,&hFor,&nPara,&pPara,&ErrBegin,&ErrEnd); //编译为模块2的全局函数 //编译表达式M0For1,我们仅调用该函数。虽然表达式是正确的,但这里检查对ErrCode进行了检查。 ErrCode=LuCom(M0For1, 0,0,0,&hFor,&nPara,&pPara,&ErrBegin,&ErrEnd); //编译为模块0的私有函数 if(ErrCode) { printf("表达式有错误!错误代码: %d \n",ErrCode); } else { pPara[0].BType=luStaData_int64; pPara[0].VType=luStaData_int64; pPara[0].x=1; //对M0For1的第1个参数赋值 pPara[1].BType=luStaData_int64; pPara[1].VType=luStaData_int64; pPara[1].x=2; //对M0For1的第2个参数赋值 Val=LuCal(hFor,pPara); //计算表达式的值,hFor即编译得到的表达式句柄 printf("表达式的值: %I64d,第1个返回参数的值: %I64d,第2个返回参数的值: %I64d \n",Val.x,pPara[0].x,pPara[1].x); } FreeLu(); //释放Lu }
运行结果:
表达式的值: 4,第1个返回参数的值: 6,第2个返回参数的值: -2
4 函数说明
本例用到了Lu的四个输出函数:初始化Lu的函数InitLu,释放Lu的函数FreeLu,编译表达式的函数LuCom和计算表达式的函数LuCal。从这里查看这些函数的说明:Lu编程指南。
5 难点分析
很少有脚本会提供模块化编译功能,但模块化编译的确很有用途,因为一个模块的私有函数和模块变量,除非该模块提供了接口(全局函数),在该模块之外是无法访问的。
实际上,C/C++程序只要获得了表达式的句柄,就可以使用函数LuCal来执行它,无论该表达式是模块私有的,还是全局的。故从C/C++程序来看,对自己编译的Lu表达式,讨论私有或者全局并没有意义。但一个C/C++程序有时并不自己编译Lu脚本(或者只编译部分脚本),例如实用程序OpenLu使用扩展库Mlu编译Lu脚本,这时它只能获得Lu脚本系统中的部分函数(全局函数,或者显示输出的私有函数)句柄,从而使私有数据得到了有效地保护。关于一个C/C++程序如何获取非自己编译的Lu脚本函数,以后会有专门的例子进行讨论。
6 其他
你可能注意到了,我的联系方式就在下面,如有不明之处或有什么建议,可随时与我进行联系。
版权所有© Lu程序设计
2002-2021,保留所有权利
E-mail: forcal@sina.com
QQ:630715621
最近更新:
2021年05月23日