QT error: LNK2019: 无法解析的外部符号 __imp__SendMessageW@ 类型的错误 以及 pragma comment 的作用(用法)

50 篇文章 2 订阅

使用QT Creator 开发一个通讯项目时候,需要使用windows系统函数,比如:

#include <windows.h>
#include <WinUser.h>
...

::IsWindow(server)
::FindWindowW(nullptr,title);   
::SendMessage(server, WM_COPYDATA, reinterpret_cast<WPARAM>(sender), reinterpret_cast<LPARAM>(&cds));

用法什么的都没错,编译的时候就报错:

widget.obj:-1: error: LNK2019: 无法解析的外部符号 __imp__SendMessageW@16,该符号在函数 "private: void __thiscall Widget::on_pushButton_clicked(void)" (?on_pushButton_clicked@Widget@@AAEXXZ) 中被引用

widget.obj:-1: error: LNK2019: 无法解析的外部符号 __imp__IsWindow@4,该符号在函数 "private: void __thiscall Widget::on_pushButton_clicked(void)" (?on_pushButton_clicked@Widget@@AAEXXZ) 中被引用

widget.obj:-1: error: LNK2019: 无法解析的外部符号 __imp__FindWindowW@8,该符号在函数 "private: void __thiscall Widget::on_pushButton_clicked(void)" (?on_pushButton_clicked@Widget@@AAEXXZ) 中被引用

经过查询,解决办法如下,在使用这几个函数的类文件(.cpp)中,加入下面一行:

#pragma comment  (lib, "User32.lib")

另外,我们延伸一下,pragma comment 的作用和用法:其实,pragma comment 有很多种用法,我们先来说一下,我们上面用到的作用,简单的来讲,就是:链接到User32.lib。它与在工程里面写上链接User32.lib作用是一样的。用这种方法写的,别的代码(程序)引用我们的代码的时候,就不需要在项目中配置了。它直接告诉连接器,链接的时候要找User32.lib,就不用在linker里面设置指定这个lib了。


我们先说一下 #pragma comment: 总的来说:

#pragma comment 是#pragma 下的一个子命令,意义是使用注释的方式引入库或编译目录。对于#pragma 命令,我们还有一个常见的用法,就是设置文件编码,例如,给.cpp文件设置utf-8 格式,就可以这样:

#pragma execution_character_set("utf-8")

1. 格式  :#pragma comment (comment-type, ["commentstring"] )

  • comment-type是一个预定义的标识符,指定注释的类型,应该是compiler,exestr,lib,linker之一。
  • commentstring是一个提供为comment-type提供附加信息的字符串。

2.参数说明(comment-type):

  • compiler  : 

      放置编译器的版本或者名字到一个对象文件,该选项是被linker忽略的。

  • exestr :

        在以后的版本将被取消。一般是用不到的。

  • lib :

        放置一个库搜索记录到对象文件中,这个类型应该是和commentstring(指定你要Linker搜索的lib的名称和路径)这个库的名字放在Object文件的默认库搜索记录的后面,linker搜索这个库就像你在命令行输入这个命令一样。你可以在一个源文件中设置多个库记录,它们在object文件中的顺序和在源文件中的顺序一样。如果默认库和附加库的次序是需要区别的,使用Z编译开关是防止默认库放到object模块

  • linker

指定一个连接选项,这样就不用在命令行输入或者在开发环境中设置了。

只有下面的linker选项能被传给Linker。/DEFAULTLIB ,/EXPORT,/INCLUDE,/MANIFESTDEPENDENCY, /MERGE,/SECTION

(1) /DEFAULTLIB:library

选项将一个 library 添加到 LINK 在解析引用时搜索的库列表。用 /DEFAULTLIB指定的库在命令行上指定的库之后和 .obj 文件中指定的默认库之前被搜索。忽略所有默认库 (/NODEFAULTLIB) 选项重写 /DEFAULTLIB:library。如果在两者中指定了相同的 library 名称,忽略库 (/NODEFAULTLIB:library) 选项将重写 /DEFAULTLIB:library

(2)/EXPORT:entryname[,@ordinal[,NONAME]][,DATA]

使用该选项,可以从程序导出函数,以便其他程序可以调用该函数。也可以导出数据。通常在 DLL 中定义导出。entryname是调用程序要使用的函数或数据项的名称。ordinal 在导出表中指定范围在 1 至 65,535 的索引;如果没有指定 ordinal,则 LINK 将分配一个。NONAME关键字只将函数导出为序号,没有entryname

(3)/INCLUDE:symbol

选项通知链接器将指定的符号添加到符号表

若要指定多个符号,请在符号名称之间键入逗号(,)、分号(;)或空格。在命令行上,对每个符号指定一次 /INCLUDE:symbol。

链接器通过将包含符号定义的对象添加到程序来解析 symbol。该功能对于添包含不会链接到程序的库对象非常有用。用该选项指定符号将通过 /OPT:REF 重写该符号的移除。

  • 14
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 8
    评论
错误信息 "error LNK2019: 无法解析外部符号 __imp_clock" 表明在程序中使用了函数 __imp_clock,但编译器和链接器无法找到该函数的定义或实现。这种错误通常是由于缺失符号的定义、编译器搜索路径错误或编译器/链接器选项错误引起的。 解决这个错误的方法有以下几种: 1. 确保包含了正确的头文件:检查代码中是否包含了正确的头文件,特别是包含了声明 __imp_clock 的头文件。如果没有包含正确的头文件,编译器将无法找到函数的定义。 2. 检查编译器的搜索路径:编译器需要知道在哪里查找函数的定义。确保编译器的搜索路径设置正确,以便能够找到函数的定义。可以通过设置编译器的选项或环境变量来修改搜索路径。 3. 检查编译器/链接器选项:检查编译器和链接器的选项设置,确保没有设置错误的选项。有时候,错误的选项设置可能导致编译器无法找到函数的定义。 下面是一个示例代码,演示了如何使用函数 __imp_clock: ```c++ #include <iostream> #include <ctime> int main() { std::clock_t start = std::clock(); // 执行一些操作 std::clock_t end = std::clock(); double duration = (end - start) / (double) CLOCKS_PER_SEC; std::cout << "程序执行时间:" << duration << " 秒" << std::endl; return 0; } ``` 这段代码使用了函数 std::clock() 来计算程序的执行时间。确保在代码中包含了正确的头文件,并且编译器能够找到函数的定义。
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值