VC++动态链接库(DLL)编程(七)――读者反馈与答复

简介:
VC++ 动态链接库 (DLL) 编程(七)
――读者反馈与答复
作者:宋宝华  e-mail:21cnbao@21cn.com
 
1.关于文章的获取
 
许多读者发来e-mail询问本系列文章的相关事宜,如:
(1)     是否已出版?
(2)     哪里可以下载打包版?
(3)     哪里可以下载笔者的其它文章?
   还有一些读者对日前笔者在天极网发表的《C语言嵌入式系统编程修炼之道》非常喜爱,给予了热情洋溢的赞扬,询问笔者能否继续创作嵌入式编程方面的文章。
对于这些问题,统一作答如下:
1)本系列文章暂时尚未出版;
2)您可以在天极网或pconline软件频道下载笔者的多数拙作。另外,我也将不定期将这些文章上传到我的博客([url]http://blog.donews.com/21cnbao/[/url])。所有文章中的例程源代码均经过亲手调试,验证无误;
    3)就嵌入式系统开发,笔者将继续进行此方面的创作,新近将推出《基于嵌入式实时OS VxWorks的多任务程序设计》及《领悟:从Windows多线程到VxWorks的多任务》。
非常感谢读者朋友对这些文章的喜爱,在下将竭尽所能地为您提供更多的好文章。
 
2.关于DLL的疑问
 
    你好,看了你写的“VC++动态链接库(DLL)编程深入浅出”,特别有收获。    只是有个地方我老搞不明白,就是用DLL导出全局变量时,指定了.lib的路径(#pragma comment(lib,"dllTest.lib")),那么.dll的文件的路径呢,我尝试着把.dll文件移到别的地方程序就无法正常运行了,请问.dll在这里怎么指定。
希望您能在百忙中抽空给我解答一下,不胜感激!
                                        一位编程爱好者
回答:
Windows按下列顺序搜索DLL
1)当前进程的可执行模块所在的目录;
2)当前目录;
3Windows 系统目录,通过GetSystemDirectory 函数可获得此目录的路径;
4Windows 目录,通过GetWindowsDirectory 函数可获得此目录的路径;
5PATH 环境变量中列出的目录。
因此,隐式链接时,DLL文件的路径不需要指定也不能指定,系统指定按照15的步骤寻找DLL,但是对应的.lib文件却需要指定路径;如果使用Windows API函数LoadLibrary动态加载DLL,则可以指定DLL的路径。
 
你好,我是一位C++初学者,我在PCONLINE看了教学之后,受益不浅。我想问一下能否在DLL里使用多线程?MSDN上用#using <mscorlib.dll>这个指令之后实现了多线程,不过好象不支持DLL..
请问有什么办法支持制作多线程DLL??能否给一个源码来?
回答:
DLL中可以处理多线程,WIN32对于多线程的支持是操作系统本身提供的一种能力,并不在于用户编写的是哪一类程序。即便是一个控制台程序,我们都可以使用多线程:
#include <stdio.h>
#include <windows.h>
void ThreadFun(void)
{
      while(1)
      {
             printf( "this is new thread\n" );
             Sleep( 1000 );
      }    
}
int main()
{
      DWORD threadID;
      CreateThread( NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFun,
             NULL, 0, &threadID );
      while(1)
      {
             printf( "this is main thread\n" );
             Sleep( 1000 );
      }
}
观察程序运行的结果为在控制台窗口上交替输出this is main threadthis is new thread
我们来看下面的一个多线程DLL的例子。
    DLL程序提供一个接口函数SendInit,在此接口中启动发送线程SendThreadFunc,在这个线程的对应工作函数中我们使用原始套接字socket发送报文。参考微软出版的经典书籍《Windows核心编程》,我们发现,不宜在DLL被加载的时候(即进程绑定时)启动一个新的线程。
这个线程等待一个CEvent事件(用于线程间通信),应用程序调用DLL中的接口函数SendMsg( InterDataPkt sendData )可以释放此事件。下面是相关的源代码:
1 )发送报文线程入口函数
///////////////////////////////////////////////////////////////////////////
// 函数名: SendThreadFunc
// 函数功能:发送报文工作线程入口函数,使用 UDP 协议
////////////////////////////////////////////////////////////////////////////
DWORD WINAPI SendThreadFunc( LPVOID lpvThreadParm )
// 提示:对于线程函数应使用 WINAPI 声明, WINAPI 被宏定义为 __stdcall
{
      /*  创建 socket */
      sendSock = socket ( AF_INET, SOCK_DGRAM, 0 );
      if ( sendSock == INVALID_SOCKET )
      {
             AfxMessageBox ( "Socket 创建失败 " );
             closesocket ( recvSock );
      }
 
      /*  获得目标节点端口与地址  */
      struct sockaddr_in desAddr;     
      desAddr.sin_family=AF_INET;
      desAddr.sin_port=htons( DES_RECV_PORT );    // 目标节点接收端口
      desAddr.sin_addr.s_addr = inet_addr( DES_IP );
 
  /*  发送数据  */
   while(1)
      {
             WaitForSingleObject( hSendEvent, 0xffffffffL );// 无限等待事件发生
             ResetEvent( hSendEvent );
            
             sendto( sendSock, (char *)sendSockData.data, sendSockData.len,
                    0, (struct sockaddr*)&desAddr, sizeof(desAddr) );      
      }
     
      return -1;
}
2 MFC 规则 DLL InitInstance 函数
/////////////////////////////////////////////////////////////////////////////
// CMultiThreadDllApp initialization
BOOL CMultiThreadDllApp::InitInstance()
{
      if ( !AfxSocketInit() )  // 初始化 socket
      {
             AfxMessageBox( IDP_SOCKETS_INIT_FAILED );
             return FALSE;
      }
 
      return TRUE;
}
3 )启动发送线程
////////////////////////////////////////////////////////////////////////////////
// 函数名: SendInit
// 函数功能: DLL 提供给应用程序调用接口,用于启动发送线程
/////////////////////////////////////////////////////////////////////////////
void SendInit(void)
{
      hSendThread = CreateThread( NULL, 1000, SendThreadFunc,
                    this, 1, &uSendThreadID );
}
4 SendMsg 函数
////////////////////////////////////////////////////////////////////////////////
// 函数名: SendMsg
// 函数功能: DLL 提供给应用程序调用接口,用于发送报文
/////////////////////////////////////////////////////////////////////////////
extern "C" void WINAPI SendMsg( InterDataPkt sendData )
{
      sendSockData = sendData;
      SetEvent( hSendEvent );    // 释放发送事件
}
以上程序仅仅是一个简单的例子,其实在许多工程应用中,我们经常看到这样的处理方式。这个 DLL 对用户而言仅仅使一个简单的接口函数 SendMsg ,对调用它的应用程序屏蔽了多线程的技术细节。与之类似, MFC 提供的 CSocket 类在底层自己采用了多线程机制,所以使我们免去了对多线程的使用。
 
您好,看了您的DLL文章,发现导出函数可以直接用_declspec(dllexport)声明或在.def文件中定义,变量的导出也一样。我想知道类是否也可以在.def文件中导出?您的文章中只讲了在类前添加_declspec(dllexport)导出类的方法。请您指教!
回答:
一般我们不采用.def文件导出类,但是这并不意味着类不能用.def文件导出类。
使用Depends查看连载2的“导出类”例程生成的DLL,我们发现其导出了如图21的众多“怪”symbol,这些symbol都是经过编译器处理的。因此,为了以.def文件导出类,我们必须把这些“怪”symbol全部导出,实在是不划算啊!所以对于类,我们最好直接以_declspec(dllexport)导出。


21 导出类时导出的symbol
 
您好,看了您的DLL文章,知道怎么创建DLL了,但是面对一个具体的工程,我还是不知道究竟应该把什么做成DLL?您能给一些这方面的经验吗?
回答:
DLL一般用于软件模块中较固定、较通用的可以被复用的模块,这里有一个非常好的例子,就是豪杰超级解霸。梁肇新大师把处理视频和音频的算法模块专门做成了两个DLL,供超级解霸的用户界面GUI程序调用,实在是DLL设计的模范教程。所谓“万变不离其宗”,超级解霸的界面再cool,用到的还是那几个DLL!具体请参考《编程高手箴言》一书。
 
您好,您的DLL文章讲的都是Windows的,请问Linux操作系统上可以制作DLL吗?如果能,和Windows有什么不一样?谢谢!
回答:
    Linux操作系统中,也可以采用动态链接技术进行软件设计,但与WindowsDLL的创建和调用方式有些不同。
Linux操作系统中的共享对象技术(Shared Object)与Windows里的DLL相对应,但名称不一样,其共享对象文件以.so作为后缀。与Linux共享对象技术相关的一些函数如下:
(1)打开共享对象,函数原型:
//打开名为filename共享对象,并返回操作句柄;
void *dlopen (const char *filename, int flag);
 (2)取函数地址,函数原型:
//获得接口函数地址
void *dlsym(void *handle, char *symbol);
(3)关闭共享对象,函数原型:
//关闭指定句柄的共享对象
int dlclose (void *handle);
 (4)动态库错误函数,函数原型:
//共享对象操作函数执行失败时,返回出错信息
const char *dlerror(void);
从这里我们分明看到Windows API――LoadLibraryFreeLibraryGetProcAddress的影子!又一个“万变不离其宗”!
 
本系列文章的连载暂时告一段落,您可以继续给笔者发送emailmailto[email]21cnbao@21cn.com[/email])讨论DLL的编程问题。对于文中的错误和纰漏,也热诚欢迎您指正。



 本文转自 21cnbao 51CTO博客,原文链接:http://blog.51cto.com/21cnbao/120761,如需转载请自行联系原作者

相关文章
|
1月前
|
C++
C++ 语言异常处理实战:在编程潮流中坚守稳定,开启代码可靠之旅
【8月更文挑战第22天】C++的异常处理机制是确保程序稳定的关键特性。它允许程序在遇到错误时优雅地响应而非直接崩溃。通过`throw`抛出异常,并用`catch`捕获处理,可使程序控制流跳转至错误处理代码。例如,在进行除法运算或文件读取时,若发生除数为零或文件无法打开等错误,则可通过抛出异常并在调用处捕获来妥善处理这些情况。恰当使用异常处理能显著提升程序的健壮性和维护性。
48 2
|
1月前
|
算法 C语言 C++
C++语言学习指南:从新手到高手,一文带你领略系统编程的巅峰技艺!
【8月更文挑战第22天】C++由Bjarne Stroustrup于1985年创立,凭借卓越性能与灵活性,在系统编程、游戏开发等领域占据重要地位。它继承了C语言的高效性,并引入面向对象编程,使代码更模块化易管理。C++支持基本语法如变量声明与控制结构;通过`iostream`库实现输入输出;利用类与对象实现面向对象编程;提供模板增强代码复用性;具备异常处理机制确保程序健壮性;C++11引入现代化特性简化编程;标准模板库(STL)支持高效编程;多线程支持利用多核优势。虽然学习曲线陡峭,但掌握后可开启高性能编程大门。随着新标准如C++20的发展,C++持续演进,提供更多开发可能性。
46 0
|
3月前
|
编译器 C++ 开发者
C++一分钟之-C++20新特性:模块化编程
【6月更文挑战第27天】C++20引入模块化编程,缓解`#include`带来的编译时间长和头文件管理难题。模块由接口(`.cppm`)和实现(`.cpp`)组成,使用`import`导入。常见问题包括兼容性、设计不当、暴露私有细节和编译器支持。避免这些问题需分阶段迁移、合理设计、明确接口和关注编译器更新。示例展示了模块定义和使用,提升代码组织和维护性。随着编译器支持加强,模块化将成为C++标准的关键特性。
167 3
|
3天前
|
存储 算法 C++
C++提高篇:泛型编程和STL技术详解,探讨C++更深层的使用
文章详细探讨了C++中的泛型编程与STL技术,重点讲解了如何使用模板来创建通用的函数和类,以及模板在提高代码复用性和灵活性方面的作用。
13 2
C++提高篇:泛型编程和STL技术详解,探讨C++更深层的使用
|
28天前
|
Rust 安全 C++
系统编程的未来之战:Rust能否撼动C++的王座?
【8月更文挑战第31天】Rust与C++:现代系统编程的新选择。C++长期主导系统编程,但内存安全问题频发。Rust以安全性为核心,通过所有权和生命周期概念避免内存泄漏和野指针等问题。Rust在编译时确保内存安全,简化并发编程,其生态系统虽不及C++成熟,但发展迅速,为现代系统编程提供了新选择。未来有望看到更多Rust驱动的系统级应用。
43 1
|
14天前
|
程序员 C++ 容器
C++编程基础:命名空间、输入输出与默认参数
命名空间、输入输出和函数默认参数是C++编程中的基础概念。合理地使用这些特性能够使代码更加清晰、模块化和易于管理。理解并掌握这些基础知识,对于每一个C++程序员来说都是非常重要的。通过上述介绍和示例,希望能够帮助你更好地理解和运用这些C++的基础特性。
31 0
|
2月前
|
人工智能 JavaScript 开发工具
C++中的AI编程助手添加
今天为大家推荐一款适配了 Viusal Studio(本文使用),VS Code(本文使用),JetBrains系列以及Vim等多种编译器环境的插件 Fitten Code,Fitten Code 是由非十大模型驱动的 AI 编程助手,它可以自动生成代码,提升开发效率,帮您调试 Bug,节省您的时间,另外还可以对话聊天,解决您编程碰到的问题。 Fitten Code免费且支持 80 多种语言:Python、C++、Javascript、Typescript、Java等。
80 8
|
1月前
|
存储 编译器 C++
打破C++的神秘面纱:一步步带你走进面向未来的编程世界!
【8月更文挑战第22天】C++是一门功能强大但学习曲线陡峭的语言,提供高性能与底层控制。本文通过实例介绍C++基础语法,包括程序结构、数据类型、控制结构和函数。从简单的“Hello, C++!”程序开始,逐步探索变量声明、数据类型、循环与条件判断,以及函数定义与调用。这些核心概念为理解和编写C++程序打下坚实基础,引导你进入C++编程的世界。
33 0
|
3月前
|
存储 C++
【C++航海王:追寻罗杰的编程之路】一篇文章带你了解二叉搜索树
【C++航海王:追寻罗杰的编程之路】一篇文章带你了解二叉搜索树
31 1
|
3月前
|
存储 自然语言处理 C++
【C++航海王:追寻罗杰的编程之路】set|map|multiset|multimap简单介绍
【C++航海王:追寻罗杰的编程之路】set|map|multiset|multimap简单介绍
35 0
【C++航海王:追寻罗杰的编程之路】set|map|multiset|multimap简单介绍