Visual Studio编译时无法解析外部符号

1>HelloWin.obj : error LNK2019: 无法解析的外部符号 __imp__PlaySoundW@12,该符号在函数 "long __stdcall WndProc(struct HWND__ *,unsigned int,unsigned int,long)" (?WndProc@@YGJPAUHWND__@@IIJ@Z) 中被引用
1>D:\MyProgramTest\Hellowin\Debug\Hellowin.exe : fatal error LNK1120: 1 个无法解析的外部命令

解决办法:

(1)首先必须在开头引入头文件#include <mmSystem.h>
(2)在解决方案中>>>>右击选择>>>>属性>>>>配置属性> >>>链接器>>>>输入>>>>附加依赖项中,写入winmm.lib,方可。

**********************************************************************************************

/*------------------------------------------------------------
HELLOWIN.C -- Displays "Hello, Windows !" in client area
------------------------------------------------------------*/

#include <windows.h>
//#pragma comment(lib,"winmm.lib")

LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;

int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
PSTR szCmdLine, int iCmdShow)
{
static TCHAR szAppName[] = TEXT ("HelloWin") ;
HWND hwnd ;
MSG msg ;
WNDCLASS wndclass ;

wndclass.style = CS_HREDRAW | CS_VREDRAW ;
wndclass.lpfnWndProc = WndProc ;
wndclass.cbClsExtra = 0 ;
wndclass.cbWndExtra = 0 ;
wndclass.hInstance = hInstance ;
wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ;
wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ;
wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
wndclass.lpszMenuName = NULL ;
wndclass.lpszClassName = szAppName ;

if (!RegisterClass (&wndclass))
{
MessageBox (NULL, TEXT ("This program requires Windows NT!"), 
szAppName, MB_ICONERROR) ;
return 0 ;
}

hwnd = CreateWindow (szAppName, // window class name
TEXT ("The Hello Program"), // window caption
WS_OVERLAPPEDWINDOW, // window style
CW_USEDEFAULT, // initial x position
CW_USEDEFAULT, // initial y position
CW_USEDEFAULT, // initial x size
CW_USEDEFAULT, // initial y size
NULL, // parent window handle
NULL, // window menu handle
hInstance, // program instance handle
NULL) ; // creation parameters

ShowWindow (hwnd, iCmdShow) ;
UpdateWindow (hwnd) ;

while (GetMessage (&msg, NULL, 0, 0))
{
TranslateMessage (&msg) ;
DispatchMessage (&msg) ;
}
return msg.wParam ;
}

LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HDC hdc ;
PAINTSTRUCT ps ;
RECT rect ;

switch (message)
{
case WM_CREATE:
PlaySound (TEXT ("hellowin.wav"), NULL, SND_FILENAME | SND_ASYNC) ;
return 0 ;

case WM_PAINT:
hdc = BeginPaint (hwnd, &ps) ;

GetClientRect (hwnd, &rect) ;

DrawText (hdc, TEXT ("Hello, Windows !"), -1, &rect,
DT_SINGLELINE | DT_CENTER | DT_VCENTER) ;

EndPaint (hwnd, &ps) ;
return 0 ;

case WM_DESTROY:
PostQuitMessage (0) ;
return 0 ;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;
}

编译结果:
error LNK2019: 无法解析的外部符号 __imp__PlaySoundW@12,该符号在函数 "long __stdcall WndProc(struct HWND__ *,unsigned int,unsigned int,long)" (?WndProc@@YGJPAUHWND__@@IIJ@Z) 中被引用

fatal error LNK1120: 1 个无法解析的外部

解决方案:在#include<windows.h>后面补上#pragma comment(lib,"winmm.lib")

问题分析:出现这种情况就是项目在编译时找不到你所引用的库文件,或者你自己声明的虚函数没有定义,二者都会导致链接错误,找到这个函数或者类所在的库,连接的时候记得包含进来就可以了。在include行结束后,在用#pragma comment(lib, "xxxxx")把默认没有的库都加上。
要知道哪个库没有,编译出错后看看是哪个API无法LINK,然后查MSDN关于该API的描述,一般在最下面会有说明,比如是哪个版本WINDOWS开始支持该函数,头文件是哪个,include用哪个,库用哪个等等。

&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&
无法解析的外部符号“symbol”,该符号在函数“function”中被引用
 
在 function 中找到了未定义的外部符号 (symbol)。若要解决此错误,请提供符号定义或移除引用它的代码。 
在 Visual C++ .NET 2003 中,如果使用了 /clr 而未将 CRT 链接到可执行文件,将生成此错误。任何由编译器在未使用 /clr:initialAppDomain 时生成的对象代码都包含对 _check_commonlanguageruntime_version 函数的引用,该函数在 C 运行时库 (CRT) 中定义。如果应用程序在运行库的版本 1 上运行,该函数将会生成一个错误信息。当前编译器生成的代码与运行库的版本 1 不兼容。因此,如果在 Visual C++ .NET 2003 中编译时不使用 CRT,则应在代码中包含 _check_commonlanguageruntime_version 函数的定义。作为使用 _check_commonlanguageruntime_version 函数的替代方法,您可以与 nochkclr.obj 链接。nochkclr.obj 包含该函数的一个空版本,当您在运行库的版本 1 上运行应用程序时,nochkclr.obj 不生成错误信息。若要使用当前编译器版本生成应用程序以在运行库的以前版本上运行,应使用 /clr:InitialAppDomain。 

若要生成一个纯 MSIL 可执行文件(不与 CRT 链接),则必须在项目中定义该函数,而不能使用 nochkclr.obj(.obj 是本机代码)。有关可验证代码的更多信息,请参见产生可验证的 C++ 托管扩展组件。有关从托管 C++ 项目创建纯 MSIL 输出文件的更多信息,请参见将 C++ 托管扩展项目从混合模式转换成纯 IL。 

本主题的其余部分讨论 LNK2019 的其他原因。 

请看下面的示例: 

extern int i; 
extern void g(); 
void f() 

i++; 
g(); 

int main() 


如果在生成中包含的某个文件中没有定义 i 和 g,链接器将生成 LNK2019。可以添加这些定义,方法是将包含这些定义的源代码文件包括为编译的一部分。或者可以将包含这些定义的 .obj 或 .lib 文件传递给链接器。 

对于从早期版本升级到当前版本的 C++ 项目,如果定义了 __UNICODE 并且入口点为 WinMain,需要将入口点函数的名称更改为 _tWinMain 或 _tmain。 

导致 LNK2019 的常见问题有: 

符号声明包含拼写错误,以致于符号声明与符号定义不同。 
使用了一个函数,但其参数的类型或数量与函数定义不匹配。 
函数声明使用和函数定义使用中的调用约定(__cdecl、__stdcall 或 __fastcall)不同。 
符号定义在编译为 C 程序的文件中,而符号是在 C++ 文件中不带 extern "C" 修饰符声明的。在此情况下,请修改声明,例如不是使用: 
extern int i; 
extern void g(); 
而使用:
extern "C" int i; 
extern "C" void g(); 
同样,如果在将由 C 程序使用的 C++ 文件中定义符号,请在定义中使用 extern "C"。 

符号定义为静态,但稍后在文件外部被引用。 
没有定义静态类成员。例如,应单独定义下面类声明中的成员变量 si: 
#include <stdio.h> 
struct X { 
static int si; 
}; 

// int X::si = 0; // uncomment this line to resolve 

void main() 

X *px = new X[2]; 
printf("\n%d",px[0].si); // LNK2019 

也可能由于为 Visual Studio .NET 2003 进行的一致性工作生成此错误:模板友元和专用化。在 Visual Studio .NET 2003 中,必须定义声明新的非模板函数的友元声明。 

要使代码在 Visual C++ 的 Visual Studio .NET 2003 和 Visual Studio .NET 版本中均有效,请显式指定友元函数的模板参数列表。 
// LNK2019.cpp 
// LNK2019 expected 
template<class T> 
void f(T) 



template<class T> 
struct S 

friend void f(T); 
// Try the folowing line instead: 
// friend void f<T>(T); 
}; 

int main() 

S<int> s; 
f(1); // unresolved external 

/VERBOSE 链接器选项帮助您查看链接器引用的文件。DUMPBIN 实用工具的 /EXPORT 和 /SYMBOLS 选项还可以帮助您查看 dll 和对象/库文件中定义的符号。
**********************************************************************************************************
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值