gcc

概要

  • GCC:GNU Compiler Collection(GUN 编译器集合),是GNU项目中符合ANSI C标准的编译系统,它可以编译C、C++、JAV、Fortran、Pascal、Object-C、Ada等语言。并且GCC是一个交叉平台编译器,能够在当前CPU平台上为多种不同体系结构的硬件平台开发软件,因此尤其适合在嵌入式领域的开发编译。

    • gcc是GCC中的GUN C Compiler(C 编译器)。
    • g++是GCC中的GUN C++ Compiler(C++编译器)。
  • GCC编译流程:
    • 预处理(Pre-Processing)
    • 编译(Compiling)
    • 汇编(Assembling)
    • 链接(Linking)

  • GCC特点:
    • GCC是一个可移植的编译器,支持多种硬件平台。例如ARM、X86等等。
    • GCC不仅是个本地编译器,它还能跨平台交叉编译。所谓的本地编译器,是指编译出来的程序只能够在本地环境进行运行。而GCC编译出来的程序能够在其他平台进行运行。例如嵌入式程序可在x86上编译,然后在arm上运行。
    • GCC有多种语言前端,用于解析不同的语言。
    • GCC是按模块化设计的,可以加入新语言和新CPU架构的支持。
    • GCC是自由软件。任何人都可以使用或更改这个软件。

基本指令及功能(以gcc为例)

  • gcc编译文件
后缀 解析 后缀 解析
.c C原始程序 .s/.S 汇编语言
.C/.cc/.cxx C++原始程序 .h 预编译头文件
.m Objective-C原始程序 .o 目标文件
.i 已预处理C原始程序 .a/.so 库文件
.ii 已预处理C++原始程序
  • gcc编译选项
选项 作用
-o file 产生目标file文件.i/.s/.o/可执行文件等
-E 只运行C预编译器
-S 产生汇编文件.s/.S后停止编译
-c 产生目标文件.o但不链接形成可执行文件
-Wall 使gcc对源程序代码有问题的地方发出警告
-I dir 将dir目录加入搜索头文件目录路径
-L dir 将dir目录加入搜索库文件目录路径
-llib 链接到名为lib的库
-g 在目标文件嵌入调试信息,方便gdb/cgdb程序调试
-ansi 只支持ANSI标准编译,禁止GNU C的某些特色(如asm/typeof关键字)
-DMACRO 以字符串"1"定义MACRO宏
-DMACRO=DEFN 以字符串"DEFN"定义MACRO宏
-IDIRECTORY 指定额外的头文件搜索路径DIRECTORY
-LDIRECTORY 指定额外的函数库搜索路径DIRECTORY
-m486 针对486进行代码优化
-o FILE 生成指定的输出文件,用在生成可执行文件时
-O0 不进行优化处理
-O/-O1/-O2/-O3 优化生成代码
-shared 生成共享目标文件,通常用在建立共享库时
-static 禁止使用共享连接
-UMACRO 取消对MACRO宏的定义
-w 不生成任何警告信息
  • 编译案例
指令一般格式:gcc [选项] 要编译的文件 [选项] [目标文件]

# 对hello.c文件进行预处理,生成了hello.i 文件
gcc -E hello.c -o hello.i
# 对预处理文件进行编译,生成了汇编文件
gcc -S hello.i -o hello.s
# 对汇编文件进行编译,生成了目标文件
gcc -c hello.s -o hello.o
# 对目标文件进行链接,生成可执行文件
gcc hello.o -o hello
# 直接编译链接成可执行目标文件
gcc hello.c -o hello
# 编译生成可重定位目标文件
gcc -c hello.c or gcc -c hello.c -o hello.o
# gcc输出警告选项
gcc -Wall hello.c -o hello
  • 独立编译和一次编译
# 一次编译
gcc hello.c main.c -o main
# 独立编译
gcc -Wall -c main.c -o main.o
gcc -Wall -c hello.c -o hello.o
gcc -Wall main.o hello.o -o main
独立编译运用到复杂项目时,当其中某个模块代码发生改变,只需编译该模块代码,
不必重新编译所有程序文件,这样可以节省编译时间。
    • 可重用函数关联功能函数
    • 模块化开发
    • 可维护性
  • 静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库。静态库比较占用磁盘空间,而且程序不可以共享静态库。运行时也是比较占内存的,因为每个程序都包含了一份静态库。
  • 动态库(.so):程序在运行的时候才去链接共享库的代码,多个程序共享使用库的代码,这样就减少了程序的体积。
  • 头文件及库文件位置
    • /usr/include及其子目录底下的include文件夹
    • /usr/local/include及其子目录底下的include文件夹
    • /usr/lib
    • /usr/local/lib
    • /lib
  • 库生成及使用
# rcs(replace and create)生成静态库
ar rcs libhello.a hell.o
# 使用静态库
gcc -Wall main.c libhello.a -o main
# 其中lhello为libhello的缩写
gcc -Wall -L main.c -o main -lhello # 生成共享库(动态库)
gcc -shared -fPIC hello.o -o libhello.so
# 使用动态库
gcc -Wall main.o -o main -L. -lhello # 库路径的搜索原则:库的搜索路径遵循几个搜索原则:从左到右搜索-I -l指定的目录,如果在这些目录中找不到,
那么gcc会从由环境 变量指定的目录进行查找。头文件的环境变量是C_INCLUDE_PATH,库的环境变量是LIBRARY_PATH.如果还是找不到,那么会从系统指定指定的目录进行搜索。
# 共享库有时候并不在当前目录下,需要配置到gcc可寻找的路径下(3种方式)
# 1. copy .so文件到系统共享库路径/usr/lib下
# 2. 在~/.bash_profile文件中,配置LD_LIBRARY_PATH变量
# 3. 配置/etc/ld.so.conf,配置完成后调用ldconfig更新ld.so.cache

gcc -gdb

gcc -cmake

  • cmake基本语法规则

    • 变量使用${}取值,IF控制语句中直接使用变量名
    • 指令(参数1 参数2 ...)
    • 指令大小写无关,参数和变量大小写相关,推荐全部使用大写
  • 指定cmake最小版本
CMAKE_MINIMUM_REQUIRED(VERSION 3.5.1)
  • 设置项目名称(此处自动引入两个变量 hello_BINARY_DIR 和 hello_SOURCE_DIR)
PROJECT(hello)
  • 生成可执行文件
ADD_EXECUTABLE(hello hello_world.cpp)
  • 设置编译类型
ADD_LIBRARY(hello hello.cpp) # 默认生成静态库
ADD_LIBRARY(hello STATIC hello.cpp) # 指定为静态库
ADD_LIBRARY(helloshared SHARED hello.cpp) # 指定为共享库

假设 hello.cpp 文件中没有 main 函数,意味着这个库中没有可执行文件。添加上述命令,cmake 后会依次生成下面几个文件:

libhello.a

libhello.a

libhello_shared.so

静态库以 .a 为后缀名,共享库以 .so 为后缀名,所有库都是一些函数打包后的集合,静态库每次被调用都会生成一个副本,而共享库则只有一个副本,更省空间。

库文件是一个压缩包,里面有编译好的二进制函数,仅有 .a 或 .so 库文件,别人并不知道里面的函数是什么样的,或如何调用;这就需要提供一个头文件,有了库文件和头文件,就可以调用这个库。

  • 设置变量

    • set直接设置变量值
    SET(SRC_LIST a.cpp b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST})
    • set追加设置变量值
    SET(SRC_LIST a.cpp)
    SET(SRC_LIST ${SRC_LIST} b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST})
    • LIST追加或删除变量值
    SET(SRC_LIST a.cpp)
    LIST(APPEND SRC_LIST b.cpp)
    LIST(REMOVE_ITEM SRC_LIST b.cpp)
    ADD_EXECTUABLE(demo ${SRC_LIST})
  • 指定源文件
    • 明确指定(空格或分好隔离开)
    ADD_LIBRARY(hello hello.cpp a.cpp b.cpp)
    • 所有源文件
    # AUX_SOURCE_DIRECTORY(dir VAR) 发现一个目录下所有的源代码文件并将列表存储在一个变量中
    AUX_SOURCE_DIRECTORY(. SRC_LIST) # 搜索当前目录下的所有.cpp文件
    ADD_LIBRARY(hello ${SRC_LIST})
  • 查找指定库文件

    FIND_LIBRARY(VAR name path)查找到指定的预编译库,并将它的路径存储在变量中。默认的搜索路径为 cmake 包含的系统库。

    类似的命令还有 find_file()、find_path()、find_program()、find_package()等。
  • 设置包含目录
# case 0
INCLUDE_DIRECTORIES(
${CMAKE_CURRENT_SOURCE_DIR}
${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_SOURCE_DIR}/include # case 1
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -I${CMAKE_CURRENT_SOURCE_DIR}")
  • 设置链接库搜索目录
# case 0
LINK_DIRECTORIES(
${CMAKE_CURRENT_SOURCE_DIR}/libs
) # case 1
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_CURRENT_SOURCE_DIR}/libs")
  • 设置target需要链接的库
# 指令TARGET_LINK_LIBRARIES(target_lib  link_lib_path)

# 指定链接静态库或动态库
TARGET_LINK_LIBRARIES(hello libhello.a)
TARGET_LINK_LIBRARIES(hello libhello_shared.so) # 设置全路径
TARGET_LINK_LIBRARIES(hello ${CMAKE_CURRENT_SOURCE_DIR}/libs/libhello.a)
TARGET_LINK_LIBRARIES(hello ${CMAKE_CURRENT_SOURCE_DIR}/libs/libhello_shared.so)
  • 编译流程

    • 编写hello.cpp源文件和CMakeLists.txt文件
    • 新建build目录,在build目录下cmake ..命令,生成了 CMakeFIles 文件夹和CMakeCache.txt, cmake_install.cmake, Makefie 文件
    • 在build路径下执行make命令
  • CMakeLists.txt文件demo

# cmake 版本
cmake_minimum_required(VERSION 2.6) ####
# CMake中指令不区分大小写
# ${} 表示取出变量中的值
#### # 项目名称, 名称后面为支持的语言,不写时默认支持所有语言
PROJECT(project_name [CXX][C][java])
# 或者
PROJECT(name )
# PROJECT(...)语句运行之后,会默认定义两个变量,后面可以直接使用
# name_BINARY_DIR 编译目录
# name_SOURCE_DIR 工程目录 # set表示设置变量名
SET(LIBRARIES /usr/lib/x86_64-linux-gnu/libm.so)
# 变量中只保存一个源文件
SET(SRC_LIST main.c) # 或者 SET(SRC_LIST “main.c”) # 把main.c保存到变量SRC_LIST中
# 也可以是保存多个源文件
SET(SRC_LIST main.c func.c func.h)
# 当然也可以保存所有的源文件,将所有源文件保存到SRC_LIST变量中
AUX_SOURCE_DIRECTORY(directory SRC_LIST)
# 把当前目录下所有源文件全部保存到SRC_LIST变量中
AUX_SOURCE_DIRECTORY(. SRC_LIST)
# 选择指定目录下的源文件保存到变量中
AUX_SOURCE_DIRECTORY(./dir/main.c SRC_LIST) # message 表示输出提示信息
MESSAGE([SEND_ERROR | STATUS | FATAL_ERROR] "message")
# STATUS 参数较为常用,输出前缀为-的信息
# 第一个参数是消息类型,后面的参数是一条或多条要显示的消息。错误类型有3种:
# SEND_ERROR:表示产生错误信息
# STATUS:表示一般的状态信息
# FATAL_ERROR:我们知道肯定是严重错误信息,cmake会立即停止执行 # 生成可执行文件
ADD_EXECUTABLE(result ${SRC_LIST}) # 添加可执行文件所需要的库
TARGET_LINK_LIBRARIES(result ${LIBRARIES}) # 更改可执行文件的输出目录
SET(EXECUTABLE_OUTPUT_PATH "${CMAKE_BINARY_DIR}/bin")
# 其中CMAKE_BINARY_DIR 变量中的CMAKE_表示项目名的统称,等价于上面所说的 name_BINARY_DIR

gcc_to_use的更多相关文章

随机推荐

  1. 记一次使用gdb诊断gc问题全过程

    原创:扣钉日记(微信公众号ID:codelogs),欢迎分享,转载请保留出处. 简介 上次解决了GC长耗时问题后,系统果然平稳了许多,这是之前的文章<GC耗时高,原因竟是服务流量小?> 然 ...

  2. 安装云崽Bot+GPT3插件

    安装云崽Bot+GPT3插件 这次我们来搭建云崽Bot,云崽Bot其实是一个用于原神的机器人,不过众多大佬开发出了很多有趣的插件供我们使用,这次我们就是用其中的一个插件Chat-gpt(其不是正宗的C ...

  3. W04 散文周刊 : 重回Emacs

    W04 散文周刊 : 重回Emacs 邮箱:ppbhoy@qq.com 博客地址: www.pipihao.com 公纵号:南湖小皮 公告:后面会开始稳定更新 使用Emacs还是有些沉重,我一直将Em ...

  4. xml基本学习

    概念:可拓展标记语言.可拓展即标签都是自定义的.标记语言即由标签构成的语言. 功能:存储数据: 配置文件 在网络中传输 语法 基本语法: xml文件后缀名为.xml xml第一行必须定义为文档声明 x ...

  5. Centos7-RAID冗余磁盘阵列

    转载csdn: Centos7-RAID冗余磁盘阵列_黏住你不让你的博客-CSDN博客

  6. OpenLayers之OGC服务加载

    一.实验内容 手动构造 GetCapabilities.GetMap 的操作链接,并在浏览器里发送HTTP 请求:利用 OpenLayers 进行 WMS 服务加载: 手动构造 GetCapabili ...

  7. odoo Web Controllers 学习总结

    环境 odoo-14.0.post20221212.tar Web Controllers Controllers 控制器需要提供可扩展性,就像Model,但不能使用相同的机制,因为先决条件(已加载模 ...

  8. wordpress宕机原因及处理方法

    2020年7月底,查看了网站日志,是wp-cron.php 导致异常. 原来这是WordPress定时任务,禁用即可. 在wp-config.php添加 /* 禁用定时任务 wp-cron */ de ...

  9. Windows打开代理快捷方式

    最近使用电脑需要频繁打开关闭网络代理(全局代理会导致一些网站打不开),在想有没有快捷方式,一搜果然有,看来有跟我一样的使用习惯的人. 传统切换代理开关,步骤如下: 设置 => 网络和 Inter ...

  10. Servlet(服务连接器)

    Servlet(Server Applet):是Java Servlet的简称,称为小服务程序或服务连接器,用Java编写的服务器端程序,具有独立于平台和协议的特性,主要功能在于交互式地浏览和生成数据 ...