应该是重定义了,但是又解决不了,看似又没有重定义啊,就在一个文件定义了啊?怎么会出现这种情况呢?关键是,编译报错,程序运行不了了。
这里提一下我的前期操作,是因为将一个头文件和源文件合并为一个头文件了,然后在主函数文件嵌入这个头文件。
一个解决办法:
vs2017错误:error:LNK2005 _main 已经在 *.obj 中定义 - 简书 (jianshu.com)
确实编译不报错,但是会警告可能会不执行。
......
试运行程序,没有问题。
应该是解决了。但总是不太放心,没有直接解决问题啊,到底哪里出问题了呢?
另外一些分析:
error:LNK2005 _main 已经在 *.obj 中定义_cmakecudacompilerld.cu.obj main已经在main.cpp.obj定义-CSDN博客https://blog.csdn.net/yan_less/article/details/73149578
感觉很可能是这里面的变量没有用extern重新声明下。但是尝试在需要使用的文件将这些变量再extern一下,结果还是报错。看来不是这里的问题。
多次尝试,才发现问题。先说背景原因:
C编译,嵌入头文件,即include语句,意思就是拷贝对应头文件到嵌入的地方,如果头文件中对变量进行定义并初始化了。那么在嵌入头文件的文件,这些变量又会初始化,C不允许重复对一个变量定义并初始化,就会报这种重复定义的错误。
分析下我的程序结构:
看前期操作,是将一个源文件和头文件合并,写成一个头文件。然后在主程序文件include这个头文件。理论上,头文件定义的变量在主文件是可以用的,不会出现重定义这种问题,因为主文件也没再定义。但实际上,头文件中变量,只能声明,不能定义。如果定义了,那么在嵌入的主文件,编译时是将头文件定义的代码又拷贝到主文件,则会形成重定义。
所以,头文件最好是只声明。C中,变量的重复声明是没有任何问题的!
所以是这种合并操作导致的。要直接解决问题,则还是要将源文件和头文件分开写,且头文件不能定义,只能声明。
实际上,多年以来一直有个问题困扰我很久,总弄不明白,就是一般一个头文件对应一个源文件,但是,一些数据类型、宏定义等,在头文件中定义,在源文件中使用,同时,在其他文件也要使用,那么源文件必须嵌入这个头文件,其他文件也需要嵌入这个头文件。然后,一些变量,在源文件中定义,别的源文件需要使用,则,就需要在头文件中声明。所以,源文件和头文件是相互引用的关系?是不是应该定义两个头文件,一个头文件只定义源文件要用的数据类型、宏等,另一个头文件,提供源文件中其他文件需要使用的变量和函数的声明?其他文件在嵌入时,只需要嵌入后一个头文件?这样就解决了源文件和头文件相互引用的问题?
但是,如果其他文件还需要用第一个头文件定义的数据类型和宏呢?因为源文件中定义的一些其他文件需要使用的变量可能是基于这些数据类型和宏,那不是还要再嵌入第一个头文件?那不是很麻烦么???
现在就很好解决了。其实,只需要在头文件中变量声明时前面加extern关键字就可以了。这些变量,是在源文件中定义的,在头文件声明,其他源文件只需要嵌入头文件,即可使用。源文件虽然嵌入头文件,再次碰到这些变量声明,但是因为用了extern,可以忽略这些声明,而只使用本文件内定义,使用即可。
而头文件中声明(定义)的数据类型、宏,可以在源文件用,也可以在其他嵌入这个头文件的文件使用。其实,这些数据类型和宏,完全可以在源文件定义,然后在头文件再声明,和变量、函数是一样的处理。只不过,一般就不需要了,只需要在头文件定义,源文件引用即可。
完美!其实是应该开始学C就该搞清楚的问题,我现在才搞清楚,哎,就这样吧,因为可能很多人都一样,不会去注意这个问题,国内没有讲述清楚底层、内幕的风气,希望能帮到一些人吧。