如何使用gcc编译器
开始...
首先,我们应该知道如何调用编译器。实际上,这很简单。我们将从那个著名的第一个C程序开始。
#include <stdio.h>
int main()
{
printf("Hello World!\n");
}
把这个文件保存为lxy.c,保存到当前目录下,在命令行下编译它:
gcc game.c
在默认情况下,C编译器将生成一个名为 a.out 的可执行文件。 你可以键入如下命令运行它:
a.out
Hello World
每一次编译程序时,新的 a.out 将覆盖原来的程序。你无法知道是哪个 程序创建了a.out。我们可以通过使用 -o 编译选项:
- gcc –o lxy lxy.c //对文件进行编译连接,-o选项指定创建的可执行文件名称
- ./lxy //运行刚才生成的lxy程序
- Hello World! //程序输出
看起来上面生成可执行文件的过程中,像是一步就完成了,但是实际上它要经历4个步骤:
下一步为我们的lxy编写一个头文件。头文件把数据类型和函数声明集中到了一处。 这可以保证数据结构定义的一致性,以便程序的每一部分都能以同样的方式看待一切事情。
#ifndef DECK_H
#define DECK_H
#define DECKSIZE 52
typedef struct deck_t
{
int card[DECKSIZE];
/* number of cards used */
int dealt;
}deck_t;
#endif /* DECK_H */
把这个文件保存为 deck.h。只能编译 .c 文件, 所以我们必须修改 lxy.c。即:
#include <stdio.h>
#include "deck.h"
int main()
{
printf("Hello World!\n");
deck_t deck;
}
gcc -o lxy lxy .c
如果没有错误,就没有问题。如果编译不能通过,那么就修改它直到能通过为止。
预编译
编译器是怎么知道 deck_t 类型是什么的呢?因为在预编译期间, 它实际上把"deck.h"文件复制到了"lxy.c"文件中。源代码中的预编译指示以"#"为前缀。 你可以通过在gcc后加上 -E 选项来调用预编译器。
gcc -E -o game_precompile.txt game.c
wc -l game_precompile.txt
3199 game_precompile.txt
几乎有3200行的输出!其中大多数来自 stdio.h 包含文件,但是如果 你查看这个文件的话,我们的声明也在那里。如果你不用 -o 选项指定 输出文件名的话,它就输出到控制台。预编译过程通过完成三个主要任务给了代码很大的 灵活性。
- 把"include"的文件拷贝到要编译的源文件中。
- 用实际值替代"define"的文本。
- 在调用宏的地方进行宏替换。
这就使你能够在整个源文件中使用符号常量(即用DECKSIZE表示一付牌中的纸牌数量), 而符号常量是在一个地方定义的,如果它的值发生了变化,所有使用符号常量的地方 都能自动更新。在实践中,你几乎不需要单独使用 -E 选项,而是让它 把输出传送给编译器。
编译
作为一个中间步骤,gcc把你的代码翻译成汇编语言。它一定要这样做,它必须通过分析 你的代码搞清楚你究竟想要做什么。如果你犯了语法错误,它就会告诉你,这样编译就失败了。 人们有时会把这一步误解为整个过程。但是,实际上还有许多工作要gcc去做呢。
汇编
as 把汇编语言代码转换为目标代码。事实上目标代码并不能在CPU上运行, 但它离完成已经很近了。编译器选项 -c 把 .c 文件转换为以 .o 为扩展名 的目标文件。 如果我们运行
gcc -c lxy.c
我们就自动创建了一个名为lxy.o的文件。这里我们碰到了一个重要的问题。我们可以用任意一个 .c 文件创建一个目标文件。正如我们在下面所看到的,在连接步骤中我们可以把这些目标文件组合成可执行文件。让我们继续介绍我们的例子。因为我们正在编写一个纸牌游戏,我们已经把一付牌定义为 deck_t,我们将编写一个洗牌函数。 这个函数接受一个指向deck类型的指针,并把一付随机的牌装入deck类型。它使用'drawn' 数组跟踪记录那些牌已经用过了。这个具有DECKSIZE个元素的数组可以防止我们重复使用 一张牌。
#include <stdlib.h>
#include <stdio.h>
#include <time.h>
#include "deck.h"
static time_t seed = ;
void shuffle(deck_t *pdeck)
{
/* Keeps track of what numbers have been used */
int drawn[DECKSIZE] = {};
int i;
/* One time initialization of rand */
if( == seed)
{
seed = time(NULL);
srand(seed);
}
for(i = ; i < DECKSIZE; i++)
{
int value = -;
do
{
value = rand() % DECKSIZE;
}
while(drawn[value] != );
/* mark value as used */
drawn[value] = ;
/* debug statement */
printf("%i\n", value);
pdeck->card[i] = value;
}
pdeck->dealt = ;
return;
}
把这个文件保存为 shuffle.c。我们在这个代码中加入了一条调试语句, 以便运行时,能输出所产生的牌号。这并没有为我们的程序添加功能,因为我们的游戏还在初级阶段,我们没有别的办法确定我们的函数是否实现了我们要求的功能。使用那条printf语句,我们就能准确 地知道现在究竟发生了什么,以便在开始下一阶段之前我们知道牌已经洗好了。在我们 对它的工作感到满意之后,我们可以把那一行语句从代码中删掉。这种调试程序的技术 看起来很粗糙,但它使用最少的语句完成了调试任务。
请注意两个问题:
- 我们用传址方式传递参数,你可以从'&'(取地址)操作符看出来。这把变量的机器地址 传递给了函数,因此函数自己就能改变变量的值。也可以使用全局变量编写程序,但是应该 尽量少使用全局变量。指针是C的一个重要组成部分,你应该充分地理解它。
- 我们在一个新的 .c 文件中使用函数调用。操作系统总是寻找名为'main'的函数,并从 那里开始执行。 shuffle.c 中没有'main'函数,因此不能编译为独立的可执行文件。 我们必须把它与另一个具有'main'函数并调用'shuffle'的程序组合起来。
运行命令
gcc -c shuffle.c
并确定它创建了一个名为 shuffle.o 的新文件。编辑lxy.c文件,在第7行,在 deck_t类型的变量 deck 声明之后,加上下面这一行:
shuffle(&deck);
现在,如果我们还象以前一样创建可执行文件,我们就会得到一个错误
gcc -o lxy lxy.c
/tmp/ccmiHnJX.o: In function `main':
/tmp/ccmiHnJX.o(.text+0xf): undefined reference to `shuffle'
collect2: ld returned 1 exit status
编译成功了,因为我们的语法是正确的。但是连接步骤却失败了,因为 我们没有告诉编译器'shuffle'函数在哪里。 那么,到底什么是连接?我们怎样告诉编译器到哪里寻找这个函数呢?
连接
连接器ld,使用下面的命令,接受前面由 as 创建的目标文件并把它转换为可执行文件gcc -o lxy lxy.o shuffle.o这将把两个目标文件组合起来并创建可执行文件 lxy。
连接器从shuffle.o目标文件中找到 shuffle 函数,并把它包括进可执行文件。 目标文件的真正好处在于,如果我们想再次使用那个函数,我们所要做的就是包含"deck.h" 文件并把 shuffle.o 目标文件连接到新的可执行文件中。
象这样的代码重用是经常发生的。虽然我们并没有编写前面作为调试语句调用的 printf 函数,连接器却能从我们用 #include <stdlib.h> 语句包含的文件中 找到它的声明,并把存储在C库(/lib/libc.so.6)中的目标代码连接进来。 这种方式使我们可以使用已能正确工作的其他人的函数,只关心我们所要解决的问题。 这就是为什么头文件中一般只含有数据和函数声明,而没有函数体。
如何使用gcc编译器的更多相关文章
- 在CentOS 7.2下升级gcc编译器的版本
默认情况下,CentOS 7.2预装的gcc版本是4.8.x,通过执行命令 gcc -v 可以看到,一般情况下这个版本的编译器已经满足需要了,但是某些特殊的时候为了支持C++更高的特性,需要对gcc编 ...
- C语言的编译过程、安装gcc编译器以及设置环境变量
以我对C语言编译过程的了解,我用了一点时间画了一个图,提供给大家参考一下,希望有些能对您的问题提上帮助. 前几天刚初步学习了C语言的编译过程,感触挺深的.在C语言中头文件其实起了一个很大的作用. 1. ...
- 利用GCC编译器生成动态链接库和静态链接库
转载请标明:http://www.cnblogs.com/winifred-tang94/ 1.编译过程 gcc –fPIC –c xxx.c 其中-fPIC是通知gcc编译器产生位置独立的目标代码. ...
- GCC编译器编译链接
在gcc编译器环境下,常见的文件扩展名的含义如下: .c:C源程序,经过预编译后的源程序也为.c文件,它可以通过-E参数输出. .h:头文件 .s:经过编译得到的汇编程序代码,它可以通过-S参数输出. ...
- Linux安装gcc编译器详解
本人使用的是CentOS 6.5 64位系统,由于在安装系统的时候并没有勾选安装gcc编译器,因此需要自行安装gcc编译器. 使用yum安装gcc 对于配备了yum的Linux发行版而言,安装gcc编 ...
- GCC编译器使用
一.GCC简介 通常所说的GCC是GUN Compiler Collection的简称,除了编译程序之外,它还含其他相关工具,所以它能把易于人类使用的高级语言编写的源代码构建成计算机能够直接执行的二进 ...
- GCC编译器和GDB调试器常用选项
http://blog.csdn.net/u014328976/article/details/46745349 GCC编译器 gcc hello.c -o hello ...
- 临时改GCC编译器,重启后失效
临时改GCC编译器,重启后失效.例如,用如下命令: export CROSS_COMPILE= <gcc 文件所在的目录>/arm-linux-gnueabihf- 本例中使用的命令如下: ...
- windows下安装,配置gcc编译器
在Windows下使用gcc编译器: 1.首先介绍下MinGW MinGW是指仅仅用自由软件来生成纯粹的Win32可运行文件的编译环境,它是Minimalist GNU on Windows的略称. ...
- gcc编译器与基本类型3
C语言发展史 1969年贝尔实验室 肯尼斯·蓝·汤普逊,丹尼斯·李奇开发了B语言 ->Unix,New B语言,改名C语言83年提出C语言标准 1989年十二月正式通过C语言标准,C89标准 C ...
随机推荐
- 第二个App“今日美文”上架【原】
App store 下载地址 开发这个App的本意 之前偶然找到一个叫<每日一文>的应用,正是我一直想找的,优点如下: 界面够简单 推荐的文章也很好,而且都不太长 每天都不一样 但是用起来 ...
- Linux创建公钥
A:192.168.1.1 B:192.168.1.2 现在想让A无密码登陆B机器 A上运行以下命令来生成公钥和私钥 ssh-keygen -t rsa -P '' 运行该命令后会生成如下两个文件 ...
- Hadoop中Combiner的作用
1.Partition 把 Map任务输出的中间结果按 key的范围划分成 R份( R是预先定义的 Reduce任务的个数),划分时通常使用hash函数如: hash(key) mod R,这样可以保 ...
- CA1060
Move P/Invokes to NativeMethods class 规则描述: 平台调用服务访问非托管代码. 平台调用方法(使用了System.Runtime.InteropServices. ...
- RecyclerView设置verticalSapcing等
RecyclerView没有像GridView那样有提供verticalSpacing属性,上StackOverflow找到了一种替代方法,代码如下 public class SpacesItemDe ...
- 【每天一个Linux命令】10. 用户账号的新建/修改/删除以及密码修改 useradd/usemod/userdel/passwd
在 Linux 系统中,与用户管理有关的文件主要有如下几个:分别是/etc/passwd,/etc/shadow,/etc/gfoup,/etc/gshadow .它们分别与用户的账号,密码,用户组及 ...
- String的那点小事儿
1.== 比较的是什么? 1 package xupengwei.string; 2 /** 3 * @describe: 4 * @author chenmo-xpw 5 * @version 20 ...
- Flume + HDFS + Hive日志收集系统
最近一段时间,负责公司的产品日志埋点与收集工作,搭建了基于Flume+HDFS+Hive日志搜集系统. 一.日志搜集系统架构: 简单画了一下日志搜集系统的架构图,可以看出,flume承担了agent与 ...
- 【不积跬步,无以致千里】mysql 多行合并函数
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- C++ Primer 学习笔记_76_模板与泛型编程 --模板定义[续]
模板与泛型编程 --模板定义[续] 四.模板类型形參 类型形參由keywordclass或 typename后接说明符构成.在模板形參表中,这两个keyword具有同样的含义,都指出后面所接的名字表示 ...