无法打开预编译头文件的解决方法及预编译头原理

无法打开预编译头文件:“Release\xxx.pch”: No such file or directory

       将【创建/使用编译头】改为“不使用编译头”,或者改成 “创建预编译头(/Yc)”, 然后编译就通过了。


         于是我也招搬了 “不使用编译头” ,结果出现下面的错误:

xxx.cpp
..\commen\xxx.cpp(2) : fatal error C1083: 无法打开包括文件:“stdafx.h”: No such file or directory

         原因是我的程序文件放了好几个目录,公共目录下自然没有“stdafx.h”,所以就编译错误了。然后我又改为“创建预编译头”,在Debug版本下,貌似对了。Release时,忘了怎么配置的,结果仍然提示  “无法打开预编译头文件 xx.pch”。

        按照网上的说法,我把stdafx.cpp单独编译了一下(在解决方案视图,右键文件->编译,则只对选中文件进行编译),然后再整体编译连接,编译通过了,但是链接又出问题了,提示如下:

warning C4727: 具有相同时间戳的名为 xxx.pch 的 PCH 已存在于yyy.obj 和 yyy.obj 中。使用第一个 PCH。
MyDlg2.obj : error LNK2005: ___@@_PchSym_@00@UdliphkzxvUexUszmarUKltKoyKrqKluKpmKqsKltKroKonKluKqoKoleBUivovzhvUnbwotOlyq@ 已经在 MyDlg.obj 中定义

         我晕!然后又改成 “使用预编译头(/Yu)”,当编译正要成功是,悲剧又来了:

.\MyDlg.cpp(4) : fatal error C1854: 无法覆盖在创建对象文件“xxx\release\mydlg.obj”的预编译头过程中形成的信息

        然后我就反复,将【创建/使用编译头】,在这几个选项中切换尝试,但错误始终存在。正在我无奈之极,突然搜到了MSDN中对错误C4727和C1854的解释:


http://msdn.microsoft.com/zh-cn/library/5a193hd8.aspx

错误 C1854

Visual Studio 2010
其他版本
无法覆盖在创建对象文件“filename”的预编译头过程中形成的信息

您对同一文件指定 /Yc(创建预编译头文件)选项后指定了 /Yu(使用预编译头文件)选项。 某些声明(如包括__declspecdllexport 的声明)使此操作无效。


http://msdn.microsoft.com/zh-cn/library/ms235540.aspx

编译器警告(等级 1)C4727

Visual Studio 2010
其他版本
在 obj_file_1 和 obj_file_2 中找到具有相同时间戳的名为 pch_file 的 PCH。使用第一个 PCH。

如果用 /Yc 编译多个 compiland,并且其中的编译器能够用相同的 .pch 时间戳标记所有的 .obj 文件,将发生 C4727。

要消除此警告,请用 /Yc /c 编译一个源文件(创建 pch),并用 /Yu /c 分别编译其他的源文件(使用 pch),然后再将它们链接到一起。

因此,如果使用了下面的选项就会生成 C4727:

cl /clr /GL a.cpp b.cpp c.cpp /Ycstdafx.h

您可改用下面的选项:

cl /clr /GL a.cpp /Ycstdafx.h /c

cl /clr /GL b.cpp c.cpp /Yustdafx.h /link a.obj

有关更多信息,请参见

  • /Yc(创建预编译的头文件)

  • /Yu(使用预编译的头文件)

        

        MSDN对C4727是这样解决的:

您可改用下面的选项:

cl /clr /GL a.cpp /Ycstdafx.h /c

cl /clr /GL b.cpp c.cpp /Yustdafx.h /link a.obj

        原本以为要手动使用命令含编译所有文件,我想还是算了,因为我的工程有30多个文件,除非写成bat,否则编译还会累死。幸好我不会bat,没有动手去写。我猜想VS功能这么强大,既然报错了,错误应该有解决办法的,不可能让我们手动去编译。在查看工程属性的时候,无意间,点到了stdafx.cpp,豁然开朗!


然后按照MSDN的那个解觉办法的原理,进行设置,就OK了。对工具和问题原理不熟,害我走这么多的弯路啊。


总结下,大致四种解决的办法:

1. 如果无意间,删掉xxx.pch文件,如果编译器报错 “C1083无法打开xxx.pch”,一般只需要清理解决方案,然后rebuild all就可以了。

2.如果1不管用,将【创建/使用预编译头】改为 “不使用编译头”,然后清理解决方案,rebuild即可。

3.如果仍然不管用,将【创建/使用预编译头】改为 “创建预编译头”,然后清理解决方案,rebuild即可。

4.如果上述方法,还不管用。将【创建/使用预编译头】改回 “使用预编译头”,然后将“stdafx.cpp”的属性中【创建/使用预编译头】,改成 "创建预编译头",将其他所有的.cpp文件的预编译属性均该为 “使用预编译头” ,清理解决方案,然后rebuild即可。

注意:每次更改属性后,都要清理解决方案,如果工程目录下的.pch文件仍然存在,要手动删掉。

尽量不要使用2、3的方法,应直接使用4的方法,虽然麻烦点,但是值得的,免得后面哪里删错了有出现这鬼错误。


预编译头是很有用的,不要因为错误而认为他碍眼,也不要因为VC每个源文件前面都要包含stdafx.h而觉得麻烦。下面是别人对这个错误以及对编译头的分析。


error无法打开预编译头文件的解决方法及预编译头原理

1。用VC.NET编辑程序,按Ctrl+F7,出现下列错误:
fatal error C1083: 无法打开预编译头文件:“Debug/UGFace.pch”: No such file or directory
解决方法:修改:项目->属性->C/C++ ->预编译头->不使用预编译头 即可。


2。学用Visual C++ 6.0的第一个例程就让我出了错.用向导生成第一个基于对话框的Project之后,当我按照书上的源程序一个字一个字地输进去之后,始终有一个错误:
fatal error C1010: unexpected end of file while looking for precompiled header directive.找了无数次之后,我决定把向导生成的包括头文件的语句:include"StdAfx.h"保留(而这之前我是把它删掉了的,因为书上的例子没有这句.)咦,这下就对了.这是为什么呢?我百思不得其解。
来到我的VC源代码目录,我注意到每个Project下面的DEBUG文件夹都特别大,而且一个扩展名为 .pch的文件占去了绝大部分,我删掉之好像对程序编译运行没有什么影响。于是抱着对.pch文件的好奇,我在网上搜到了我疑惑之处的解答。这就是VC++6.0给我们带来的:预编译头文件。预编译头文件(一般扩展名为.PCH),是把一个工程中较稳定的代码预先编译好放在一个文件(.PCH)里。这些预先编译好的代码可以是任何的C/C++代码,甚至可以是inline函数,只是它们在整个工程中是较为稳定的,即在工程开发过程中不会经常被修改的代码。
为什么需要预编译头文件?
一言以蔽之:提高编译速度.一般地,编译器以文件为单位编译。如果修改了一工程中的一个文件则所有文件都要重新编译,包括头文件里的所有东西(eg.Macro宏,Preprocessor预处理),而VC程序中,这些头文件中所包括的东西往往是非常大的,编译之将占很长的时间。但它们又不常被修改,是较稳定的,为单独的一个小文件而重新编译整个工程的所有文件导致编译效率下降,因此引入了.PCH文件。
如何使用预编译头文件以提高编译速度?
要使用预编译头文件,必须指定一个头文件(.H),它包含我们不会经常修改的代码和其他的头文件,然后用这个头文件来生成一个预编译头文件(.PCH),VC默认的头文件就是StdAfx.h,因为头文件是不能编译的,所以我们还需要一个.CPP文件来作桥梁,VC默认的文件为StdAfx.cpp,这个文件里只有一句代码就是:#include "StdAfx.h"。接下来要用它生成.PCH文件,涉及到几个重要的预编译指令:/Yu,/Yc,/Yx,/Fp。简单地说,/Yc是用来生成.PCH文件的编译开关。在Project->setting->C/C++的Category里的Precompiled Header,然后在左边的树形视图中选择用来编译生成.PCH文件的.CPP文件(默认即StdAfx.cpp),你就可以看到/Yc这个开关,它表示这个文件编译了以后是否生成.PCH文件(可能/Yc的c表示create)。/Fp指令指定生成的.PCH文件的名字及路径(可能/Fp的p代表path)。/Yu的u即use,工程中只要包括了.H文件的文件都会有这个/Yu指令。如果选择自动Automatic...的话则原来为/Yc的地方就换成了/Yx指令。如果选择自动,则每次编译时编译器会看以前有没有生成过.PCH文件,有则不现生成否则就再次编译产生.PCH文件。
注意:
A,实际上,由Appzard项目向导生成的默认的头文件及CPP文件StdAfx.h和StdAfx.cpp可以是任何名字的.原因很简单。但如果你要这样做就要记得修改相应的Project->setting...下的几个预编译指(/Yc,/Yu,/Yx,/Fp)的参数。
B.在任何一个包括了将要预编译的头文件而使用了.PCH文件的工程文件的开头,一定必须要是在最开头,你要包含那个指定生成.PCH文件的.H文件(通过.CPP文件包括,默认为StdAfx.cpp),如果没包括将产生我最开头产生的错误.如果不是在最开头包括将产生让你意想不到的莫名其妙错误,如若不信,盍为试之?
C.预编译文件.PCH生成之很耗时间,而且生成之后它也很占磁盘空间,常在5-6M,注意项目完成之后及时清理无用的.PCH文件以节约磁盘空间。
D.如果丢了或删了.PCH文件而以后要再修改工程文件时,可将指定的/Yc的.CPP文件(默认为StdAfx.cpp)重新编译一次即可再次生成.PCH文件,不用傻傻的按F7或Rebuild All


以前还碰到过另外一种情况:新建一个工程,随便找一个cpp文件,按ctrl+f7系统将会提示:fatal error C1083: 无法打开预编译的头文件:”Debug/xxx.pch”: No such file or directory(其中xxx是工程的名字)这种情况也是一样的原因,为vc的stdafx.h头文件未编译所致。也可以这样解决:先F7,编译后再ctrf+f7。

注意:VS智能设备程序(如WM5)预编译头文件为stdafx.h。更改设置在项目->XXXX属性->配置属性->C/C++->预编译头 的右侧第一项


你可能感兴趣的:(vs)