<?xml version="1.0" encoding="UTF-8" ?>
<rss version="2.0">
<channel>
<title><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></title> 
<link>http://www.jackxiang.com/index.php</link> 
<description><![CDATA[赢在IT，Playin' with IT,Focus on Killer Application,Marketing Meets Technology.]]></description> 
<language>zh-cn</language> 
<copyright><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></copyright>
<item>
<link>http://www.jackxiang.com/post//</link>
<title><![CDATA[[转贴精华]Linux平台gcc和动态共享库的基础知识]]></title> 
<author>jack &lt;xdy108@126.com&gt;</author>
<category><![CDATA[WEB2.0]]></category>
<pubDate>Wed, 17 Dec 2008 08:09:31 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//</guid> 
<description>
<![CDATA[ 
	一、GNU gcc的编译工具用法<br/><br/>我们先来写一个简单的C程序：hello.c<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2.&nbsp;&nbsp; <br/>&nbsp;&nbsp; 3. void print_hello() &#123;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 4.&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;Hello World&#92;n&quot;);&nbsp;&nbsp;<br/>&nbsp;&nbsp; 5. &#125;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 6.&nbsp;&nbsp; <br/>&nbsp;&nbsp; 7. int main(int argc, char argv&#91;&#93;) &#123;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 8.&nbsp;&nbsp;&nbsp;&nbsp;print_hello();&nbsp;&nbsp;<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp; return 0;&nbsp;&nbsp;<br/>&nbsp;&nbsp;10. &#125;&nbsp;&nbsp;</div><br/><br/>定义了一个print_hello函数，调用main函数打印Hello World。 如何编译它呢？<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. gcc -o hello -O2 hello.c&nbsp;&nbsp;</div><br/><br/><br/>-o参数指定生成的可执行程序的文件名， -O2是优化级别。该命令会编译生成hello可执行程序，看看这个文件：ls -l hello<br/>C代码<br/><br/>&nbsp;&nbsp; 1. -rwxr-xr-x&nbsp;&nbsp; 1 robbin users 11939 2008-11-02 13:48 hello&nbsp;&nbsp;<br/><br/><br/>有11KB大小。<br/><br/>看看他链接了哪些系统动态链接库，用ldd命令：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. ldd hello&nbsp;&nbsp;</div><br/><br/><br/>输出信息为：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. libc.so.6 =&gt; /lib64/tls/libc.so.6 (0x0000002a9566d000)&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2. /lib64/ld-linux-x86-64.so.2 (0x0000002a95556000)&nbsp;&nbsp;</div><br/><br/><br/>libc是C语言标准函数库，ld是动态链接器。<br/><br/>接着我们看看hello这个程序里面有哪些符号，用nm命令：<br/>hello有11KB，体积偏大，去处符号表可以给它瘦身，我们用strip命令：<br/>最后如果我们想从可执行程序里面提取出来一点什么文本信息的话，还可以用strings命令：<br/>C代码<br/><br/>&nbsp;&nbsp; 1. strings hello&nbsp;&nbsp;<br/><br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2.&nbsp;&nbsp; <br/>&nbsp;&nbsp; 3. void print_hello() &#123;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 4.&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;Hello World&#92;n&quot;);&nbsp;&nbsp;<br/>&nbsp;&nbsp; 5. &#125;&nbsp;&nbsp;</div><br/><br/>而main.c的代码是：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. int main(int argc, char argv&#91;&#93;) &#123;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2.&nbsp;&nbsp;&nbsp;&nbsp;print_hello();&nbsp;&nbsp;<br/>&nbsp;&nbsp; 3.&nbsp;&nbsp; return 0;&nbsp;&nbsp;<br/>&nbsp;&nbsp; 4. &#125;&nbsp;&nbsp;</div><br/><br/>hello.c是我们的动态共享库，在hello.c里面我们声明和实现了各种公用的函数，最后main.c可以去调用这些公用函数。首先我们要把hello.c编译成为动态共享库：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. gcc -o libhello.so -O2 -fPIC -shared hello.c&nbsp;&nbsp; </div><br/><br/><br/>-fPIC参数声明链接库的代码段是可以共享的，-shared参数声明编译为共享库。请注意这次我们编译的共享库的名字叫做libhello.so，这也是Linux共享库的一个命名的惯例了：后缀使用so，而名称使用libxxxx格式。<br/><br/>然后编译main.c的时候，我们需要更多的参数让gcc知道如何寻找共享库：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. gcc -o main -O2 -L. -lhello main.c&nbsp;&nbsp; </div><br/><br/>引用<br/>./main: error while loading shared libraries: libhello.so: cannot open shared object file: No such file or directory<br/><br/><br/><br/>找不到libhello.so这个共享库，怎么回事？这是因为libhello.so并不在操作系统默认的共享库的路径下面，我们可以临时指定一下链接路径：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH&nbsp;&nbsp;</div><br/><br/>这次main程序链接到了libhello.so这个共享库上面。<br/><br/>三、关于Linux的动态共享库的设置<br/><br/>可执行程序找不到要链接的动态共享库，这是Linux上面编译和运行程序很容易碰到的问题，通过上面的小例子，我们已经大致了解共享库的一点基本原理，接下来我们要探讨一下怎么设置程序寻找动态共享库的行为。<br/><br/>Linux操作系统上面的动态共享库大致分为三类：<br/><br/>1、操作系统级别的共享库和基础的系统工具库<br/><br/>比方说libc.so, libz.so, libpthread.so等等，这些系统库会被放在/lib和/usr/lib目录下面，如果是64位操作系统，还会有/lib64和/usr /lib64目录。如果操作系统带有图形界面，那么还会有/usr/X11R6/lib目录，如果是64位操作系统，还有/usr/X11R6 /lib64目录。此外还可能有其他特定Linux版本的系统库目录。<br/><br/>这些系统库文件的完整和版本的正确，确保了Linux上面各种程序能够正常的运行。<br/><br/>2、应用程序级别的系统共享库<br/><br/>并非操作系统自带，但是可能被很多应用程序所共享的库，一般会被放在/usr/local/lib和/usr/local/lib64这两个目录下面。很多你自行编译安装的程序都会在编译的时候自动把/usr/local/lib加入gcc的-L参数，而在运行的时候自动到/usr/local/lib下面去寻找共享库。<br/><br/>以上两类的动态共享库，应用程序会自动寻找到他们，并不需要你额外的设置和担心。这是为什么呢？因为以上这些目录默认就被加入到动态链接程序的搜索路径里面了。Linux的系统共享库搜索路径定义在/etc/ld.so.conf这个配置文件里面。这个文件的内容格式大致如下：<br/><br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. /usr/X11R6/lib64&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2. /usr/X11R6/lib&nbsp;&nbsp;<br/>&nbsp;&nbsp; 3. /usr/local/lib&nbsp;&nbsp;<br/>&nbsp;&nbsp; 4. /lib64&nbsp;&nbsp;<br/>&nbsp;&nbsp; 5. /lib&nbsp;&nbsp;<br/>&nbsp;&nbsp; 6. /usr/lib64&nbsp;&nbsp;<br/>&nbsp;&nbsp; 7. /usr/lib&nbsp;&nbsp;<br/>&nbsp;&nbsp; 8. /usr/local/lib64&nbsp;&nbsp;<br/>&nbsp;&nbsp; 9. /usr/local/ImageMagick/lib&nbsp;&nbsp;</div><br/><br/><br/><br/>假设我们自己编译安装的ImageMagick图形库在/usr/local/ImageMagick目录下面，并且希望其他应用程序都可以使用 ImageMagick的动态共享库，那么我们只需要把/usr/local/ImageMagick/lib目录加入/etc/ld.so.conf文件里面，然后执行：ldconfig 命令即可。<br/><br/>ldcofig将搜索以上所有的目录，为共享库建立一个缓存文件/etc/ld.so.cache。为了确认ldconfig已经搜索到ImageMagick的库，我们可以用上面介绍的strings命令从ld.so.cache里面抽取文本信息来检查一下：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. strings /etc/ld.so.cache &#124; grep ImageMagick </div>&nbsp;&nbsp;<br/><br/><br/>输出结果为：<br/>C代码<br/><br/><br/><div class="code">&nbsp;&nbsp; 1. /usr/local/ImageMagick/lib/libWand.so.10&nbsp;&nbsp;<br/>&nbsp;&nbsp; 2. /usr/local/ImageMagick/lib/libWand.so&nbsp;&nbsp;<br/>&nbsp;&nbsp; 3. /usr/local/ImageMagick/lib/libMagick.so.10&nbsp;&nbsp;<br/>&nbsp;&nbsp; 4. /usr/local/ImageMagick/lib/libMagick.so&nbsp;&nbsp;<br/>&nbsp;&nbsp; 5. /usr/local/ImageMagick/lib/libMagick++.so.10&nbsp;&nbsp;<br/>&nbsp;&nbsp; 6. /usr/local/ImageMagick/lib/libMagick++.so&nbsp;&nbsp;</div><br/><br/>已经成功了！<br/><br/>3、应用程序独享的动态共享库<br/><br/>有很多共享库只被特定的应用程序使用，那么就没有必要加入系统库路径，以免应用程序的共享库之间发生版本冲突。因此Linux还可以通过设置环境变量 LD_LIBRARY_PATH来临时指定应用程序的共享库搜索路径，就像我们上面举的那个例子一样，我们可以在应用程序的启动脚本里面预先设置 LD_LIBRARY_PATH，指定本应用程序附加的共享库搜索路径，从而让应用程序找到它。<br/>
]]>
</description>
</item><item>
<link>http://www.jackxiang.com/post//#blogcomment</link>
<title><![CDATA[[评论] [转贴精华]Linux平台gcc和动态共享库的基础知识]]></title> 
<author> &lt;user@domain.com&gt;</author>
<category><![CDATA[评论]]></category>
<pubDate>Thu, 01 Jan 1970 00:00:00 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//#blogcomment</guid> 
<description>
<![CDATA[ 
	
]]>
</description>
</item>
</channel>
</rss>