在mac上编译cpp文件出现连接器报错问题(gcc vs g++)

一、背景

由于在不同操作系统上学习和使用C++时,环境不同,操作结果会产生差异。

我在mac上写了一个最简单的 c++ 程序,源码如下:

/* hello.cpp*/
#include <iostream>
int main(int argc, char *argv[]){
    std::cout << "hello world" << std::endl;
    return(0);
}%

编译代码:gcc hello.cpp

编译时报错,报错如下

二、报错信息

jie@A-MacBookPro c-test % gcc hello.cpp
Undefined symbols for architecture arm64:
  "std::__1::locale::use_facet(std::__1::locale::id&) const", referenced from:
      std::__1::ctype<char> const& std::__1::use_facet<std::__1::ctype<char> >(std::__1::locale const&) in hello-3ecbb1.o
  "std::__1::ios_base::getloc() const", referenced from:
      std::__1::basic_ios<char, std::__1::char_traits<char> >::widen(char) const in hello-3ecbb1.o
  "std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >::__init(unsigned long, char)", referenced from:
      std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >::basic_string(unsigned long, char) in hello-3ecbb1.o
  "std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >::~basic_string()", referenced from:
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> > std::__1::__pad_and_output<char, std::__1::char_traits<char> >(std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >, char const*, char const*, char const*, std::__1::ios_base&, char) in hello-3ecbb1.o
  "std::__1::basic_ostream<char, std::__1::char_traits<char> >::put(char)", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::endl<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&) in hello-3ecbb1.o
  "std::__1::basic_ostream<char, std::__1::char_traits<char> >::flush()", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::endl<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&) in hello-3ecbb1.o
  "std::__1::basic_ostream<char, std::__1::char_traits<char> >::sentry::sentry(std::__1::basic_ostream<char, std::__1::char_traits<char> >&)", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
  "std::__1::basic_ostream<char, std::__1::char_traits<char> >::sentry::~sentry()", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
  "std::__1::cout", referenced from:
      _main in hello-3ecbb1.o
  "std::__1::ctype<char>::id", referenced from:
      std::__1::ctype<char> const& std::__1::use_facet<std::__1::ctype<char> >(std::__1::locale const&) in hello-3ecbb1.o
  "std::__1::locale::~locale()", referenced from:
      std::__1::basic_ios<char, std::__1::char_traits<char> >::widen(char) const in hello-3ecbb1.o
  "std::__1::ios_base::__set_badbit_and_consider_rethrow()", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
  "std::__1::ios_base::clear(unsigned int)", referenced from:
      std::__1::ios_base::setstate(unsigned int) in hello-3ecbb1.o
  "std::terminate()", referenced from:
      ___clang_call_terminate in hello-3ecbb1.o
  "___cxa_begin_catch", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
      ___clang_call_terminate in hello-3ecbb1.o
  "___cxa_call_unexpected", referenced from:
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >::failed() const in hello-3ecbb1.o
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >::ostreambuf_iterator(std::__1::basic_ostream<char, std::__1::char_traits<char> >&) in hello-3ecbb1.o
  "___cxa_end_catch", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
  "___gxx_personality_v0", referenced from:
      std::__1::basic_ostream<char, std::__1::char_traits<char> >& std::__1::__put_character_sequence<char, std::__1::char_traits<char> >(std::__1::basic_ostream<char, std::__1::char_traits<char> >&, char const*, unsigned long) in hello-3ecbb1.o
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> > std::__1::__pad_and_output<char, std::__1::char_traits<char> >(std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >, char const*, char const*, char const*, std::__1::ios_base&, char) in hello-3ecbb1.o
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >::failed() const in hello-3ecbb1.o
      std::__1::ostreambuf_iterator<char, std::__1::char_traits<char> >::ostreambuf_iterator(std::__1::basic_ostream<char, std::__1::char_traits<char> >&) in hello-3ecbb1.o
      std::__1::basic_ios<char, std::__1::char_traits<char> >::widen(char) const in hello-3ecbb1.o
ld: symbol(s) not found for architecture arm64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

大致提示是 ld链接代码有问题:ld: symbol(s) not found for architecture arm64

三、解决方案

换成 :g++ hello.cpp

四、原因

4.1 首先看下gcc 和 g++ 的版本

gcc版本

jie@A-MacBookPro c-test % gcc --version
Configured with: --prefix=/Applications/Xcode.app/Contents/Developer/usr --with-gxx-include-dir=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include/c++/4.2.1
Apple clang version 13.0.0 (clang-1300.0.29.3)
Target: arm64-apple-darwin21.2.0
Thread model: posix
InstalledDir: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin

g++版本

jie@A-MacBookPro c-test % g++ --version
Configured with: --prefix=/Applications/Xcode.app/Contents/Developer/usr --with-gxx-include-dir=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include/c++/4.2.1
Apple clang version 13.0.0 (clang-1300.0.29.3)
Target: arm64-apple-darwin21.2.0
Thread model: posix
InstalledDir: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin

从以上两个命令的版本信息结果看出,输出结果都是一样的。那为什么会出现使用gcc 报错,g++又可以的。这就要从gcc的历史说起。

gcc时GNU开发的针对c的编译器,刚开始只支持编译c代码,随着gcc的发展愈发强大,后面gcc也支持编译c++、Objective-c和java等,在编译时需要通过设定参数指定编译的语言。所以后来,gcc默认编译的是c代码。把参数给用户设置显然没有那么友好,于是就专门针对c++ 开发了g++编译器。

所以gcc是一个编译器集合,而g++是针对c++的编译器。

因此,想要使用gcc编译c++的代码需要加上参数 -lstdc++ 指令即可。

例:gcc -lstdc++ hello.cpp

说明:-l 表示链接 std c++标准库。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
GCC(GNU Compiler Collection)是一款广泛使用的编译器套件,支持多种编程语言,包括C、C++、Objective-C、Fortran、Java等。GCC的功能组成可以大致分为以下几个部分: 1. 前端(Front-end):GCC的前端负责将源代码解析并生成中间表示(IR),同时进行语法分析、语义分析和类型检查等相关工作。针对不同的编程语言,GCC提供了相应的前端模块,如C前端(gcc/c)、C++前端(gcc/cc1plus)、Objective-C前端(gcc/cc1obj)等。 2. 优化器(Optimizer):GCC的优化器对生成的中间表示进行优化,以提高程序的性能和执行效率。优化器可以进行诸如常量传播、死代码消除、循环优化、函数内联等一系列优化操作。 3. 后端(Back-end):GCC的后端将优化后的中间表示翻译成目标机器的汇编代码。GCC的后端部分是与目标机器架构相关的,针对不同的目标机器架构,需要提供相应的后端模块,如x86后端(gcc/cc1)、ARM后端(gcc/cc1arm)等。 4. 连接器(Linker):GCC连接器负责将编译后的目标文件(或库文件)进行链接,生成可执行文件或共享库。连接器将不同的目标文件合并成一个整体,并解析符号引用和重定位等操作。 除了以上主要的功能组成部分,GCC还提供了一些辅助工具和库,如预处理器(cpp)用于处理源代码中的宏定义和条件编译,汇编器(as)用于将汇编代码转换为目标文件,以及一些用于静态分析和调试的工具等。 总之,GCC作为一款强大而灵活的编译器套件,通过前端、优化器、后端和连接器等组成部分的协作,实现了将高级语言代码转换为可执行的目标机器代码的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值