制作vb dll的总结

前段时间,同学接了一个项目,在网页上显示autoCAD的图。在做项目的过程中,遇到一个棘手的问题。在使用vb语言开发的第三方插件中,有一个函数是地址传值的方式传参。想在js中调用该函数,但是,js不支持类似指针的概念。后来在网上百度,发现可以使用dll封装的方式解决该问题。即把这个ocx(插件)用Activex封装一层。 JavaScript –> ActiveX –> 第三方ocx。  做一个ActiveX来调用第三方ocx,给它来个引用传递就行了。详见:http://blog.sina.com.cn/s/blog_704d0c1c0100mft7.html

在制作该dll的过程中,遇到了很多问题。开发环境为vb。首先打开vb,选择新建AvtiveX dll。在出来的界面中编写代码。

dll是一个函数库,他不能独立运行,没有main函数。但是dll有一个dllMain函数,该该函数即为该dll的入口函数。如果缺少该入口函数,那么该dll在加载的时候即会报”找不到入口点

dllRegisterServer“的错误。

Public Function DllMain(ByVal hInst As Long, ByVal fdwReason As Long, ByVal lpvReserved As Long) As Boolean
Select Case fdwReason
Case DLL_PROCESS_DETACH
Case DLL_PROCESS_ATTACH
DllMain = True
Case DLL_THREAD_ATTACH
Case DLL_THREAD_DETACH
End Select
End Function

在添加了dllMain入口函数后,又出现了问题,即在调用dll中的函数时在js报“对象不支持此属性或方法的错误”。经过一番调试,找到了错误所在,DllMain函数没有处理线程加载的情况。修改DllMain函数

Public Function DllMain(ByVal hInst As Long, ByVal fdwReason As Long, ByVal lpvReserved As Long) As Boolean
Select Case fdwReason
Case DLL_PROCESS_DETACH
Case DLL_PROCESS_ATTACH
DllMain = True
Case DLL_THREAD_ATTACH

DllMain=True

Case DLL_THREAD_DETACH
End Select
End Function

至于dll文件的编译,参考:

转帖] VB创建能作为输出函数的DLL

DLL, 函数输出

QQ594363468

 

DLL文件(即Dynamic Link Library,动态链接库)作为系统的一个重要的组成部分,除了一些小程序外,我们几乎能在所有软件中看到DLL文件,而且如果使用VB写的一些收费软件有一个DLL代替关键函数,想破解都难了。因此,我们就有必要来学习如何编写DLL

对于很多初学VB的网友来说,VB是一门比较简单的语言,而MicroSoft公司在开发VB时,也增添了很多功能。所以,我个人觉得VB除了部分不足之外还是挺不错的,当然我指的不足是主要是VB运行库方面的问题。

对于很多VB初学者来说,都会有这么一个错误的认识——VB中创建的DLL只是COM组件,无法作为输出函数的DLL(即VB写的DLL是不完整的DLL)。然而,对于一些VB的高手来说,他们就知道应该如何写具有返回值的DLL。其实,国外早已经有人做出了在VB中制作标准DLLAdd-Ins了,而且在网上也有对于写VB创建能作为输出函数的DLL的相关文章。今天,我再来讲讲这个话题——VB创建能作为输出函数的DLL

 

对于VB编写DLL,按照我的经验,主要分为两步:一、“安内”,二、“攘外”

 

首先,“攘外必先安内”

我们要知道如何使DLL拥有一个能被调用的API函数。很简单,我不想多说,就是在Module里面写一个PublicFunction

 

接下来,我们就可以“攘外”了

 

那么,先来介绍一下编译技术

 

编译器的编译技术可以分为Native Compile(自然编译)与P-Code Compile(伪编译)两种。

    自然编译是编译器将高级语言转换为汇编代码,并经链接生成EXE程序的过程。

    伪编译是编译器将高级语言转换为某种编码后,将能解释、执行此编码的一段程序一同链接,生成EXE程序。

    伪代码P-Code,最早应该叫做Pascal-Code,其名称起源于一个Pascal编译器使用的“中间代码”编译技术。现在一般作为Pseudo-code(伪代码)或Packed-code(压缩代码)简称。

采用伪代码编译时,每个VB源文件(包括.frm(窗体)、.bas(标准模块)、.cls(类模块))经VB IDE编译后各自生成相应的.obj文件,交链接程序Link.exe生成伪编译的可执行文件(EXEDLLOCX等)。

    而用自然编译时,每个VB源文件由C2.exe编译生成汇编代码,生成相应的obj文件,再由Link.exe链接成为完整的可执行文件。

 

别看编译只有两步,其实,有大玄奥在里头。。。

MicroSoft公司,可以说是十分的狡猾,因为在默认的方式下,VB编译的两步使用到的命令行都是他们设定好的,所以,跳过了很多有用的部分,包括我们今天讲的输出函数部分。如果在LINK的时候添加EXPORT选项,实际上是可以输出函数的。但是,在VBLinkd的命令行中将这个选项部分跳过了。而且过分的是:VB在构造EXE后会将编译出来的OBJ文件删除,这样就无法手动通过Link来创建我们需要的DLL了。

 

根据前人的方法,加上自己修改,用一个比较龌龊的方法来变通

 

打开“VB6.0”,新建一个“标准EXE 工程”,把Form移除,添加一个Module

在模块里输入一下代码:

 

    Sub Main() 

        If Command = "" Then Exit Sub '命令行不为空 

        Dim intCMD As Integer 

        'Clipboard.Clear '清空剪贴板 

        'Clipboard.SetText Command '将命令行复制到剪贴板 

        '去掉 复制到剪贴板 LINK  速度会快些 

        intCMD = MsgBox("编译命令行如下:" & vbCrLf & Command & vbCrLf & "点击“是”继续编译,点击“否”编译DLL,点击“取消”中断编译", vbYesNoCancel + vbInformation, "编译") 

        Select Case intCMD 

            Case vbYes '选择“是” 继续编译 

                Shell "VBLINK " & Command, vbHide 

            Case vbNo '选择“否” 

                Dim FuncName As String, cmd As String, l As Integer 

                FuncName = InputBox("请输入 输出的函数名称(对个函数 请用英文半角分号“;”分开)", _ 

                "非法操作出错我不管") '模块内能编译出来的输出函数名 

                '修改命令行 

                If FuncName = "" Then Exit Sub 

                Do 

                    FuncName = Replace(FuncName, ";", " /EXPORT:") 

                Loop Until 0 = InStr(FuncName, ";") 

                l = InStr(Command, "vbaS") 

                cmd = Left(Command, l + 4) & "/EXPORT:" & FuncName & Right(Command, Len(Command) - l - 3) 

                Shell "VBLINK " & cmd, vbHide 

            Case vbCancel '中断编译 

                Exit Sub 

        End Select 

    End Sub 

'大家可能看出来了,  没错  就是在每个输出函数前加/EXPORT: 

 

    生成MyLink.exe,复制到VB的安装文件夹里,把安装文件夹的LINK.EXE改成VBLink.exe,把我们写的MyLink.exe改成Link.exe

 

    到这一步,你应该懂了吧。。。

 

    来做个示例

 

我新建了一个ActiveX DLL的工程,添加了一个Module,写了一个PublicFunction,如下

Public Function Add(ByVal A As Long) As Long

   Add = A + A

End Function

 

生成“E:\Math.dll

 

被截获的命令行(VB默认命令行)如下:

"E:\Class1.OBJ" "E:\Module1.OBJ" "E:\Math.OBJ" "C:\Program Files\Microsoft Visual Studio\VB98\VBAEXE6.LIB" /ENTRY:__vbaS /OUT:"E:\Math.dll" /BASE:0x11000000 /SUBSYSTEM:WINDOWS,4.0 /VERSION:1.0 /DLL /INCREMENTAL:NO /OPT:REF /MERGE:.rdata=.text /IGNORE:4078

 

修改后命令行(修改后的命令行)如下:

"E:\Class1.OBJ" "E:\Module1.OBJ" "E:\Math.OBJ" "C:\Program Files\Microsoft Visual Studio\VB98\VBAEXE6.LIB" /ENTRY:__vbaS /EXPORT:Add /OUT:"E:\Math.dll" /BASE:0x11000000 /SUBSYSTEM:WINDOWS,4.0 /VERSION:1.0 /DLL /INCREMENTAL:NO /OPT:REF /MERGE:.rdata=.text /IGNORE:4078 

 

大家对比两行命令可以发现在/ENTRY:_vbaS后面,我增加了/EXPORT:Add,这就是关键所在

 

现在,我又新建了一个EXE工程,加一个VB API声明,代码部分:

 

Private Declare Function Add Lib "E:\Math.dll" (ByVal A As Long) As Long 'API声明

 

Private Sub Form_Load()

   MsgBox Add(2)

End Sub

 

看到效果没,成功了吧

 

至于String型么,投递时StrConv(String,vbUnicode )返回时StrConv(String,vbFromUnicode )

 

学了这个,大家不知道有没有什么心得呢,当然这个要看个人悟性的问题。师父领进门,修行在个人嘛,何况还是被我这种半调子的菜鸟教呢

 

这个...天好蓝...水好清...外挂...Winsock...API...DLL...监视API的值...修改...嘿嘿...我什么都没说

 

如果遇到不懂的地方...就来问我啊...QQ594363468

 

 

 

 

 

 

 

dll的注册:在cmd中执行Regsvr32 *.dll

注册之后即可使用。

查看ProggId 的方法:

在网上下载一个叫OLEVIEW的软件,在其中查找你的dll注册信息。一般来讲,你的dll应该在Automation  Objects中,名字和你做dll时的工程名一样

 

原文地址:https://www.cnblogs.com/elnino/p/2858635.html