用c++ 给易语言写支持库学习记录

 

废话我就不对说 直接开始

易语言官方下载的易语言安装路径下 有一个SDK文件夹

我们点进入cpp文件夹里面提供是c++的SDK

elib文件夹里就是sdk 我们新建一个win32项目

 

这里我用的是VS2015 

 

创建一个空的项目

包含易语言官方的SDK目录即可

 

需要注意的地方是 预定义处理里面 如果你需要编译成静态库的话 这两个是必不可少的

代码生成选择多线程(MT)

目标平台选择7.0 支持XP系统

如果需要生成fne后缀的(动态链接库)无法静态编译

预处理器需要这样设置 因为dll和lib的宏名不一样如果填写的不对 生成的就不对

主要实现 入口函数dllMain

main.cpp

 1 #include <windows.h>
 2 BOOL APIENTRY DllMain(HMODULE hModule,
 3     DWORD  ul_reason_for_call,
 4     LPVOID lpReserved
 5 )
 6 {
 7     switch (ul_reason_for_call)
 8     {
 9     case DLL_PROCESS_ATTACH:
10     case DLL_THREAD_ATTACH:
11     case DLL_THREAD_DETACH:
12     case DLL_PROCESS_DETACH:
13         break;
14     }
15     return TRUE;
16 }
main.cpp

功能实现 Elibfne.h 头文件

 1 #ifndef __ELIBFNE__
 2 #define __ELIBFNE__
 3 #include <windows.h>
 4 #include <tchar.h>
 5 #include <cassert>
 6 #if 1        //易语言SDK 必须带的
 7 #include <lib2.h>
 8 #include <lang.h>
 9 #include <fnshare.h>
10 #include <fnshare.cpp>
11 #endif
12 #define LIBARAYNAME "ELIB_MessageNotify"
13 EXTERN_C void AddFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf);
14 EXTERN_C void SubFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf);
15 EXTERN_C void MulFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf);
16 EXTERN_C void DivFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf);
17 EXTERN_C INT WINAPI  ELIB_MessageNotify(INT nMsg, DWORD dwParam1, DWORD dwParam2);
18 #ifndef __E_STATIC_LIB
19 #define LIB_GUID_STR "{1BF8530F-D7F9-4FE2-B94D-1906950FE19F}" /*GUID串: // {E0E54C4E-67DA-47C6-89BE-0DF9E8360AAA}, 必须使用guidgen.exe生成  数字签名*/
20 #define LIB_MajorVersion 1 /*库主版本号*/
21 #define LIB_MinorVersion 1 /*库次版本号*/
22 #define LIB_BuildNumber 20180810 /*构建版本号*/
23 #define LIB_SysMajorVer 3 /*系统主版本号*/
24 #define LIB_SysMinorVer 0 /*系统次版本号*/
25 #define LIB_KrnlLibMajorVer 3 /*核心库主版本号*/
26 #define LIB_KrnlLibMinorVer 0 /*核心库次版本号*/
27 #define LIB_NAME_STR "Elib_TEST" /*支持库名*/
28 #define LIB_DESCRIPTION_STR "\r\n简单的测试下封装支持库\r\n\
29 吹牛逼专用" /*功能描述*/
30 #define LIB_Author "    ┏━━━━━ Cool  ━━━━┓" /*作者名称*/
31 #define LIB_ZipCode "    ┣━━━━━━━━━━━━━━┫" /*邮政编码*/
32 #define LIB_Address "    ┃       QQ:252858527        ┃" /*通信地址*/
33 #define LIB_Phone    "    ┣━━━━━━━━━━━━━━┫" /*电话号码*/
34 #define LIB_Fax        "    ┃  JBec:http://dwz.cn/jb-ec  ┃" /*QQ号码*/
35 #define LIB_Email     "    ┣━━━━━━━━━━━━━━┫" /*电子邮箱*/
36 #define LIB_HomePage "    ┃            NULL            ┃" /*主页地址*/
37 #define LIB_Other    "    ┗━━━━━━━━━━━━━━┛" /*其它信息*/
38 #define LIB_TYPE_COUNT 1 /*命令分类数量*/
39 #define LIB_TYPE_STR "0000基本命令\0""\0" /*命令分类*/
40 #endif
41 
42 #endif
ElibFne.h

功能实现 Elibfne.cpp 源文件

  1 #include "ElibFne.h"
  2 
  3 /*
  4 @ 常量 定义导出 
  5 */
  6 
  7 #ifndef __E_STATIC_LIB
  8 
  9 LIB_CONST_INFO constStruct[] =
 10 {
 11     /*
 12     常量参数注释
 13     @ 中文名称
 14     @ 英文名称
 15     @ 常量的介绍
 16     @ 命令的学习难度级别
 17     @ 常量类型
 18     @ 文本类型存放的内容
 19     @ 布尔类型 浮点类型存放的内容
 20     */
 21     { _WT("ELB_VERSION"), _WT("取ELIB版本"), _WT("获取支持库版本号 返回文本型内容"), LVL_HIGH, CT_TEXT, _WT("1.0"), NULL },//文本常量
 22     {_WT("ELB_BOOL"),_WT("ELB布尔类型常量"),_WT("测试ELB布尔类型 常量返回布尔类型内容"),LVL_SIMPLE,CT_BOOL,NULL,1},
 23     { _WT("ELB_DOUBLE"),_WT("ELB小数类型常量"),_WT("测试ELB浮点类型 常量返回浮点类型内容"),LVL_SIMPLE,CT_NUM,NULL,1}
 24 
 25 };
 26 /*
 27 @ 导出 结构体类型
 28 */
 29 LIB_DATA_TYPE_ELEMENT structIntType[] =
 30 {
 31     /*{ 成员类型 ,数组成员 , 中文名称 ,英文名称 ,成员解释 ,枚举类型 ,默认值}*/
 32     { SDT_INT, NULL,_T("左边"), _T("left"), _T("left"), NULL, 0},
 33     { SDT_INT, NULL,_T("定边"), _T("top"), _T("top"), NULL, 0 },
 34     { SDT_INT, NULL,_T("右边"), _T("right"), _T("right"), NULL, 0 },
 35     { SDT_INT, NULL,_T("底边"), _T("bottom"), _T("bottom"), NULL, 0 }
 36 };
 37 LIB_DATA_TYPE_ELEMENT structTypes[] =
 38 {
 39     /*{ 成员类型 ,数组成员 , 中文名称 ,英文名称 ,成员解释 ,枚举类型 ,默认值}*/
 40     { SDT_TEXT, NULL,_T("数据内容"), _T("data"), _T("要发送的文本数据"), NULL, 0},
 41     { SDT_SHORT, NULL,_T("短整数类型"), _T("short"), _T("短整数类型测试"), NULL, 0 },
 42     { SDT_FLOAT, NULL,_T("小数类型测试"), _T("float"), _T("小数类型测试"), NULL, 0 },
 43     { SDT_BOOL, NULL,_T("布尔类型"), _T("Boolean"), _T("布尔类型测试"), NULL, 0 },
 44     { SDT_DATE_TIME, NULL,_T("日期时间类型测试"), _T("TimeDate"), _T("日期时间类型测试"), NULL, 0 },
 45     { SDT_SUB_PTR, NULL,_T("子程序指针类型测试"), _T("SUBPTR"), _T("子程序指针类型测试"), NULL, 0 },
 46     { SDT_BIN, NULL,_T("字节集类型测试"), _T("bin"), _T("字节集类型测试"), NULL, 0 }
 47 };
 48 
 49 static LIB_DATA_TYPE_INFO DataTypes[] =
 50 {
 51     /* { 中文名称, 英文名称, 数据描述, 索引数量, 命令索引, 对象状态, 图标索引, 事件数量, 事件指针, 属性数量, 属性指针, 界面指针, 元素数量, 元素指针 } */
 52     { _WT("矩形"), _WT("RECT"), _WT("窗口矩形数据结构"), 0, NULL, NULL, 0, 0, NULL, 0, NULL, NULL, sizeof(structIntType) / sizeof(structIntType[0]), structIntType },
 53     { _WT("复杂数据结构"), _WT("Struct"), _WT("测试多种数据类型"), 0, NULL, NULL, 0, 0, NULL, 0, NULL, NULL, sizeof(structTypes) / sizeof(structTypes[0]), structTypes }
 54 };
 55 #endif // !1
 56 
 57 /*
 58 函数的实现都需要定义在宏的外面以便静态和动态库都能使用,但ExecuteCommand,Commands则只需定义在宏的里面供动态库使用。
 59 pRetData 输出数据指针。当对应CMD_INFO中m_dtRetType为_SDT_NULL(即定义无返回值)时,pRetData无效;
 60 iArgCount 函数参数个数
 61 pArgInf 函数参数指针
 62 */
 63 EXTERN_C void AddFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf)
 64 {
 65     if (iArgCount==2)
 66     {
 67         pRetData->m_int = pArgInf[0].m_int + pArgInf[1].m_int;
 68     }
 69     return;
 70 }
 71 
 72 EXTERN_C void SubFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf)
 73 {
 74     if (iArgCount == 2)
 75     {
 76         pRetData->m_int = pArgInf[0].m_int - pArgInf[1].m_int;
 77     }
 78     return;
 79 }
 80 
 81 EXTERN_C void MulFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf)
 82 {
 83     if (iArgCount == 2)
 84     {
 85         pRetData->m_int = pArgInf[0].m_int * pArgInf[1].m_int;
 86     }
 87     return;
 88 }
 89 
 90 EXTERN_C void DivFunc(PMDATA_INF pRetData, INT iArgCount, PMDATA_INF pArgInf)
 91 {
 92     if (iArgCount == 2 && pArgInf[0].m_int!=0)
 93     {
 94         pRetData->m_int = pArgInf[0].m_int / pArgInf[1].m_int;
 95     }
 96     return;
 97 }
 98 
 99 
100 #ifndef __E_STATIC_LIB
101 PFN_EXECUTE_CMD ExecuteCommand[] =
102 {
103     // 所有需要库中调用的  函数  都列在这里,用逗号隔开
104     AddFunc,
105     SubFunc,
106     MulFunc,
107     DivFunc
108 };
109 
110 static const char* const CommandNames[] =
111 {
112     // 所有需要库中调用的  函数名  都写在这里,用逗号隔开
113     "AddFunc",
114     "SubFunc",
115     "MulFunc",
116     "DivFunc"
117 };
118 /*----------------定义支持库命令参数表-----------------------*/
119 ARG_INFO AddFunc_CommandArgs[] =
120 {
121     //函数参数数组定义写在这里,每个{}为一个参数的表述,用逗号隔开
122     /* { 参数名称, 参数描述, 图像索引, 图像数量, 参数类型(参见SDT_), 默认数值, 参数类别(参见AS_) } */
123     { _WT("arg1"), _WT("arg1"), 0, 0, SDT_INT, NULL, NULL },
124     { _WT("arg2"), _WT("arg2"), 0, 0, SDT_INT, NULL, NULL }
125 };
126 ARG_INFO SubFunc_CommandArgs[] =
127 {
128     //函数参数数组定义写在这里,每个{}为一个参数的表述,用逗号隔开
129     /* { 参数名称, 参数描述, 图像索引, 图像数量, 参数类型(参见SDT_), 默认数值, 参数类别(参见AS_) } */
130     { _WT("arg1"), _WT("arg1"), 0, 0, SDT_INT, NULL, NULL },
131     { _WT("arg2"), _WT("arg2"), 0, 0, SDT_INT, NULL, NULL }
132 };
133 ARG_INFO MulFunc_CommandArgs[] =
134 {
135     //函数参数数组定义写在这里,每个{}为一个参数的表述,用逗号隔开
136     /* { 参数名称, 参数描述, 图像索引, 图像数量, 参数类型(参见SDT_), 默认数值, 参数类别(参见AS_) } */
137     { _WT("arg1"), _WT("arg1"), 0, 0, SDT_INT, NULL, NULL },
138     { _WT("arg2"), _WT("arg2"), 0, 0, SDT_INT, NULL, NULL }
139 };
140 ARG_INFO DivFunc_CommandArgs[] =
141 {
142     //函数参数数组定义写在这里,每个{}为一个参数的表述,用逗号隔开
143     /* { 参数名称, 参数描述, 图像索引, 图像数量, 参数类型(参见SDT_), 默认数值, 参数类别(参见AS_) } */
144     { _WT("arg1"), _WT("arg1"), 0, 0, SDT_INT, NULL, NULL },
145     { _WT("arg2"), _WT("arg2"), 0, 0, SDT_INT, NULL, NULL }
146 };
147 
148 /*---------------声明要导出的支持库命令----------------------*/
149 static CMD_INFO Commands[] =
150 {
151     /* { 中文名称, 英文名称, 对象描述, 所属类别(-1是数据类型的方法), 命令状态(CT_), 返回类型(SDT_), 此值保留, 对象等级(LVL_), 图像索引, 图像数量, 参数个数, 参数信息 } */
152 
153     //{ _WT("GetAmongText"), _WT("取关键词间文本"), _WT("截取两个关键词之间的文本"), 1,NULL, SDT_TEXT, 0, LVL_SECONDARY,0,0, 5, fnGetAmongstr_CommandArgs } ,
154 
155     { _WT("ELIB_加法"), _WT("AddFunc"), _WT("加法运算 返回类型整数型"), 1, NULL, SDT_INT, 0, LVL_SIMPLE, 0, 0, 2, AddFunc_CommandArgs },
156     { _WT("ELIB_减法"), _WT("SubFunc"), _WT("减法运算 返回类型整数型"), 1, NULL, SDT_INT, 0, LVL_SIMPLE, 0, 0, 2, SubFunc_CommandArgs },
157     { _WT("ELIB_乘法"), _WT("MulFunc"), _WT("乘法运算 返回类型整数型"), 1, NULL, SDT_INT, 0, LVL_SIMPLE, 0, 0, 2, MulFunc_CommandArgs },
158     { _WT("ELIB_除法"), _WT("DivFunc"), _WT("除法运算 返回类型整数型"), 1, NULL, SDT_INT, 0, LVL_SIMPLE, 0, 0, 2, DivFunc_CommandArgs },
159 
160 
161 };
162 
163 #endif
164 /*-----------------支持库消息处理函数------------------*/
165 
166 EXTERN_C INT WINAPI ELIB_MessageNotify(INT nMsg, DWORD dwParam1, DWORD dwParam2)
167 {
168 #ifndef __E_STATIC_LIB
169     if (nMsg == NL_GET_CMD_FUNC_NAMES) // 返回所有命令实现函数的的函数名称数组(char*[]), 支持静态编译的动态库必须处理
170         return (INT)CommandNames;
171     else if (nMsg == NL_GET_NOTIFY_LIB_FUNC_NAME) // 返回处理系统通知的函数名称(PFN_NOTIFY_LIB函数名称), 支持静态编译的动态库必须处理
172         return (INT)LIBARAYNAME;
173     else if (nMsg == NL_GET_DEPENDENT_LIBS) return (INT)NULL;
174     // 返回静态库所依赖的其它静态库文件名列表(格式为\0分隔的文本,结尾两个\0), 支持静态编译的动态库必须处理
175     // kernel32.lib user32.lib gdi32.lib 等常用的系统库不需要放在此列表中
176     // 返回NULL或NR_ERR表示不指定依赖文件  
177 #endif
178     return ProcessNotifyLib(nMsg, dwParam1, dwParam2);
179 }
180 /*定义支持库基本信息*/
181 #ifndef __E_STATIC_LIB
182 static LIB_INFO LibInfo =
183 {
184     /* { 库格式号, GUID串号, 主版本号, 次版本号, 构建版本号, 系统主版本号, 系统次版本号, 核心库主版本号, 核心库次版本号,
185     支持库名, 支持库语言, 支持库描述, 支持库状态,
186     作者姓名, 邮政编码, 通信地址, 电话号码, 传真号码, 电子邮箱, 主页地址, 其它信息,
187     类型数量, 类型指针, 类别数量, 命令类别, 命令总数, 命令指针, 命令入口,
188     附加功能, 功能描述, 消息指针, 超级模板, 模板描述,
189     常量数量, 常量指针, 外部文件} */
190     LIB_FORMAT_VER,
191     _T(LIB_GUID_STR),
192     LIB_MajorVersion,
193     LIB_MinorVersion,
194     LIB_BuildNumber,
195     LIB_SysMajorVer,
196     LIB_SysMinorVer,
197     LIB_KrnlLibMajorVer,
198     LIB_KrnlLibMinorVer,
199     _T(LIB_NAME_STR),
200     __GBK_LANG_VER,
201     _WT(LIB_DESCRIPTION_STR),
202     _LIB_OS(__OS_WIN),
203     _WT(LIB_Author),
204     _WT(LIB_ZipCode),
205     _WT(LIB_Address),
206     _WT(LIB_Phone),
207     _WT(LIB_Fax),
208     _WT(LIB_Email),
209     _WT(LIB_HomePage),
210     _WT(LIB_Other),
211     sizeof(DataTypes) / sizeof(DataTypes[0]),
212     DataTypes,
213     LIB_TYPE_COUNT,
214     _WT(LIB_TYPE_STR),
215     sizeof(Commands) / sizeof(Commands[0]),
216     Commands,
217     ExecuteCommand,
218     NULL,
219     NULL,
220     ELIB_MessageNotify,
221     NULL,
222     NULL,
223     sizeof(constStruct) / sizeof(constStruct[0]),
224     constStruct,
225     NULL
226 };
227 PLIB_INFO WINAPI GetNewInf()
228 {
229     return (&LibInfo);
230 };
231 #endif
Elibfne.cpp

编译生成完以后分别把.fne和静态库文件复制到易语言lib目录和static_lib目录下即可

复制到易语言安装目录下的lib目前下

复制到易语言安装目录下的static_lib目前下

 

打开易语言选择你刚添加的支持库测试即可

 

 

 测试结果可用

 

 也可以下载 当前项目 链接: https://share.weiyun.com/51Z3Bxn (密码:bCDD)

转载于:https://www.cnblogs.com/guolongzheng/p/9449414.html

由于论坛缺少好用支持库且易本身C++版本过低导致很多C++新特性无法使用。本次利用C++11/14的标准库以及一些C++知名库(RapidJson,Curl)编支持库使用,以至于编程上不会太落后。 C++11/14标准库相对于微软类库而言微软无关,可实现跨平台。且其拥有很多高级语法,其效率及稳定性毋庸置疑。如果能直接用标准库完成坚决不要重复造轮子。     此次封装了线程、线程池、哈希表(UnOrderedMap)、读锁、互斥、定时器、计时器、Json、Curl等。其中Json封装于RapidJson,此库为C++最快的Json库,效率高于论坛其他工具几百倍。 Curl为知名Http库,很多公司及个人都是首选。     由于易语言5.6版本核心库其他版本不太一样导致静态编译过程中出现一些问题,所以请大家最好不要使用5.6版本。由于使用到了高版本C++库所以易语言自带的VC6编译器肯定不能编译, 在此本支持库使用了论坛的VS2014编译器,完美实现静态编译,如果你本身有这个编译器也请一定用本次配套的替换使用,否则会出现少库情况。     至于编译出来的程序能否支持XP,我想说这是肯定的,具体操作方法请参见压缩包里的说明。 备用地址 :链接:https://pan.baidu.com/s/1gY8Gm_kxMH1GOZiYaZeYZw   提取码:u1hs (里面包含编译器,支持库,例程) 文件较大,已包含 编辑所需的链接器
========== 1.易支持库机制 ========== !!!易支持库的库信息、命令信息等等均使用某个结构储存!!!易源码中的自定义数据类型、类模块、组件即为支持库中的库定义数据类型!!!易语言 通过 GetNewInf() 获取支持库信息!!!所以支持库必须导出这个函数 ========== 2.支持库命令调用 ========== !!!支持库的命令调用方式也比较特殊!!!每个命令需要三个参数:返回值指针,参数数目,参数信息指针!!!做个一个库定义数据类型的方法,每个方法的第一个参数为指向该对象的结构指针 ========== 3.简单说下易源码 ========== !!!易保存源码时没有保存各信息名称(比如命令名、数据类型名),而是保存了索引!!!所以支持库内的各信息有必要时可以修改!!!但不要改动TA们的顺序,这会导致前期版本的源码出现问题 ========== 4.模版说明 ========== !!!本模版整天来说思路清晰,代码连小白都能理解!!!模版多数的参数使用通用型,使得模版简便!!!模版内没有多说废话, 直接把关键的过了一遍, 足够支持库了!!!每条命令都演示了支持库的各种操作,能想到的都先了!!!暂时没组件,目前没什么时间了,玩着电脑也冷~!!!尽量抽时间将组件的开发也模块化!!!或者各位有心人士可参考 e\sdk\cpp\samples\HtmlView 一下 ========== 5.关于静态库 ========== !!!本人目前只做了动态库,不做静态库原因很多!!!本来打算做静态库的,不过考虑到某些原因:!!!静态库误报,动态库转到静态库时也会出现一些未知问题!!!个人也建议使用独立编译,5.3已经恢复!!!更建议用 黑月Cool编译!!!或者直接就是一个“懒”字!!!!各路有心人士可动手改一下!!!我也弄好了静态连接名的录入机制 ========== 6.为何选择支持库 ========== !!!其实用模块扩展程序也是足够了!!!选择做支持库还是安全的!!!而且支持库有许多模块做不到的功能 ========== 7.关于易功能函数 ========== !!!顾名思义, 就是易内部提供给支持库的函数!!!主要的用法呢还是得参考 lib2.h(位于 e\sdk\cpp\elib\lib.h)!!!虽说是一个C++头文件,但是语文好点、懂易语言就能理解!!!小学生的语文水平就差不多啦!!!我还是把一些常用的到了 类_易功能 这个类里面!!!各位同学可以看看、 、 ========== 【最后】 ========== !!!直说:本人纯属菜鸟一枚, 各路大神勿喷即可 ==========??【信息】??========== !!!模版:易支持库模版!!!作者:SalHe(Rabbit Group)!!!声明:大鸟勿喷即可!!!日期:2014年11月8日怎么编译我还是简单说一下,针对一下小白。编译的时候将文件名的后缀改为fne在放入易语言支持库目录就好了支持库一发布后就不要修改数字签名和支持库文件名了
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值