.C与.CPP 在编译时的区别

本文深入探讨了使用gcc、g++、cl等编译器编译C和C++程序时的差异,重点阐述了在不同编译方式下生成的obj文件symbol的不同,导致以不同方式编译的obj文件无法成功链接的现象,并通过demo示例进行验证。
结论
            1.gcc认为.c的为C程序,.cpp的为C++程序;
            2.g++认为.c的为C++程序,.cpp的为C++程序;
            3.VC++的编译器cl认为.c的为C程序,.cpp的为C++程序;
            4.C程序与C++程序中同样的函数在编译后的obj文件中的symbol是不同的,所以以C方式编译的obj文件与以C++方式
               编译的obj文件无法成功链接。
 
            使个demo说明一下:
            准备工作:
            为gcc、g++、cl(VC++编译器)、link(VC++链接器) 设置好环境变量
             //demo.cpp
            #include <stdio.h>
            #include "foo.h"
            int main()
            { 
                printHello(); 
                return 0;
            }
 
             //foo.h
            void printHello();
 
             //foo.c
            #include <stdio.h>
            void printHello()
            {
                printf("Hello MM");
            }
 
              1.gcc、g++测试,在windows的cmd下:
            D:\> g++ demo.cpp -o demo.obj -c (以C++方式编译生成demo.obj文件,-c选项表示只编译不链接)
            D:\> gcc foo.c -o foo.obj -c (以C方式编译生成foo.obj文件)
            D:\> g++ demo.obj foo.obj -o demo (链接demo.obj、foo.obj文件)
            demo.obj(.text+0x2b):demo.cpp: undefined reference to `printHello()'
            collect2: ld returned 1 exit status
                 提示说 找不到printHello,因为按照C++的编译方式去找printHello应该对应某一种格式的symbol,但是我们
            的foo.obj中printHello的symbol是另外一种格式的,所以找不到了。
             把foo.c改成foo.cpp就可以成功链接了。
 
            2. cl、link测试,在windows的cmd下:
            D:\> cl demo.cpp /c (以C++方式编译生成demo.obj文件,/c选项表示只编译不链接)
            D:\> cl foo.c /c (以C方式编译生成foo.obj文件)
            D:\> link demo.obj foo.obj
            Microsoft (R) Incremental Linker Version 6.00.8168
            Copyright (C) Microsoft Corp 1992-1998. All rights reserved.
            demo.obj : error LNK2001: unresolved external symbol "void __cdecl printHello(void)"( ?printHello@@YAXXZ)
            demo.exe : fatal error LNK1120: 1 unresolved externals
            理由同上,把foo.c改成foo.cpp就可以成功链接了。
 
<think>好的,我现在需要帮助用户解决如何通过.h文件.cpp文件编译生成可执行文件的问题。首先,我得回想一下用户提到的引用内容,特别是引用[1]引用[2],这两个引用提供了关于gcc编译参数使用的信息。用户的问题集中在如何使用头文件(.h)文件(.cpp)来生成可执行文件,所以需要结合这些引用中的内容来构建答案。 首先,我应该考虑基本的编译步骤。对于C++项目,通常的步骤是将多个.cpp文件编译成对象文件(.o),然后链接这些对象文件生成可执行文件。用户可能对如何指定头文件路径链接库有疑问,尤其是当项目结构比较复杂的时候。 根据引用[1],编译时需要用到-I参数来指定头文件的路径,-L来指定库文件的路径,-l来链接具体的库。但用户的问题似乎更基础,可能还没有涉及到外部库,所以重点可能放在如何正确处理多个.cpp文件文件上。 接下来,我需要分步骤解释编译过程。比如,第一步是预处理编译每个.cpp文件生成对应的.o文件,第二步是将这些.o文件链接成可执行文件。可能需要给出具体的g++命令示例,这样用户可以直接复制使用。 另外,用户可能遇到的常见问题包括头文件找不到,或者多个.cpp文件没有正确编译导致链接错误。这时候需要提醒用户使用-I参数来指定头文件目录,或者确保所有相关的.cpp文件都被编译并链接。 还需要注意用户使用的是.cpp文件,即C++代码,所以应该使用g++而不是gcc。虽然gcc也可以处理C++,但使用g++更合适,因为它会自动链接C++标准库。 然后,考虑到用户可能对多文件项目的编译不太熟悉,给出一个简单的示例会很有帮助。例如,假设用户有一个main.cpp一个functions.cpp,以及对应的functions.h,如何将这些文件编译成可执行文件。需要展示编译每个源文件为对象文件,再链接的步骤,或者直接编译所有源文件的一步命令。 另外,可能需要提到Makefile的使用,作为自动化编译的工具,但用户的问题可能更关注手动编译的方法,所以这部分可以作为扩展内容简要提及。 最后,根据用户的要求,生成相关问题。这些问题应该编译过程、参数使用、处理多文件项目相关,比如如何指定头文件路径、如何处理多个.cpp文件、静态库动态库的区别等。 总结起来,回答的结构应该是先说明基本步骤,给出具体命令示例,解释各个参数的作用,然后提供常见问题解决方法,最后附上相关问题。同时,确保引用正确的参考文献,比如在提到-I、-L、-l参数时引用[1],在基本编译命令时引用[2]。</think>以下是通过.h.cpp文件编译生成可执行文件的完整步骤说明: ### 一、基础编译方法 1. **直接编译(适用于简单项目)** 使用g++编译器同时编译所有.cpp文件: ```bash g++ main.cpp functions.cpp -o myprogram ``` 这会自动处理`#include "functions.h"`的依赖关系,生成名为`myprogram`的可执行文件[^2] 2. **分步编译(推荐用于复杂项目)** ```bash # 编译每个cpp为对象文件 g++ -c main.cpp -I./include # -I指定头文件目录 g++ -c functions.cpp -I./include # 链接对象文件生成可执行文件 g++ main.o functions.o -o myprogram ``` ### 二、关键参数说明 - `-I<directory>`:指定头文件搜索路径(当.h文件不在当前目录时必需) - `-o <filename>`:指定输出文件名 - `-c`:仅编译不链接(生成.o中间文件) - `-Wall`:建议添加以显示所有警告信息 ### 三、项目结构示例 假设目录结构为: ``` project/ ├── include/ │ └── functions.h ├── src/ │ ├── main.cpp │ └── functions.cpp ``` 编译命令应为: ```bash g++ src/main.cpp src/functions.cpp -Iinclude -o myprogram ``` ### 四、常见错误处理 1. **头文件找不到**:检查路径是否正确,使用`-I`参数 2. **未定义的引用**:确保所有相关.cpp文件都参编译 3. **C++11特性支持**:添加`-std=c++11`参数 ### 五、进阶工具建议 对于多文件项目,推荐使用Makefile自动化编译: ```makefile CXX = g++ CXXFLAGS = -Iinclude -Wall -std=c++11 myprogram: main.o functions.o $(CXX) $^ -o $@ %.o: src/%.cpp $(CXX) $(CXXFLAGS) -c $< -o $@ ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值