⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dso.html

📁 这个是我在web培训时老师提供的手册
💻 HTML
字号:
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
<meta name="keywords" content="Apache, 中文, 手册, 中文版, 中文手册, 中文版手册, 参考手册, 中文参考手册, 金步国" />
<meta name="description" content="Apache 2.2 中文版参考手册" />
<meta name="author" content="金步国" />
<link href="./style/css/manual-zip.css" rel="stylesheet" media="all" type="text/css" title="Main stylesheet" />
<link href="./style/css/manual-zip-100pc.css" rel="alternate stylesheet" media="all" type="text/css" title="No Sidebar - Default font size" />
<link href="./style/css/manual-print.css" rel="stylesheet" media="print" type="text/css" />
<title>动态共享对象(DSO)支持 - Apache 2.2 中文版参考手册</title>
</head>
<body id="manual-page">
<div id="page-header"><p class="menu"><a href="./mod/index.html">模块索引</a> | <a href="./mod/directives.html">指令索引</a> | <a href="./faq/index.html">常见问题</a> | <a href="./glossary.html">词汇表</a> | <a href="./sitemap.html">站点导航</a></p><p class="apache">Apache HTTP Server 版本2.2</p><img alt="" src="./images/feather.gif" /></div>
<div class="up"><a href="./index.html"><img title="&lt;-" alt="&lt;-" src="./images/left.gif" /></a></div>
<div id="path"><a href="http://www.apache.org/">Apache</a> &gt; <a href="http://httpd.apache.org/">HTTP Server</a> &gt; <a href="http://httpd.apache.org/docs/">文档</a> &gt; <a href="./index.html">版本2.2</a></div>

<div id="translation-info">   <a href="./translator_announcement.html#thanks">致谢</a> | <a href="./translator_announcement.html#announcement">译者声明</a> | 本篇译者:<a href="mailto:&#099;sfr&#0097;nk&#0064;&#099;itiz&#046;n&#0101;t">金步国</a> | 本篇译稿最后更新:2006年1月2日 | <a href="./translator_announcement.html#last_new">获取最新版本</a></div>
<div id="page-content"><div id="preamble"><h1>动态共享对象(DSO)支持</h1>


    <p>Apache HTTP服务器是一个模块化的软件,管理员可以通过选择服务器中包含的模块进行功能增减。模块可以在编译时被静态包含进<code class="program"><a href="./programs/httpd.html">httpd</a></code>二进制文件,也可以编译成独立于<code class="program"><a href="./programs/httpd.html">httpd</a></code>二进制文件的动态共享对象(DSO)。DSO模块可以与服务器一起编译,也可以用Apache扩展工具(<code class="program"><a href="./programs/apxs.html">apxs</a></code>)单独编译。</p>

    <p>本文阐述如何使用DSO模块及其工作原理。</p>
  </div>
	<div class="top"><a href="#page-header"><img alt="top" src="./images/up.gif" /></a></div>
<div class="section">
<h2><a name="implementation" id="implementation">实现</a></h2>

<table border="1" cellpadding="0" cellspacing="0" bordercolor="#AAAAAA" class="related">
<tr><th>相关模块</th><th>相关指令</th></tr>
<tr><td><ul><li><code class="module"><a href="./mod/mod_so.html">mod_so</a></code></li></ul></td><td><ul><li><code class="directive"><a href="./mod/mod_so.html#loadmodule">LoadModule</a></code></li></ul></td></tr>
</table>

    <p>Apache对独立模块的DSO支持是建立在只能被静态编译进Apache核心的<code class="module"><a href="./mod/mod_so.html">mod_so</a></code>基础之上的,这是<code class="module"><a href="./mod/core.html">core</a></code>以外唯一不能作为DSO存在的模块,而其他所有已发布的Apache模块,都可以通过<a href="install.html">安装文档</a>中阐述中的<code class="program"><a href="./programs/configure.html">编译选项</a></code> <code>--enable-<em>module</em>=shared</code> 被独立地编译成DSO并使之生效。一个被编译为<code>mod_foo.so</code>的DSO模块,可以在<code>httpd.conf</code>中使用<code class="module"><a href="./mod/mod_so.html">mod_so</a></code>的<code class="directive"><a href="./mod/mod_so.html#loadmodule">LoadModule</a></code>指令,在服务器启动或重新启动时被加载。</p>

    <p>新提供的支持程序<code class="program"><a href="./programs/apxs.html">apxs</a></code>(<dfn>APache
    eXtenSion</dfn>)可以在Apache源代码树<em>之外</em>编译基于DSO的模块,从而简化了Apache DSO模块的建立过程。其原理很简单:安装Apache时,<code class="program"><a href="./programs/configure.html">configure</a></code>的 <code>make install</code> 命令会安装Apache C头文件,并把依赖于特定平台的编译器和连接器参数传给<code class="program"><a href="./programs/apxs.html">apxs</a></code>程序,使用户可以脱离Apache的发布源代码树编译其模块源代码,而不改变支持DSO的编译器和连接器的参数。</p>
</div><div class="top"><a href="#page-header"><img alt="top" src="./images/up.gif" /></a></div>
<div class="section">
<h2><a name="usage" id="usage">用法概要</a></h2>

    <p>Apache2.0 的DSO功能简要说明:</p>

    <ol>
      <li>编译并安装<em>已发布</em>的Apache模块,比如编译<code>mod_foo.c</code>为<code>mod_foo.so</code>的DSO模块:
<div class="example"><p><code>
$ ./configure --prefix=/path/to/install --enable-foo=shared<br />
$ make install
</code></p></div>
      </li>

      <li>编译并安装<em>第三方</em>模块,比如编译<code>mod_foo.c</code>为<code>mod_foo.so</code>的DSO模块:
<div class="example"><p><code>
$ ./configure --add-module=<var>module_type</var>:/path/to/3rdparty/mod_foo.c  --enable-foo=shared<br />
$ make install
</code></p></div>
      </li>

      <li>配置Apache以便<em>以后安装</em>共享模块:
<div class="example"><p><code>
$ ./configure --enable-so<br />
$ make install
</code></p></div>
      </li>

      <li>用<code class="program"><a href="./programs/apxs.html">apxs</a></code>在Apache源码树<em>以外</em>编译并安装<em>第三方</em>模块,比如编译<code>mod_foo.c</code>为<code>mod_foo.so</code>的DSO模块:
<div class="example"><p><code>
$ cd /path/to/3rdparty<br />
$ apxs -c mod_foo.c<br />
$ apxs -i -a -n foo mod_foo.la
</code></p></div>
      </li>
    </ol>

    <p>共享模块编译完毕后,必须在<code>httpd.conf</code>中用<code class="directive"><a href="./mod/mod_so.html#loadmodule">LoadModule</a></code>指令使Apache启用该模块。</p>
</div><div class="top"><a href="#page-header"><img alt="top" src="./images/up.gif" /></a></div>
<div class="section">
<h2><a name="background" id="background">背景知识</a></h2>

    <p>现代的类Unix系统都有一种叫<em>动态共享对象</em>(DSO)的动态连接/加载的巧妙的机制,从而可以在运行时将编译成特殊格式的代码加载到一个可执行程序的地址空间。</p>

    <p>加载的方法通常有两种:其一是在可执行文件启动时由系统程序<code>ld.so</code>自动加载;其二是在可执行程序中手动地通过Unix加载器的系统接口执行系统调用<code>dlopen()/dlsym()</code>进行加载。</p>

    <p>按第一种方法,DSO通常被称为<em>共享库</em>(shared libraries)或者<em>DSO库</em>(DSO libraries),使用<code>libfoo.so</code>或<code>libfoo.so.1.2</code>的文件名,存储在系统目录中(通常是<code>/usr/lib</code>),并在编译安装时使用连接器参数 <code>-lfoo</code> 建立了指向可执行程序的连接。通过设置连接器参数 <code>-R</code> 或者环境变量<code>LD_LIBRARY_PATH</code> ,库中硬编码了可执行文件的路径,使Unix加载器能够在可执行程序启动时定位到位于<code>/usr/lib</code>目录中的<code>libfoo.so</code> ,以解析可执行文件中尚未解析的位于DSO中的符号。</p>

    <p>通常,DSO不会引用可执行文件中的符号(因为它是通用代码的可重用库),也不会有后继的解析动作。可执行文件无须自己作任何动作以使用DSO中的符号,而完全由Unix加载器代办(事实上,调用<code>ld.so</code>的代码是被连入每个可执行文件的非静态运行时启动代码的一部分)。动态加载公共库代码的优点是明显的:只需要在系统库<code>libc.so</code>中存储一次库代码,从而为每个程序节省了磁盘存储空间。</p>

    <p>按第二种方法,DSO通常被称为<em>共享对象</em>(shared
    objects)或<em>DSO文件</em>(DSO files),可以使用任何文件名(但是规范的名称是<code>foo.so</code>),被存储在程序特定的目录中,也不会自动建立指向其所用的可执行文件的连接,而由可执行文件在运行时自己调用<code>dlopen()</code>来加载DSO到其地址空间,同时也不会进行为可执行文件解析DSO中符号的操作。Unix加载器会根据可执行程序的输出符号表和已经加载的DSO库自动解析DSO中尚未解析的符号(尤其是无所不在的<code>libc.so</code>中的符号),如此DSO就获得了可执行程序的符号信息,就好象是被静态连接一样。</p>

    <p>最后,为了利用DSO API的优点,可执行程序必须用<code>dlsym()</code>解析DSO中的符号,以备稍后在诸如指派表<em>等等</em>中使用。也就是说,可执行程序必须自己解析其所需的符号。这种机制的优点是允许不加载可选的程序部件,直到程序需要的时候才被动态地加载(也就不需要内存开销),以扩展程序的功能。</p>

    <p>虽然这种DSO机制看似很直接,但至少有一个难点,就是在用DSO扩展程序功能(第二种方法)时为DSO对可执行程序中符号的进行解析,这是因为,"反向解析"可执行程序中的DSO符号在所有标准平台上与库的设计都是矛盾的(库不会知道什么程序会使用它)。实际应用中,可执行文件中的全局符号通常不是重输出的,因此不能为DSO所用。所以在运行时用DSO来扩展程序功能,就必须找到强制连接器输出所有全局符号的方法。</p>

    <p>共享库是一种典型的解决方法,因为它符合DSO机制,而且为操作系统所提供的几乎所有类型的库所使用。另一方面,使用共享对象并不是许多程序为扩展其功能所采用的方法。</p>

    <p>截止到1998年,只有少数的软件包使用DSO机制在运行时扩展其功能,诸如 Perl 5(通过其XS机制和DynaLoader模块),Netscape Server等。从1.3版本开始,Apache也加入此列,因为Apache已经用了基于指派表(dispatch-list-based)的方法来连接外部模块到Apache的核心。所以Apache也就当然地在运行时用DSO来加载其模块。</p>
</div><div class="top"><a href="#page-header"><img alt="top" src="./images/up.gif" /></a></div>
<div class="section">
<h2><a name="advantages" id="advantages">优点和缺点</a></h2>

    <p>上述基于DSO的功能有如下优点:</p>

    <ul>
      <li>由于服务器包的装配工作可以在运行时使用<code>httpd.conf</code>中的配置命令<code class="directive"><a href="./mod/mod_so.html#loadmodule">LoadModule</a></code>来进行,而不是在编译中使用<code class="program"><a href="./programs/configure.html">编译选项</a></code>来进行,因此显得更灵活。比如,只需要安装一个Apache,就可以运行多个不同的服务器实例(如标准&amp;SSL版本,浓缩&amp;功能加强版本[mod_perl、PHP])。</li>

      <li>服务器可以在安装后使用第三方模块被轻易地扩展。这至少对厂商发行包的维护者有巨大的好处,他可以建立一个Apache核心包,而为诸如PHP、mod_perl、mod_fastcgi等扩展另建附加的包。</li>

      <li>更简单的Apache模块原型。使用DSO配合<code class="program"><a href="./programs/apxs.html">apxs</a></code>,可以脱离Apache源代码树,仅需要一个 <code>apxs -i</code> 和一个 <code>apachectl restart</code> 命令,就可以把刚开发的新模块纳入到运行中的Apache服务器。</li>
    </ul>

    <p>DSO有如下缺点:</p>

    <ul>
      <li>由于并不是所有操作系统都支持动态加载代码到一个程序的地址空间,因此DSO机制并不能用于所有平台。</li>

      <li>由于Unix加载器必须进行符号解析,服务器的启动会慢20%左右。</li>

      <li>在某些平台上,位置独立代码(positon independent code[PIC])需要复杂的汇编语言技巧来实现相对寻址,而绝对寻址则不需要,因此服务器在运行时会慢5%左右。</li>

      <li>由于DSO模块不能在所有平台上被其他基于DSO的库所连接(<code>ld -lfoo</code>),比如,基于a.out的平台通常不提供此功能,而基于ELF的平台则提供,因此DSO机制并不能被用于所有类型的模块。或者可以这样说,编译为DSO文件的模块只能使用由Apache核心、C库(<code>libc</code>)和Apache核心所用的所有其他动态或静态的库、含有独立位置代码的静态库(<code>libfoo.a</code>)所提供的符号。而要使用其他代码,就只能确保Apache核心本身包含对此代码的引用,或者自己用<code>dlopen()</code>来加载此代码。</li>
    </ul>

</div></div>
<div id="footer">
<p class="apache">本文允许自由使用、分发、转载,但必须保留译者署名;详见:<a href="./translator_announcement.html#announcement">译者声明</a>。</p>
<p class="menu"><a href="./mod/index.html">模块索引</a> | <a href="./mod/directives.html">指令索引</a> | <a href="./faq/index.html">常见问题</a> | <a href="./glossary.html">词汇表</a> | <a href="./sitemap.html">站点导航</a></p></div>
</body></html>

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -