Qt中纯C++项目发布为dll的方法(超详细步骤)

目录

  • 一般创建方法
  • 导出普通函数的方法&调用方法
  • 导出类及其成员函数的方法&调用方法

众所周知,我们可以将C++项目中的类以及函数导出,形成 .dll 文件,以供其他程序使用,下面将说明Qt环境下的使用方法。

首先创建共享库,步骤如下:

下一步会出现类对话框等等,不用管它,直接点击创建即可,稍后再将类都删了。

创建好以后你会发现有一个包含global的头文件,这个文件中定义了__declspec(dllexport)、__declspec(dllimport)等,也可以删掉(如果要按照下面的方法的话,删掉即可)。

接下来还有重要的一步,请在.pro文件中,加入

CONFIG += dll //即使你的代码中写成了 CONFIG += staticlib,也要改过来

接下来就开始我们具体的的创建方法吧!

按照导出dll的的操作划分,有两种模式:

  • 导出普通方法(导出后可静态调用,也可动态调用)

首先是头文件内容:

头文件:

在方法声明时,前面加上__declspec(dllimport),也可用#define定义,看代码:

//为了和将来用到的程序中公用一个头文件,创建dll时用到的是__declspec(dllexport),而使用dll时用到__declspec(dllimport),完全可以各自写一下
#define TESTDLLSHARED_EXPORT __declspec(dllexport)
#ifdef TESTDLLSHARED_EXPORT
#else
#define TESTDLLSHARED_EXPORT __declspec(dllimport)
#endif

//下面我要定义4个普通函数:
TESTDLLSHARED_EXPORT int test1();

TESTDLLSHARED_EXPORT int test2(void);

TESTDLLSHARED_EXPORT int test3(int a);

TESTDLLSHARED_EXPORT int test4(int a,int b);

 然后在对应的cpp源文件中实现test1、test2、test3、test4这几个方法(为了简单,我只输出了一句话):

.cpp源文件:

//记得加上上面的头文件
//记得加上iostream头文件,不然cout不能用

int test1()
{
    std::cout<<"test1"<<std::endl;
}

int test2(void)
{
    std::cout<<"test2"<<std::endl;
}

int test3(int a)
{
    std::cout<<"test3"<<std::endl;
}

int test4(int a,int b)
{
    std::cout<<"test4"<<std::endl;
}

接下来就可以创建了,创建成功后(创建失败请仔细检查,代码已验证过),你会在你的项目输出目录下找到一个和项目名称一致的.dll文件。

OK!接下来我们来使用这个dll:

使用dll时有两种调用方法,一种是静态调用,一种是动态调用。

  • 静态调用

首先,创建一个常规的C++项目,将上面生成.dll复制到你的项目输出目录中(也就是和.exe文件在一起);

接下来,打开常规C++项目中的.pro文件,加入详细的.dll文件地址,格式如下

LIBS += 项目输出路径\dll文件全称

如我的就是

LIBS += D:\Desktop\Go\C++learnProgram\Qt\build-test_dll-Qt-Release\release\HpTickDll.dll

接下来,将创建.dll文件时的头文件复制到当前项目路径下,并添加到项目中,注意:如果你没有按照我上面的#define条件定义方式,请重新写__declspec(dllimport)。

接下来,在.cpp文件中包含该头文件,就可以尽情地使用之前的函数了,如直接用test1()等等,就和平时编写一样的。

  • 动态调用

!!!特别注意:我们在创建时没有用到extern "C",也没有用到.def 文件保持函数名不变(尝试了很多次也不会用.def文件,会的欢迎留言),因此动态调用时函数名要改!因为编译器已经将函数名改了!

So,你一定会问我们怎么知道dll中的函数名变成啥了?不要着急,网上直接搜“.dll查看器”,遍地都是,下载下来后,选择我们刚才的.dll文件就可以看了,下面是我们的这几个函数test1、test2、test3、test4的新名称(你的可能和我的不一样哦):

请注意看红色框中的部分(不要管其他的,我的文件里面还有其他东西),这就是四个函数在.dll文件中的名称,我小小地猜测了一下,后面的v代表参数为void类型,i的个数代表int类型参数的个数,前面的字母就不太清楚了(注意:参数个数并未正确列出,不过我们只需要正确的函数名,不影响)。

OK!这就好办了!看步骤:

同样是创建一个常规C++项目,不同的是.pro文件中不用加“LIBS += 项目输出路径\dll文件全称”这句话了。

接下来在.cpp文件中写主代码(不用添加之前的头文件):

#include <iostream>
#include <windows.h>

int main()
{
    //首先定义函数指针,用来接收不同参数的函数
    typedef int (CALLBACK *Fucv)();
    typedef int (CALLBACK *Fuci)(int);
    typedef int (CALLBACK *Fucii)(int,int);

    //获得.dll文件的句柄,需要头文件windows.h的支持
    HINSTANCE hdll=LoadLibrary(L"testDll.dll");     //L指宽字符串,若不写L,则会出现错误,详情请自查

    //注意这里要用.dll文件中的函数名
    Fucv t1=(Fucv)GetProcAddress(hdll,"_Z5test1v");
    Fucv t2=(Fucv)GetProcAddress(hdll,"_Z5test2v");
    Fuci t3=(Fuci)GetProcAddress(hdll,"_Z5test3i");
    Fucii t4=(Fucii)GetProcAddress(hdll,"_Z5test4ii");

    //现在的t1就执行的test1的功能,以此类推
    t1();
    t2();
    t3(1);
    t4(1,2);

    FreeLibrary(hdll);

    return 0;
}

以上就是普通函数创建.dll和使用.dll的方法,看起来动态调用是不是很麻烦?但它有许多优点(请自查),而且据说这里的函数名可以利用.def文件实现不改变名称,省去好多麻烦,但是我尝试了各种方法,如在.pro中利用DEF_FILE添加.def文件还是不行55555,会的可以留言哦!

  • 导出类及其成员函数(导出后可静态调用。暂时不会动态调用,某些书上说类不支持动态调用,网上有说在类中写一个方法返回类对象,但是个人认为这种做法是不对的,因为此时肯定是用自己定义的函数指针去定义这个返回类对象方法,不可能成功【已验证】,当然或许还有更好的方法,有知道的欢迎交流哈)

对于类的话,创建时:在class的后面,类名的前面加上定义的__declspec(dllexport);使用时,换成__declspec(dllimport),也可以参照前面的#define条件定义法。举例:

#define HPTICKDLLSHARED_EXPORT __declspec(dllexport)
#ifdef HPTICKDLLSHARED_EXPORT
#else
#define HPTICKDLLSHARED_EXPORT __declspec(dllimport)
#endif

class HPTICKDLLSHARED_EXPORT HpTickDll  //我在这里定义了类HpTickDll
{   
public:
    int  Start();                       //注意成员函数之前不用加HPTICKDLLSHARED_EXPORT
    int  GetTime();

private:

    LARGE_INTEGER li;
    LONGLONG start, end, freq;
    int useTime;
};

静态调用的方法和普通函数一样,直接可以使用类及其成员函数(别忘了添加头文件),就不细说了。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏崔庆才的专栏

腾讯云上 Winpcap 网络编程四之主机通信

由于腾讯云上提供了Windows系统,所以我们这次Winpcap编程选用腾讯云主机实验,让大家简要了解两台云主机的通信方法以及实践过程。

59100
来自专栏JavaEdge

高性能队列——Disruptor总论1 背景2 Java内置队列3 ArrayBlockingQueue的问题4 Disruptor的设计方案代码样例性能等待策略Log4j 2应用场景

这里所说的队列是系统内部的内存队列,而不是Kafka这样的分布式队列 Disruptor特性限于3.3.4

30030
来自专栏MasiMaro 的技术博文

Vista 及后续版本的新线程池

在上一篇的博文中,说了下老版本的线程池,在Vista之后,微软重新设计了一套线程池机制,并引入一组新的线程池API,新版线程池相对于老版本的来说,它的可控性更高...

16930
来自专栏偏前端工程师的驿站

asp.net 解码gb2312下urlencode后的字符串

公司网站前期的网页用了gb2312保存用户数据,而我负责的部分用的是utf8,今天恰好要获取前期录入的数据于是毫无悬念地出现乱码问题,经过一番网上的搜索还是找不...

22950
来自专栏微信公众号:Java团长

从并发编程到分布式系统——如何处理海量数据(上)

在这里想写写自己在学习并发处理的学习思路,也会聊聊自己遇到的那些坑,以此为记,希望鞭策自己不断学习、永不放弃!

10110
来自专栏Golang语言社区

go语言实现http服务端与客户端

go语言的net/http包的使用非常的简单优雅 (1)服务端 [plain] view plain copy package main import ...

36070
来自专栏Python自动化测试

测试驱动之xml文件的处理

Xml是可扩展标记语言,关于xml的技术本人这里不在介绍,感兴趣的同学可以去w3c看看详细的资料,这里,我仅仅介绍的是如何获取xml文档结构中的...

15730
来自专栏SDNLAB

ODL应用开发之MD-SAL中级教程

1. 简介 本次我们从开始设计到最终完成一个应用的开发,主要设计datastore和RPC定义和实现。Opendaylight 开发使用了OSGi框架,OSGi...

79380
来自专栏从零开始学自动化测试

python笔记9-多线程Threading之阻塞(join)和守护线程(setDaemon)

前言 今天小编YOYO请xiaoming和xiaowang吃火锅,吃完火锅的时候会有以下三种场景: - 场景一:小编(主)先吃完了,xiaoming(客)和xi...

40260
来自专栏微服务生态

淘宝Tedis组件究竟是个啥(一)

淘宝的Tedis组件究竟是个啥呢?可能有一些朋友没有听过这个名字,有一些朋友会经常使用,那么今天我就来和大家深入分析一下,它的使用和原理。

12220

扫码关注云+社区

领取腾讯云代金券