怎么快速构建自己的C/C++程序?——有关编译、静态链接和SCons

1. 写在前面

最初写C++是在Visual Studio这个IDE里,那时我并没有makefile的概念,对程序的编译和链接的一些概念也是比较模糊。在VS下,随便增加h/cpp文件,基本上按下编译运行只要不报错就能运行。

后来开始尝试在linux平台写程序时,为避免编写makefile,曾经一度在Ubuntu里使用Code Block这个IDE。

使用IDE也有很多不爽的地方。再后来我就习惯于在Win下用Sublime的文本编辑器写程序,因为它有很好的目录层级,然后扔到linux上去编译。发现不用makefile基本控制不了不断膨胀的代码,然而又发现即便用各种自动工具去生成makefile学习曲线依然有些陡峭。

在探索cmake、autuomake的途中,了解到了SCons。这是一个可以用python去编写类似makefile的工具,它的使用难度相比cmake和automake都要简单不少。有了SCons这一神器,makefile便不再可怕,而是一件很简单的事情了。

本文首先简单介绍有关编译和静态链接的相关知识,不去涉及更复杂的动态链接等知识,真的是简单介绍;再介绍如何用SCons简单且快速地去构建自己的程序。对编译和链接过程有所了解,可以在构建过程更清楚发生了什么,明白链接时报错的意思。

2. 编译与静态链接

2.1 编译Hello World

当我们编写最简单的"Hello World"代码后,将它编译运行。

#include <stdio.h>
int main()
{
printf("Hello World");
return 0;
} gcc hello.c

这时会经过四个步骤,分别是预处理、编译、汇编和链接。

  • 预处理:处理掉源代码文件中那些“#”号开始的预编译指令,删除掉注释等。
  • 编译:最复杂的部份,经过词法分析、语法分析、语义分析及优化产生相应的汇编语言文件。
  • 汇编:将汇编代码转换机器语言,生成可重定位目标文件,也就是构建过程中常见的.o文件。
  • 链接:将必要的目标文件组合起来,生成可执行文件。

2.2 目标文件

目标文件有三种:

  1. 可重定位目标文件。在Linux是常见的.o文件。包含二进制代码和数据,可以在链接时与其他可重定位目标文件合并,创建可执行目标文件。
  2. 可执行目标文件。包含了二进制代码,可以被拷贝至内存并执行。
  3. 共享目标文件。特殊类型的可重定位目标文件,可以在加载或者运行时被动态地加载到内存中并链接。

在程序构建的过程,编译器和汇编器生成可重定位目标和共享目标文件,链接器生成可执行文件。它们在Linux下使用ELF格式存储,是可执行和可链接格式(Executable and Linkable Format, ELF)的缩写。

2.2.1 可重定位目标文件

目标文件中有变量定义在别的模块中。在C/C++中模块之间可以通过函数调用,可以去使用别的模块定义的全局变量。

2.3 静态链接

2.3.1 符号解析

2.2.2 重定位

3. 使用Scons构建自己的程序

以下是scons编译panguan的脚本

#!/usr/bin/scons -f
# coding:utf-8
import os process_name = 'panguan-v0.1' def all_dir(ori_dir):
src_set = []
for root, dirs, files in os.walk(ori_dir):
for dir in dirs:
path = os.path.join(root, dir)
src_set.append(path)
return src_set def all_src_file(dir):
src_set = []
for root, dirs, files in os.walk(dir):
for file in files:
path = os.path.join(root, file)
if file.lower().endswith(('.cpp', '.c')):
src_set.append(path)
return src_set def main():
env = Environment(ENV=os.environ) # add the gcc complie(above 4.8) pos
env.PrependENVPath('PATH', '/usr/local/gcc-4.8.5/bin')
env.PrependENVPath('LIBRARY_PATH', '/usr/local/gcc-4.8.5/lib64')
env.PrependENVPath('CPLUS_INCLUDE_PATH',
'/usr/local/gcc-4.8.5/include/c++/4.8.5') # add the nanomsg lib pos
# env.PrependENVPath('CPLUS_INCLUDE_PATH', 'lib/include')
# env.PrependENVPath('LIBRARY_PATH', 'lib/lib')
# env.PrependENVPath('LD_LIBRARY_PATH', 'lib/lib') # modify argues
env.Replace(CCFLAGS='-Wl,--no-as-needed -std=c++11 -Wall -g -Wno-reorder -lpthread -pthread')
env.Replace(LINKFLAGS='-Wl,--no-as-needed -std=c++11 -Wall -g -Wno-reorder -lpthread -pthread')
header_path = []
header_path += all_dir('common')
env.Prepend(CPPPATH=header_path) common_src = []
common_src += all_src_file('common')
env.StaticLibrary('common/common', common_src)
env.Prepend(LIBPATH=['common'])
env.Prepend(LIBS=['common']) header_path += all_dir('src')
env.Prepend(CPPPATH=header_path) src_file = []
src_file += all_src_file('src')
env.Program(process_name, src_file) if __name__ != '__main__':
main()

怎么快速构建自己的C/C++程序?——有关编译、静态链接和SCons的更多相关文章

  1. C程序的编译与链接

    编译器驱动程序 编译器驱动程序可以在用户需要时调用语言预处理器.编译器.汇编器和链接器. 例如使用GNU编译系统,我们需要使用如下命令来调用GCC驱动程序: gcc -o main main.c 编译 ...

  2. 窥探C语言程序的编译、链接与.h文件

    概述 C语言程序从源文件经过编译.链接生成可执行文件.那么编译与链接分别做了什么? 开发中为什么使用.h编写函数的声明?接下来使用案例说清楚为什么这样编写代码. C语言程序的编译和链接 C语言程序从源 ...

  3. C/C++程序从编译到链接的过程

    编译器是一个神奇的东西,它能够将我们所编写的高级语言源代码翻译成机器可识别的语言(二进制代码),并让程序按照我们的意图按步执行.那么,从编写源文件代码到可执行文件,到底分为几步呢?这个过程可以总结为以 ...

  4. Linux中程序的编译和链接过程

    1.从源码到可执行程序的步骤:预编译.编译.链接.strip 预编译:预编译器执行.譬如C中的宏定义就是由预编译器处理,注释等也是由预编译器处理的. 编译: 编译器来执行.把源码.c .S编程机器码. ...

  5. 原创 C++应用程序在Windows下的编译、链接:第一部分 概述

    本文是对C++应用程序在Windows下的编译.链接的深入理解和分析,文章的目录如下: 我们先看第一章概述部分. 1概述 1.1编译工具简介 cl.exe是windows平台下的编译器,link.ex ...

  6. 关于一个程序的编译过程 zkjg面试

    http://blog.csdn.net/gengyichao/article/details/6544266 一 以下是C程序一般的编译过程: 从图中看到: 将编写的一个c程序(源代码 )转换成可以 ...

  7. C++应用程序在Windows下的编译、链接(一)概述

    C++应用程序在Windows下的编译.链接(一)概述 本文是对C++应用程序在Windows下的编译.链接的深入理解和分析,文章的目录如下: 我们先看第一章概述部分. 1概述 1.1编译工具简介 c ...

  8. 13_传智播客iOS视频教程_OC程序的编译链接

    C程序的编译.链接.执行怎么来的?在.C文件里面写上符合C语言部分的源代码.OC也是一样的.记住:OC程序的后缀名是.m. 为什么要链接?第一个.o的目标文件里面它启动不了.因为它没有启动代码我们要加 ...

  9. Java Swing快速构建窗体应用程序

    以前接触java感觉其在桌面开发上,总是不太方便,没有一个好的拖拽界面布局工具,可以快速构建窗体. 最近学习了一下NetBeans IDE 8.1,感觉其窗体设计工具还是很不错的 , 就尝试一下做了一 ...

随机推荐

  1. PWA之Web 应用清单

    原文 简书原文:https://www.jianshu.com/p/5c96242188e8 大纲 1.什么是Web 应用清单 2.“清单文件”:Web App Manifest 规范的应用 3.we ...

  2. 从源码角度实现一个自己的Promise

    原文链接:https://geniuspeng.github.io/2017/12/14/my-promise/ 关于Promise的概念以及意义就不在这里介绍了,最近看到了一些实现Promise的核 ...

  3. [Angular2 Router] Setup page title with Router events

    Article import 'rxjs/add/operator/filter'; import 'rxjs/add/operator/map'; import 'rxjs/add/operator ...

  4. php 获取根目录

    在网站根目录的index.php文件里 define('BASE_PATH',str_replace('\\','/',realpath(dirname(__FILE__).'/'))."/ ...

  5. ios开发抽屉效果的封装使用

    #import "DragerViewController.h" #define screenW [UIScreen mainScreen].bounds.size.width @ ...

  6. 【浅墨Unity3D Shader编程】之中的一个 夏威夷篇:游戏场景的创建 &amp; 第一个Shader的书写

    本系列文章由@浅墨_毛星云 出品.转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/40723789 作者:毛星云(浅墨)  ...

  7. gcc新版本号引起的编译错误(命令运行时的外部库输入位置)

    昨天,遇到一个比較bug的错误,用gcc来编译几个简单的文件出错,编译环境为x86_64的Ubuntu12.04.gcc版本号号例如以下: gcc (Ubuntu/Linaro 4.6.3-1ubun ...

  8. HTML中DOM对象的属性和方法的层级关系是怎样的?(目录即层次)

    HTML中DOM对象的属性和方法的层级关系是怎样的?(目录即层次) 一.总结 一句话总结:目录就是测试题 1.document取得元素(get element)的方式有哪几种? 解答:四种,分别是id ...

  9. freemarker中间split字符串切割

    freemarker中间split字符串切割 1.简易说明 split切割:用来依据另外一个字符串的出现将原字符串切割成字符串序列 2.举例说明 <#--freemarker中的split字符串 ...

  10. 【38.46%】【codeforces 615E】Hexagons

    time limit per test 1 second memory limit per test 256 megabytes input standard input output standar ...