如何在pr 照片全选为什么点不动pr 右键 中英 翻译

如何在linux下用GCC编译.c和.h文件
如何在linux下用GCC编译.c和.h文件
最近下载了《C语言的科学和艺术》这本书的库函数,里面包括一些诸如:genlib.c、genlib.h等带.c和.h的文件。当然除了这些外包中还有,config.csh、cslib.shar、Makefile等这几个文件。
由于如果要运行书中的代码必须使用库函数#include &genlib.h &,所以必须对源文件进行编译。
还请各位同仁给出命令详细过程。
昵称: msnlogo &时间:
昵称: dai_weitao &时间:
昵称: yangb_soso &时间:
昵称: daocaoren0 &时间:
昵称: msnlogo &时间:
昵称: caipjin &时间:【转载】详解GCC的下载和安装(源码安装) - zhiyinjixu - 博客园
下载:&浏览:&查看Changes:&
现在很多程序员都应用GCC,怎样才能更好的应用GCC。目前,GCC可以用来编译C/C++、FORTRAN、JAVA、OBJC、ADA等语言的程序,可根据需要选择安装支持的语言。本文以在Redhat Linux安装GCC4.1.2为例(因在项目开发过程中要求使用,没有用最新的GCC版本),介绍Linux安装GCC过程。
安装之前,系统中必须要有cc或者gcc等编译器,并且是可用的,或者用环境变量CC指定系统上的编译器。如果系统上没有编译器,不能安装源代码形式的GCC 4.1.2。如果是这种情况,可以在网上找一个与你系统相适应的如RPM等二进制形式的GCC软件包来安装使用。本文介绍的是以源代码形式提供的GCC软件包的安装过程,软件包本身和其安装过程同样适用于其它Linux和Unix系统。
系统上原来的GCC编译器可能是把gcc等命令文件、库文件、头文件等分别存放到系统中的不同目录下的。与此不同,现在GCC建议我们将一个版本的GCC安装在一个单独的目录下。这样做的好处是将来不需要它的时候可以方便地删除整个目录即可(因为GCC没有uninstall功能);缺点是在安装完成后要做一些设置工作才能使编译器工作正常。在本文中采用这个方案安装GCC 4.1.2,并且在安装完成后,仍然能够使用原来低版本的GCC编译器,即一个系统上可以同时存在并使用多个版本的GCC编译器。
按照本文提供的步骤和设置选项,即使以前没有安装过GCC,也可以在系统上安装上一个可工作的新版本的GCC编译器。
在GCC网站上(http://gcc.gnu.org)或者通过网上搜索可以查找到下载资源。目前GCC的最新版本为 4.2.1。可供下载的文件一般有两种形式:gcc-4.1.2.tar.gz和gcc-4.1.2.tar.bz2,只是压缩格式不一样,内容完全一致,下载其中一种即可。
拷贝gcc-4.1.2.tar.bz2(我下载的压缩文件)到/usr/local/src(根据自己喜好选择)下,根据压缩格式,选择下面相应的一种方式解包(以下的&%&表示命令行提示符):
% tar zxvf gcc-4.1.2.tar.gz
% bzcat gcc-4.1.2.tar.bz2 | tar xvf -
新生成的gcc-4.1.2这个目录被称为源目录,用${srcdir}表示它。以后在出现${srcdir}的地方,应该用真实的路径来替换它。用pwd命令可以查看当前路径。
在${srcdir}/INSTALL目录下有详细的GCC安装说明,可用浏览器打开index.html阅读。
3. 建立目标目录
目标目录(用${objdir}表示)是用来存放编译结果的地方。GCC建议编译后的文件不要放在源目录${srcdir]中(虽然这样做也可以),最好单独存放在另外一个目录中,而且不能是${srcdir}的子目录。
例如,可以这样建立一个叫 /usr/local/gcc-4.1.2的目标目录:
% mkdir /usr/local/gcc-4.1.2
% cd gcc-4.1.2
以下的操作主要是在目标目录 ${objdir} 下进行。(否则会出错,后面有解释)
配置的目的是决定将GCC编译器安装到什么地方(${destdir}),支持什么语言以及指定其它一些选项等。其中,${destdir}不能与${objdir}或${srcdir}目录相同。
配置是通过执行${srcdir}下的configure来完成的。其命令格式为(记得用你的真实路径替换${destdir}):
% ${srcdir}/configure --prefix=${destdir} [其它选项]
例如,如果想将GCC 4.1.2安装到/usr/local/gcc-4.1.2目录下,则${destdir}就表示这个路径。
在我的机器上,我是这样配置的:
% ../gcc-4.1.2/configure --prefix=/usr/local/gcc-4.1.2 --enable-threads=posix --disable-checking --enable--long-long --host=i386-redhat-linux--with-system-zlib --enable-languages=c,c++,java
将GCC安装在/usr/local/gcc-4.1.2目录下,支持C/C++和JAVA语言,其它选项参见GCC提供的帮助说明。
执行下面的命令将编译好的库文件等拷贝到${destdir}目录中(根据你设定的路径,可能需要管理员的权限):
% make install
至此,GCC 4.1.2安装过程就完成了。
7. 其它设置
GCC 4.1.2的所有文件,包括命令文件(如gcc、g++)、库文件等都在${destdir}目录下分别存放,如命令文件放在bin目录下、库文件在 lib下、头文件在include下等。由于命令文件和库文件所在的目录还没有包含在相应的搜索路径内,所以必须要作适当的设置之后编译器才能顺利地找到并使用它们。
7.1 gcc、g++、gcj的设置
要想使用GCC 4.1.2的gcc等命令,简单的方法就是把它的路径${destdir}/bin放在环境变量PATH中。我不用这种方式,而是用符号连接的方式实现,这样做的好处是我仍然可以使用系统上原来的旧版本的GCC编译器。
首先,查看原来的gcc所在的路径:
% which gcc
在我的系统上,上述命令显示:/usr/bin/gcc。因此,原来的gcc命令在/usr/bin目录下。我们可以把GCC 4.1.2中的gcc、g++、gcj等命令在/usr/bin目录下分别做一个符号连接:
% cd /usr/bin
% ln -s ${destdir}/bin/gcc gcc412
% ln -s ${destdir}/bin/g++ g++412
% ln -s ${destdir}/bin/gcj gcj412
这样,就可以分别使用gcc412、g++412、gcj412来调用GCC 4.1.2的gcc、g++、gcj完成对C、C++、JAVA程序的编译了。同时,仍然能够使用旧版本的GCC编译器中的gcc、g++等命令。
(cool,我感觉棒极了!!1)
7.2 库路径的设置
将${destdir}/lib路径添加到环境变量LD_LIBRARY_PATH中,例如,如果GCC 4.1.2安装在/usr/local/gcc-4.1.2目录下,在RH Linux下可以直接在命令行上执行
% export LD_LIBRARY_PATH=/usr/local/gcc-4.1.2/lib
最好添加到系统的配置文件中,这样就不必要每次都设置这个环境变量了,在文件$HOME/.bash_profile中添加下面两句:
LD_LIBRARY_PATH=/usr/local/gcc-4.1.2/lib:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH
重启系统设置生效,或者执行命令
% source $HOME/.bash_profile
用新的编译命令(gcc412、g++412等)编译你以前的C、C++程序,检验新安装的GCC编译器是否能正常工作。
完成了Linux安装GCC,之后你就能轻松地编辑了。
from:/art/804.htm
在RHLinux下安装gcc-4.0.1方法比较简单,但是安装过程中有些环节是需要注意的,否则,可能会导致安装不成功,或者安装报错。具体安装过程如下:
首先,下载并解压缩gcc的RPM包至源目录(如/opt/gcc-4.0.1)
1、解压缩RPM包:
[root@linuxopt]# tar xjvf gcc-4.0.1.tar.bz2 (解压后生成源目录/opt/gcc-4.0.1)
2、创建安装目标目录:
[root@linux opt]# mkdir /usr/local/gcc-4.0.1/
3、进入安装目标目录:
[root@linux opt]# cd /usr/local/gcc-4.0.1/ (这一步很重要,配置安装文件时,需要在目标目录下执行configure命令)
[root@linux opt]# pwd
/usr/local/gcc-4.0.1
4、配置安装文件:
[root@linux gcc-4.0.1]# /opt/gcc-4.0.1/configure --prefix=/usr/local/gcc-4.0.1/ (这一步非常重要,需要在安装的目标目录下,执行源目录 /opt/gcc-4.0.1/中的configure命令,配置将gcc安装到目标目录/usr/local/gcc-4.0.1/)
creating cache ./config.cache
checking host system type... i686-pc-linux-gnu
5、编译安装文件:
[root@linux gcc-4.0.1]# pwd
/usr/local/gcc-4.0.1
[root@linux gcc-4.0.1]# make (在目标目录下执行编译)
6、安装gcc:
[root@linux gcc-4.0.1]# pwd
/usr/local/gcc-4.0.1
[root@linux gcc-4.0.1]# make install (在目标目录下执行安装)
如果安装过程中步骤和命令没有错误,你肯定能安装成功。
---------------------------------------------------------------------
  首先,现在最新的包GCC 4.2.bz2,一般的到处都有的下,linux下的下载速度很满,20多k,很恶心,我喜欢迅雷的下载速度,在windows下下载,大概42M,下载速度2到4M,回到linux下,挂载,ntfs格式的我的,具体不说了,说安装!&  进入到挂载的目录下,先cp GCC4.2.bz2 /azuo,&  cd /azuo,&  tar -xvf GCC 4.2.bz2 ,&  得到gcc-4.2,&  cd gcc-4.2,在/usr目录下建立一个文件夹就是现在要存放新的gcc的目录,&  mkdir /usr/gcc4&  ./configure &prefix=/usr/gcc4&  回车,就会有配置信息,只要不报错就可以了,&  make,这个过程很久,因为我没有设置一些具体的选项,因此,所有的组件几乎都要编译一遍,我的电脑cpu:AMD 3200+X2 ,1.5g内存,大概花费一个半小时。&  到这个时候为止,/usr/gcc4下还没有任何东西,编译过程都是发生在源文件夹,让我们再来一个动作,所有的要用到的东西都会配置到目标文件夹下,&  make install;&  这个过程也不短,等着就是了。&  一切都已经弄好了之后就是使用最新的gcc了,可以看到,在/usr/gcc4/bin下有gcc,g++,等一些东西,都是可以用的,写两个程序:&  aa.c:&  1 #include&  2&  3 int main(void) {&  4 printf("ad");&  5 return 1;&  6 }&  gcc -o aa aa.c&  执行 ./aa&  上面的这个是c的,下面这个就是c++的了:&  a.cpp:&  1 #include&  2&  3 int main(void) {&  4 cout && "Ok!" &&&  5 return 1;&  6 }&  g++ -o a a.cpp&  执行就是了./a&  一切ok了,就可以让最新的gcc工具取代原来的工具了。&  看看原来的gcc是什么版本的,我们好卸载它:&  [root@BTazuo bin]# rpm -qa gcc&  gcc-4.1.2-27.fc7&  [root@BTazuo bin]# rpm -e gcc-4.1.2-27.fc7&  error: Failed dependencies:&  gcc is needed by (installed) systemtap-0.5.13-1.fc7.i386&  gcc = 4.1.2-27.fc7 is needed by (installed) gcc-c++-4.1.2-27.fc7.i386&  gcc = 4.1.2-27.fc7 is needed by (installed) gcc-gfortran- 4.1.2 -27.fc7. i386&  [root@BTazuo bin]# rpm -e gcc-c++-4.1.2-27.fc7.i386&  [root@BTazuo bin]# rpm -e gcc-gfortran-4.1.2-27.fc7.i386&  [root@BTazuo bin]# rpm -e gcc-4.1.2-27.fc7&  error: Failed dependencies:&  gcc is needed by (installed) systemtap-0.5.13-1.fc7.i386&  [root@BTazuo bin]# g++&  bash: g++: command not found&  卸载成功&  [root@BTazuo bin]# gcc&  gcc: 没有输入文件 ,可见gcc犹在&  [root@BTazuo bin]# rpm -e systemtap-0.5.13-1.fc7.i386&  [root@BTazuo bin]# gcc&  gcc: 没有输入文件&  [root@BTazuo bin]# rpm -e gcc-4.1.2-27.fc7&  [root@BTazuo bin]# gcc&  bash: /usr/lib/ccache/gcc: 没有那个文件或目录&  最后的卸载成功!&  这个时候,要注意了,我的gcc在/usr/bin下面有,在/usr/lib/ccache这个目录下也有,分别在这两个下面都要建立一个链接:&  [root@BTazuo bin]# ln -s /usr/gcc4/bin/g++ g++&  [root@BTazuo bin]# g++&  g++: 没有输入文件&  可见g++已经装好了,可以使用了。&  然后是gcc:&  [root@BTazuo bin]# ln -s /usr/gcc4/bin/gcc gcc&  [root@BTazuo bin]# gcc&  bash: /usr/lib/ccache/gcc: 没有那个文件或目录&  [root@BTazuo bin]# ./gcc&  gcc: 没有输入文件&  可见还要给另外一个目录建立一个gcc的链接:&  [root@BTazuo lib]# ln -s /usr/gcc4/bin/gcc /usr/lib/ccache/gcc&  [root@BTazuo lib]# gcc&  gcc: 没有输入文件&  到此为止,gcc和g++都已经建立好了,可以用了,最后把原来的包和解压文件都可以删除了,以节省硬盘空间!
-------------------------------------------------------------------------------------------------------
下面介绍其在Red Hat Linux 9.0编译器给gcc 3.2.2环境下的安装方法。一.确定安装环境本安装方法适用于Red Hat Linux 9.0操作系统,自带的GCC编译器是GCC 3.2.2版本。其他系列的linux操作系统或是其他版本GCC下安装过程可能有些细节上的不同。注:因为不同版本的GCC编译器下一些库的定义可能有不同,因此首先要确定一下本机的GCC编译器版本。确定方法是进入命令行输入命令gcc &v得到的结果如图1.1所示:图1.1可见本机的GCC版本为GCC 3.2.2,根据simplescalar网站上的一些介绍,该模拟器的开发工具可能是GCC2.7左右,比较接近GCC 3.2.2,因此估计安装过程会比较顺利。二.获得安装包完成本安装过程的安装包可以在下下载,本安装所需要的安装包共有以下三个三 建立安装目录,解压安装包建立安装目录为/root/simplescalar,将安装包复制到安装目录,整个过程如下:解压缩,命令为tar &zxvf,加压缩完毕后用rm*.tgz命令删除压缩包,整个过程输入命令如下:tar &zxvf simplesim-3v0d.tgztar &zxvf simpletools-2v0.tgztar &zxvf simpleutils-2v0.tgzrm*.tgz加压缩后得到如下七个文件夹:四.安装binutils2.5.2首先用configure命令配置程序的安装环境和参数,生成Makefile文件,整个过程如下:注:configure命令的参数含义说明-host:配置安装环境-target:配置成littleEndian模式-with-gnu-as 加载汇编器-with-gnu-ld 加载链接器-prefix 设置安装目录此时遇到两个错误,如下所示:&&从错误说明可以估计出错误来自于libiberty文件夹下的dummy.c文件。网上一些帖子说是dummy.c中定义的宏functions.def文件中的函数定义与声明不一致引起的,将它们改成一致就可以通过编译。但是这样做的话,继续编译依然会遇到许多错误。这里我试出了最好的方法就是将dummy.c文件中的内容全部删除(即将dummy.c变成空文件)然后再make一次,这回不报任何错误,编译一次通过!运行make install命令,这时binutils-2.5.2安装成功!五.安装simplescalarSimplescalar是最简单的一个安装过程,运行下列命令即可完成安装!六.安装gcc-2.6.3安装好simplescalar后在安装文件夹/root/simplescalar下可以找到一个名为bin的文件夹,里面包含的是一些simplescalar自带的工具,如链接工具等,该文件内容如下:可见,里面没有C编译工具gcc,因此还需要安装gcc2.6.3作为simplescalar的内置编译工具,下面介绍安装方法。首先对安装环境进行配置,生成Makefile文件,具体方法如下图所示。Makefile文件生成完毕后,运行make命令,此时出现以下错误:由错误报告可知,这是由于sys_errlist的定义不一致造成的,打开cccp.c文件发现其194行附近有如下代码,第194行为extern char *sys_errlist[]。显然要消除冲突,只需改变宏编译的分支方向,使其不走这一分支即可。尝试在这段代码前面如175行加上#define bsd4_4,修改后再次make,该错误改正,遇到下一个错误。&&第二个错误是sdbout.c文件中的一些常量没有定义,如下:因此第一估计是头文件的问题,打开sdbout.c发现其包含的头文件有如下几个:&可以肯定报错的原因是宏编译的分支的问题(走不同的宏编译分支,可能包含syms.h或者是gsyms.h)。经过多次尝试,发现在前面加上#undef&&USG即可解决该错误(即把gsyms.h包含进去)。继续make,发生第三个错误,如下:这个错误和第一个错误一样,是由于sys_errlist定义冲突引起的,因此进入gcc.c文件,在172行之前(这里加在167行)加上#define bsd4_4即可。继续make,发生第四个错误,如下:和前面一样,只要在g++.c文件的第90行代码段之前(这里加在85行)加上#define bsd4_4即可。继续make,发生第五个错误,如下:&这里提示是将cp/g++.c文件中第213行的sys_errlist改成strerror或者strerror_r,然而改过之后会报strerror未定义的错误,上网搜了很久也没发现有这两个定义的头文件名,最后根据函数名感觉这段代码(pfatal_with_name)的功能可能是获得错误名,将其删除估计对工作影响不大,因此干脆将这个函数放空,放空后再运行make,果然OK,不报任何错误,编译通过!运行make install,gcc 2.6.3安装成功!返回simplescalar,进入bin文件夹,可以发现里面多了一个sslittle-na-sstrix-gcc文件,该文件是simplescalar的内建C编译器,如下所示:进一步运行./sslittle-na-sstrix-gcc &v测试出该内建编译器版本为gcc 2.6.3。到此为止,整个安装过程结束,下面进行测试。七.测试为了测试simplescalar是否能够顺利运行,我们对其进行测试,测试程序依然采用最经典的hello world!程序,程序如下:& & #include&main(){printf("Hello World!\n");return 0;}编辑好程序后将其保存在/root/simplescalar文件夹下,文件名为hello.c,用刚刚安装的simplescalar内建编译器编译,编译方法如下。& & 运行结果如下:八 结束语本安装方法适用于Red Hat Linux 9.0操作系统,自带的GCC编译器是GCC 3.2.2版本。其他系列的linux操作系统或是其他版本GCC下安装过程可能有些细节上的不同,主要原因可能是高版本的GCC没有兼容低版本的一些库文件,还有就是遵循的C标准可能会有点出入。如果机器上的GCC正好是低版本的,估计可能一次编译通过!-----------------------------------------------------------------------------------------------------
  1. 下载&  在GCC网站上(http://gcc.gnu.org/)或通过网上搜索能查找到下载资源。目前GCC的最新版本为3.4.0。可供下载的文件一般有两种形式:gcc-3.4.0.tar.gz和gcc-3.4.0.tar.bz2,只是压缩格式不相同,内容完全一致,下载其中一种即可。&  2. 解压缩&  根据压缩格式,选择下面相应的一种方式解包(以下的"%"表示命令行提示符):&  % tar xzvf gcc-3.4.0.tar.gz或% bzcat gcc-3.4.0.tar.bz2 | tar xvf -&  新生成的gcc-3.4.0这个目录被称为源目录,用${srcdir}表示他。以后在出现${srcdir}的地方,应该用真实的路径来替换他。用pwd命令能查看当前路径。&  在${srcdir}/INSTALL目录下有周详的GCC安装说明,可用浏览器打开index.html阅读。&  3. 建立目标目录&  目标目录(用${objdir}表示)是用来存放编译结果的地方。GCC建议编译后的文件不要放在源目录${srcdir]中(虽然这样做也能),最佳独立存放在另外一个目录中,而且不能是${srcdir}的子目录。&  例如,能这样建立一个叫 gcc-build的目标目录(和源目录${srcdir}是同级目录):&  % mkdir gcc-build% cd gcc-build&  以下的操作主要是在目标目录 ${objdir} 下进行。&  4. 设置&  设置的目的是决定将GCC编译器安装到什么地方(${destdir}),支持什么语言及指定其他一些选项等。其中,${destdir}不能和${objdir}或${srcdir}目录相同。&  设置是通过执行${srcdir}下的configure来完成的。其命令格式为(记得用你的真实路径替换${destdir}):&  % ${srcdir}/configure --prefix=${destdir} [其他选项]&  例如,如果想将GCC3.4.0安装到/usr/local/gcc-3.4.0目录下,则${destdir}就表示这个路径。&  在我的机器上,我是这样设置的:&  % ../gcc-3.4.0/configure --prefix=/usr/local/gcc-3.4.0--enable-threads=posix --disable-checking --enable--long-long--host=i386-redhat-linux --with-system-zlib --enable-languages=c,c++,java&  将GCC安装在/usr/local/gcc-3.4.0目录下,支持C/C++和JAVA语言,其他选项参见GCC提供的帮助说明。&  5. 编译&  % make&  这是个漫长的过程。在我的机器上(P4-1.6),这个过程用了50多分钟。&  6. 安装&  执行下面的命令将编译好的库文件等拷贝到${destdir}目录中(根据你设定的路径,可能需要管理员的权限):&  % make install&  至此,GCC 3.4.0安装过程就完成了。&  6. 其他设置&  GCC3.4.0的所有文件,包括命令文件(如gcc、g++)、库文件等都在${destdir}目录下分别存放,如命令文件放在bin目录下、库文件在lib下、头文件在include下等。由于命令文件和库文件所在的目录还没有包含在相应的搜索路径内,所以必须要作适当的设置之后编译器才能顺利地找到并使用他们。&  6.1 gcc、g++、gcj的设置&  要想使用GCC3.4.0的gcc等命令,简单的方法就是把他的路径${destdir}/bin放在环境变量PATH中。我不用这种方式,而是用符号连接的方式实现,这样做的好处是我仍然能使用系统上原来的旧版本的GCC编译器。&  首先,查看原来的gcc所在的路径:&  % which gcc&  在我的系统上,上述命令显示:/usr/bin/gcc。因此,原来的gcc命令在/usr/bin目录下。我们能把GCC3.4.0中的gcc、g++、gcj等命令在/usr/bin目录下分别做一个符号连接:&  % cd /usr/bin% ln -s ${destdir}/bin/gcc gcc34% ln -s ${destdir}/bin/g++ g++34% ln -s ${destdir}/bin/gcj gcj34&  这样,就能分别使用gcc34、g++34、gcj34来调用GCC3.4.0的gcc、g++、gcj完成对C、C++、JAVA程式的编译了。同时,仍然能够使用旧版本的GCC编译器中的gcc、g++等命令。&  6.2 库路径的设置&  将${destdir}/lib路径添加到环境变量LD_LIBRARY_PATH中,最佳添加到系统的设置文件中,这样就不必要每次都设置这个环境变量了。&  例如,如果GCC 3.4.0安装在/usr/local/gcc-3.4.0目录下,在RHLinux下能直接在命令行上执行或在文件/etc/profile中添加下面一句:&  setenv LD_LIBRARY_PATH /usr/local/gcc-3.4.0/lib:$LD_LIBRARY_PATH&  7. 测试&  用新的编译命令(gcc34、g++34等)编译你以前的C、C++程式,检验新安装的GCC编译器是否能正常工作。&  8. 根据需要,能删除或保留${srcdir}和${objdir}目录。Gcc详解以及静态库、动态库生成
1。gcc包含的c/c++编译器gcc,cc,c++,g++,gcc和cc是一样的,c++和g++是一样的,(没有看太明白前面这半句是什么意思:))一般c程序就用gcc编译,c++程序就用g++编译2。gcc的基本用法gcc test.c这样将编译出一个名为a.out的程序gcc test.c -o test这样将编译出一个名为test的程序,-o参数用来指定生成程序的名字3。为什么会出现undefined reference to 'xxxxx'错误?首先这是链接错误,不是编译错误,也就是说如果只有这个错误,说明你的程序源码本身没有问题,是你用编译器编译时参数用得不对,你没有指定链接程序要用到得库,比如你的程序里用到了一些数学函数,那么你就要在编译参数里指定程序要链接数学库,方法是在编译命令行里加入-lm。4。-l参数和-L参数-l参数就是用来指定程序要链接的库,-l参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是m,他的库文件名是libm.so,很容易看出,把库文件名的头lib和尾.so去掉就是库名了。好了现在我们知道怎么得到库名了,比如我们自已要用到一个第三方提供的库名字叫libtest.so,那么我们只要把libtest.so拷贝到/usr/lib里,编译时加上-ltest参数,我们就能用上libtest.so库了(当然要用libtest.so库里的函数,我们还需要与libtest.so配套的头文件)。放在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接了,但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用-l参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”,也就是链接程序ld在那3个目录里找不到libxxx.so,这时另外一个参数-L就派上用场了,比如常用的X11的库,它放在/usr/X11R6/lib目录下,我们编译时就要用-L/usr/X11R6/lib -lX11参数,-L参数跟着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb/ccc -ltest另外,大部分libxxxx.so只是一个链接,以RH9为例,比如libm.so它链接到/lib/libm.so.x,/lib/libm.so.6又链接到/lib/libm-2.3.2.so,如果没有这样的链接,还是会出错,因为ld只会找libxxxx.so,所以如果你要用到xxxx库,而只有libxxxx.so.x或者libxxxx-x.x.x.so,做一个链接就可以了ln -s libxxxx-x.x.x.so libxxxx.so手工来写链接参数总是很麻烦的,还好很多库开发包提供了生成链接参数的程序,名字一般叫xxxx-config,一般放在/usr/bin目录下,比如gtk1.2的链接参数生成程序是gtk-config,执行gtk-config --libs就能得到以下输出"-L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic -lgmodule -lglib -ldl -lXi -lXext -lX11 -lm",这就是编译一个gtk1.2程序所需的gtk链接参数,xxx-config除了--libs参数外还有一个参数是--cflags用来生成头文件包含目录的,也就是-I参数,在下面我们将会讲到。你可以试试执行gtk-config --libs --cflags,看看输出结果。现在的问题就是怎样用这些输出结果了,最笨的方法就是复制粘贴或者照抄,聪明的办法是在编译命令行里加入这个`xxxx-config --libs --cflags`,比如编译一个gtk程序:gcc gtktest.c `gtk-config --libs --cflags`这样就差不多了。注意`不是单引号,而是1键左边那个键。除了xxx-config以外,现在新的开发包一般都用pkg-config来生成链接参数,使用方法跟xxx-config类似,但xxx-config是针对特定的开发包,但pkg-config包含很多开发包的链接参数的生成,用pkg-config --list-all命令可以列出所支持的所有开发包,pkg-config的用法就是pkg-config pagName --libs --cflags,其中pagName是包名,是pkg-config--list-all里列出名单中的一个,比如gtk1.2的名字就是gtk+,pkg-config gtk+ --libs --cflags的作用跟gtk-config --libs --cflags是一样的。比如:gcc gtktest.c `pkg-config gtk+ --libs --cflags`。5。-include和-I参数-include用来包含头文件,但一般情况下包含头文件都在源码里用#include xxxxxx实现,-include参数很少用。-I参数是用来指定头文件目录,/usr/include目录一般是不用指定的,gcc知道去那里找,但是如果头文件不在/usr/include里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude参数了,如果不加你会得到一个"xxxx.h: No such file or directory"的错误。-I参数可以用相对路径,比如头文件在当前目录,可以用-I.来指定。上面我们提到的--cflags参数就是用来生成-I参数的。6。-O参数这是一个程序优化参数,一般用-O2就是,用来优化程序用的,比如gcc test.c -O2,优化得到的程序比没优化的要小,执行速度可能也有所提高(我没有测试过)。7。-shared参数编译动态库时要用到,比如gcc -shared test.c -o libtest.so8。几个相关的环境变量PKG_CONFIG_PATH:用来指定pkg-config用到的pc文件的路径,默认是/usr/lib/pkgconfig,pc文件是文本文件,扩展名是.pc,里面定义开发包的安装路径,Libs参数和Cflags参数等等。CC:用来指定c编译器。CXX:用来指定cxx编译器。LIBS:跟上面的--libs作用差不多。CFLAGS:跟上面的--cflags作用差不多。CC,CXX,LIBS,CFLAGS手动编译时一般用不上,在做configure时有时用到,一般情况下不用管。环境变量设定方法:export ENV_NAME=xxxxxxxxxxxxxxxxx9。关于交叉编译交叉编译通俗地讲就是在一种平台上编译出能运行在体系结构不同的另一种平台上,比如在我们地PC平台(X86 CPU)上编译出能运行在sparc CPU平台上的程序,编译得到的程序在X86 CPU平台上是不能运行的,必须放到sparc CPU平台上才能运行。当然两个平台用的都是linux。这种方法在异平台移植和嵌入式开发时用得非常普遍。相对与交叉编译,我们平常做的编译就叫本地编译,也就是在当前平台编译,编译得到的程序也是在本地执行。用来编译这种程序的编译器就叫交叉编译器,相对来说,用来做本地编译的就叫本地编译器,一般用的都是gcc,但这种gcc跟本地的gcc编译器是不一样的,需要在编译gcc时用特定的configure参数才能得到支持交叉编译的gcc。为了不跟本地编译器混淆,交叉编译器的名字一般都有前缀,比如sparc-xxxx-linux-gnu-gcc,sparc-xxxx-linux-gnu-g++ 等等10。交叉编译器的使用方法使用方法跟本地的gcc差不多,但有一点特殊的是:必须用-L和-I参数指定编译器用sparc系统的库和头文件,不能用本地(X86)的库(头文件有时可以用本地的)。例子:sparc-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/sparcIncludeLinux下生成动态和静态库在用c写程序时,很多时候需要存储一些简单的数据,如果为此而用mysql数据库就有些大才小用了,可以把这些数据以结构的形写入文件,然后再需要时读取文件,取出数据。
如下是定义函数的源文件和头文件:
源文件struct.c:
#include "struct.h"
//第一个参数是要写入的文件名,第二个参数是缓冲区,第三个参数是缓冲区大小,第四个参数是打开文件流的形态,返回TRUE表示写入成功,返回FALSE表示写入失败
int writeStruct(const char *fileName,char *buffer,int bufferLen,char *mode){ FILE *fileID = NULL;
fileID = fopen(fileName,mode); if (fileID == NULL){ perror("fopen"); goto writeE } rewind(fileID); ret = fwrite(buffer,bufferLen,1,fileID); if (ret &= 0){ perror("fwrite"); goto writeE } if (fileID != NULL){ fclose(fileID); fileID = NULL; } return TRUE;
writeEnd: if (fileID != NULL){ fclose(fileID); fileID = NULL; } return FALSE; }
//第一个参数是要读取的文件名,第二个参数是缓冲区,第三个参数是缓冲区大小,第四个参数是打开文件流的形态,返回TRUE表示读取成功,返回FALSE表示读取失败
int readStruct(const char *fileName,char *buffer,int bufferLen,char *mode){ FILE *fileID = NULL;
fileID = fopen(fileName,mode); if (fileID == NULL){ perror("fopen"); goto readE } rewind(fileID); memset(buffer,0,sizeof(buffer)); ret = fread(buffer,bufferLen,1,fileID); if (ret &= 0){ strcat(buffer,"\0"); }else{ perror("fread") ; goto readE } if (fileID != NULL){ fclose(fileID); fileID = NULL; } return TRUE;
readEnd: if (fileID != NULL){ fclose(fileID); fileID = NULL; } return FALSE; }
头文件struct.h:
#ifndef OWNSTRUCT_H_ #define OWNSTRUCT_H_
#include #include #include
#define FALSE 0 #define TRUE 1
//第一个参数是要写入的文件名,第二个参数是缓冲区,第三个参数是缓冲区大小,第四个参数是打开文件流的形态,返回TRUE表示写入成功,返回FALSE表示写入失败
int writeStruct(const char *fileName,char *buffer,int bufferLen,char *mode);
//第一个参数是要读取的文件名,第二个参数是缓冲区,第三个参数是缓冲区大小,第四个参数是打开文件流的形态,返回TRUE表示读取成功,返回FALSE表示读取失败
int readStruct(const char *fileName,char *buffer,int bufferLen,char *mode);
为了使用方便,可以把这两个函数接口定义为动态链接库或静态链接库。用动态链接库编译生成的可执行文件需调用.so文件方可正常运行,灵活但稍显麻烦;用静态链接库编译生成的可执行文件可直接运行,不用再调用如.so般的依赖库文件,简单但不灵活。
静态链接库:
1、编译生成目标文件
gcc -c struct.c
2、创建静态库
ar cqs libstruct.a struct.o (顺序不能乱)
3、链接静态链接库,生成可执行文件
gcc main.c -static -L. -lstruct -o main
动态链接库:
1、编译成动态链接库
gcc struct.c -fPIC -shared -o libstruct.so
2、链接动态链接库,生成可执行文件
gcc main.c -L. -lstruct -o main
3、设置库文件的环境路径
1)在bashrc或profile文件里用LD_LIBRARY_PATH定义,然后用source加载。
2)把库路径添加到ld.so.conf文件中,然后用ldconfig加载。
3)ldconfig /home/user/lib,仅能暂时性使用,若下次ldconfig时此目录下的动态链接库就不能被共享了。
gcc一些参数解析
-shared:指定生成动态链接库。
-static:指定生成静态链接库。
-fPIC:表示编译为位置独立的代码,用于编译共享库。目标文件需要创建成位置无关码,概念上就是在可执行程序装载它们的时候,它们可以放在可执行程序的内存里的任何地方。
-L.:表示要连接的库在当前目录中。
-l:指定链接时需要的动态库。编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称。
-Wall:生成所有警告信息。
-ggdb:此选项将尽可能的生成gdb的可以使用的调试信息。
-g:编译器在编译的时候产生调试信息。
-c:只激活预处理、编译和汇编,也就是把程序做成目标文件(.o文件)。
-Wl,options:把参数(options)传递给链接器ld。如果options中间有逗号,就将options分成多个选项,然后传递给链接程序。
TA的最新馆藏[转]&[转]&[转]&[转]&[转]&}

我要回帖

更多关于 pr右键不显示 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信