<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <HTML ><HEAD ><TITLE >PHP 的命令行模式</TITLE ><META NAME="GENERATOR" CONTENT="Modular DocBook HTML Stylesheet Version 1.79"><LINK REL="HOME" TITLE="PHP 手册" HREF="index.html"><LINK REL="UP" TITLE="特点" HREF="features.html"><LINK REL="PREVIOUS" TITLE="被安全模式限制或屏蔽的函数" HREF="features.safe-mode.functions.html"><LINK REL="NEXT" TITLE="函数参考" HREF="funcref.html"><META HTTP-EQUIV="Content-type" CONTENT="text/html; charset=UTF-8"></HEAD ><BODY CLASS="chapter" BGCOLOR="#FFFFFF" TEXT="#000000" LINK="#0000FF" VLINK="#840084" ALINK="#0000FF" ><DIV CLASS="NAVHEADER" ><TABLE SUMMARY="Header navigation table" WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" ><TR ><TH COLSPAN="3" ALIGN="center" >PHP 手册</TH ></TR ><TR ><TD WIDTH="10%" ALIGN="left" VALIGN="bottom" ><A HREF="features.safe-mode.functions.html" ACCESSKEY="P" >上一页</A ></TD ><TD WIDTH="80%" ALIGN="center" VALIGN="bottom" ></TD ><TD WIDTH="10%" ALIGN="right" VALIGN="bottom" ><A HREF="funcref.html" ACCESSKEY="N" >下一页</A ></TD ></TR ></TABLE ><HR ALIGN="LEFT" WIDTH="100%"></DIV ><DIV CLASS="chapter" ><H1 ><A NAME="features.commandline" >章 43. PHP 的命令行模式</A ></H1 ><P > 从版本 4.3.0 开始,PHP 提供了一种新类型的 <TT CLASS="literal" >SAPI</TT >(Server Application Programming Interface,服务端应用编程端口)支持,名为 <TT CLASS="literal" >CLI</TT >,意为 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >Command Line Interface</I ></SPAN >,即命令行接口。顾名思义,该 <TT CLASS="literal" >SAPI</TT > 模块主要用作 PHP 的开发外壳应用。<TT CLASS="literal" >CLI SAPI</TT > 和其它 <TT CLASS="literal" >SAPI</TT > 模块相比有很多的不同之处,我们将在本章中详细阐述。值得一提的是,<TT CLASS="literal" >CLI</TT > 和 <TT CLASS="literal" >CGI</TT > 是不同的 SAPI,尽管它们之间有很多共同的行为。 </P ><P > <TT CLASS="literal" >CLI SAPI</TT > 最先是随 PHP 4.2.0 版本发布的,但仍旧只是一个实验性的版本,并需要在运行 <B CLASS="command" >./configure</B > 时加上 <CODE CLASS="option" >--enable-cli</CODE > 参数。从 PHP 4.3.0 版本开始,<TT CLASS="literal" >CLI SAPI</TT > 成为了正式模块,<CODE CLASS="option" >--enable-cli</CODE > 参数会被默认得设置为 on,也可以用参数 <CODE CLASS="option" >--disable-cli</CODE > 来屏蔽。 </P ><P > 从 PHP 4.3.0开始,CLI/CGI 二进制执行文件的文件名、位置和是否存在会根据 PHP 在系统上的安装而不同。在默认情况下,当运行 <B CLASS="command" >make</B > 时,CGI 和 CLI 都会被编译并且分别放置在 PHP 源文件目录的 <TT CLASS="filename" >sapi/cgi/php</TT > 和 <TT CLASS="filename" >sapi/cli/php</TT > 下。可以注意到两个文件都被命名为了 php。在 <B CLASS="command" >make install</B > 的过程中会发生什么取决于配置行。如果在配置的时候选择了一个 SAPI 模块,如 apxs,或者使用了 <CODE CLASS="option" >--disable-cgi</CODE > 参数,则在 <B CLASS="command" >make install</B > 的过程中,CLI 将被拷贝到 <TT CLASS="filename" >{PREFIX}/bin/php</TT >,除非 CGI 已经被放置在了那个位置。因此,例如,如果在配置行中有 <CODE CLASS="option" >--with--apxs</CODE >,则在 <TT CLASS="literal" >make install</TT > 的过程中,CLI 将被拷贝到 <TT CLASS="literal" >{PREFIX}/bin/php</TT >。如果希望撤销 CGI 执行文件的安装,请在 <B CLASS="command" >make install</B > 之后运行 <B CLASS="command" >make install-cli</B >。或者,也可以在配置行中加上 <CODE CLASS="option" >--disable-cgi</CODE > 参数。 </P ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 由于 <CODE CLASS="option" >--enable-cli</CODE > 和 <CODE CLASS="option" >--enable-cgi</CODE > 同时默认有效,因此,不必再配置行中加上 <CODE CLASS="option" >--enable-cli</CODE > 来使得 CLI 在 <B CLASS="command" >make install</B > 过程中被拷贝到 <TT CLASS="filename" >{PREFIX}/bin/php</TT >。 </P ></BLOCKQUOTE ></DIV ><P > 在 PHP 4.2.0 到 PHP 4.2.3 之间的 Windows 发行包中,CLI 的文件名为 <TT CLASS="filename" >php-cli.exe</TT >,相同文件夹下的 <TT CLASS="filename" >php.exe</TT > 为 CGI。从 PHP 4.3.0 版本开始,Windows 的发行包中 CLI 的执行文件为 <TT CLASS="filename" >php.exe</TT >,被放置在一个单独的名为 <TT CLASS="filename" >cli</TT > 的文件夹下,即 <TT CLASS="filename" >cli/php.exe</TT >。在 PHP 5 中,CLI 存在于主文件夹中,名为 <TT CLASS="filename" >php.exe</TT >,而 CGI 版本名为 <TT CLASS="filename" >php-cgi.exe</TT >。 </P ><P > 从 PHP 5 起,一个名为 <TT CLASS="filename" >php-win.exe</TT > 的新文件随包发布。它相当于 CLI 版本,但是 php-win 不输出任何内容,便不提供控制台(不会弹出“DOS 窗口”)。这种方式类似于 php-gtk。需要使用 <CODE CLASS="option" >--enable-cli-win32</CODE > 选项来配置它。 </P ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >如何得知自己使用的是哪个 SAPI?: </B > 在命令行下,运行 <B CLASS="command" >php -v</B > 便能得知该 <TT CLASS="filename" >php</TT > 是 CGI 还是 CLI。请参考函数 <A HREF="function.php-sapi-name.html" ><B CLASS="function" >php_sapi_name()</B ></A > 以及常量 <TT CLASS="constant" ><B >PHP_SAPI</B ></TT >。 </P ></BLOCKQUOTE ></DIV ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 在 PHP 4.3.2 中加入了 Unix 的 <TT CLASS="literal" >man</TT > 页面。可以在命令行中键入 <B CLASS="command" >man php</B > 来查看。 </P ></BLOCKQUOTE ></DIV ><P > 以下为 <TT CLASS="literal" >CLI SAPI</TT > 和其它 <TT CLASS="literal" >SAPI</TT > 模块相比的显著区别: <P ></P ><UL ><LI ><P > 与 <TT CLASS="literal" >CGI SAPI</TT > 不同,其输出没有任何头信息。 </P ><P > 尽管 <TT CLASS="literal" >CGI SAPI</TT > 提供了取消 HTTP 头信息的方法,但在 <TT CLASS="literal" >CLI SAPI</TT > 中并不存在类似的方法以开启 HTTP 头信息的输出。 </P ><P > CLI 默认以安静模式开始,但为了保证兼容性,<CODE CLASS="option" >-q</CODE > 和 <CODE CLASS="option" >--no-header</CODE > 参数为了向后兼容仍然保留,使得可以使用旧的 CGI 脚本。 </P ><P > 在运行时,不会把工作目录改为脚本的当前目录(可以使用 <CODE CLASS="option" >-C</CODE > 和 <CODE CLASS="option" >--no-chdir</CODE > 参数来兼容 CGI 模式)。 </P ><P > 出错时输出纯文本的错误信息(非 HTML 格式)。 </P ></LI ><LI ><P > <TT CLASS="literal" >CLI SAPI</TT > 强制覆盖了 <TT CLASS="filename" >php.ini</TT > 中的某些设置,因为这些设置在外壳环境下是没有意义的。 </P ><P > <DIV CLASS="table" ><A NAME="AEN7823" ></A ><P ><B >表 43-1. 覆盖 <TT CLASS="filename" >php.ini</TT > 设置选项</B ></P ><TABLE BORDER="1" CLASS="CALSTABLE" ><COL><COL><COL><THEAD ><TR ><TH >设置选项</TH ><TH ><TT CLASS="literal" >CLI SAPI</TT > 默认值</TH ><TH >备注</TH ></TR ></THEAD ><TBODY ><TR ><TD ><A HREF="ref.errorfunc.html#ini.html-errors" >html_errors</A ></TD ><TD ><TT CLASS="constant" ><B >FALSE</B ></TT ></TD ><TD > 无意义的 HTML 标记符会使得出错信息很凌乱,所以在外壳下阅读报错信息是十分困难的。因此将该选项的默认值改为 <TT CLASS="constant" ><B >FALSE</B ></TT >。 </TD ></TR ><TR ><TD ><A HREF="ref.outcontrol.html#ini.implicit-flush" >implicit_flush</A ></TD ><TD ><TT CLASS="constant" ><B >TRUE</B ></TT ></TD ><TD > 在命令行模式下,所有来自 <A HREF="function.print.html" ><B CLASS="function" >print()</B ></A > 和 <A HREF="function.echo.html" ><B CLASS="function" >echo()</B ></A > 的输出将被立即写到输出端,而不作任何地缓冲操作。如果希望延缓或控制标准输出,仍然可以使用 <A HREF="ref.outcontrol.html" >output buffering</A > 设置项。 </TD ></TR ><TR ><TD ><A HREF="ref.info.html#ini.max-execution-time" >max_execution_time</A ></TD ><TD >0(无限值)</TD ><TD > 鉴于在外壳环境下使用 PHP 的无穷的可能性,最大运行时间被设置为了无限值。为 web 开发的应用程序可能只需运行几秒钟时间,而外壳应用程序的运行时间可能会长的多。 </TD ></TR ><TR ><TD ><A HREF="ini.core.html#ini.register-argc-argv" >register_argc_argv</A ></TD ><TD ><TT CLASS="constant" ><B >TRUE</B ></TT ></TD ><TD > <P > 由于该设置为 <TT CLASS="constant" ><B >TRUE</B ></TT >,将总是可以在 <TT CLASS="literal" >CLI SAPI</TT > 中访问到 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >argc</I ></SPAN >(传送给应用程序参数的个数)和 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >argv</I ></SPAN >(包含有实际参数的数组)。 </P > <P > 对于 PHP 4.3.0,在使用 <TT CLASS="literal" >CLI SAPI</TT > 时,PHP 变量 <TT CLASS="literal" >$argc</TT > 和 <TT CLASS="literal" >$argv</TT > 已被注册并且设定了对应的值。而在这之前的版本,这两个变量在 <TT CLASS="literal" >CGI</TT > 或者 <TT CLASS="literal" >模块</TT > 版本中的建立依赖于将 PHP 的设置选项 <A HREF="ini.core.html#ini.register-globals" >register_globals</A > 设为 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >on</I ></SPAN >。除了版本和 <TT CLASS="literal" >register_globals</TT > 设定以外,可以随时通过调用 <A HREF="reserved.variables.html#reserved.variables.server" >$_SERVER</A > 或者 <CODE CLASS="varname" >$HTTP_SERVER_VARS</CODE > 来访问它们。例如:<CODE CLASS="varname" >$_SERVER['argv']</CODE > </P > </TD ></TR ></TBODY ></TABLE ></DIV > </P ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 这些设置无法在设置文件 <TT CLASS="filename" >php.ini</TT > 或任何指定的其它文件中被初始化为其它值。这些默认值被限制在所有其它的设置文件被解析后改变。不过,它们的值可以在程序运行的过程中被改变(尽管对于该运行过程来说,这些设置项是没有意义的)。 </P ></BLOCKQUOTE ></DIV ></LI ><LI ><P > 为了减轻外壳环境下的工作,我们定义了如下常量: <DIV CLASS="table" ><A NAME="AEN7883" ></A ><P ><B >表 43-2. CLI 专用常量</B ></P ><TABLE BORDER="1" CLASS="CALSTABLE" ><COL><COL><THEAD ><TR ><TH >常量名称</TH ><TH >描 述</TH ></TR ></THEAD ><TBODY ><TR ><TD ><TT CLASS="constant" ><B >STDIN</B ></TT ></TD ><TD > 一个已打开的指向 <TT CLASS="literal" >stdin</TT > 的流。可以用如下方法来调用: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> <font color="#0000BB"><?php<br /><br />$stdin </font><font color="#007700">= </font><font color="#0000BB">fopen</font><font color="#007700">(</font><font color="#DD0000">'php://stdin'</font><font color="#007700">, </font><font color="#DD0000">'r'</font><font color="#007700">);<br /><br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > 如果想从 <TT CLASS="literal" >stdin</TT > 读取一行内容,可以使用 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> <font color="#0000BB"><?php<br />$line </font><font color="#007700">= </font><font color="#0000BB">trim</font><font color="#007700">(</font><font color="#0000BB">fgets</font><font color="#007700">(</font><font color="#0000BB">STDIN</font><font color="#007700">)); </font><font color="#FF8000">// 从 STDIN 读取一行<br /></font><font color="#0000BB">fscanf</font><font color="#007700">(</font><font color="#0000BB">STDIN</font><font color="#007700">, </font><font color="#DD0000">"%d\n"</font><font color="#007700">, </font><font color="#0000BB">$number</font><font color="#007700">); </font><font color="#FF8000">// 从 STDIN 读取数字<br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > </TD ></TR ><TR ><TD ><TT CLASS="constant" ><B >STDOUT</B ></TT ></TD ><TD > 一个已打开的指向 <TT CLASS="literal" >stdout</TT > 的流。可以用如下方式来调用: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> <font color="#0000BB"><?php<br /><br />$stdout </font><font color="#007700">= </font><font color="#0000BB">fopen</font><font color="#007700">(</font><font color="#DD0000">'php://stdout'</font><font color="#007700">, </font><font color="#DD0000">'w'</font><font color="#007700">);<br /><br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > </TD ></TR ><TR ><TD ><TT CLASS="constant" ><B >STDERR</B ></TT ></TD ><TD > 一个已打开的指向 <TT CLASS="literal" >stderr</TT > 的流。可以用如下方式来调用: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> <font color="#0000BB"><?php<br /><br />$stderr </font><font color="#007700">= </font><font color="#0000BB">fopen</font><font color="#007700">(</font><font color="#DD0000">'php://stderr'</font><font color="#007700">, </font><font color="#DD0000">'w'</font><font color="#007700">);<br /><br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > </TD ></TR ></TBODY ></TABLE ></DIV > </P ><P > 有了以上常量,就无需自己建立指向诸如 <TT CLASS="literal" >stderr</TT > 的流,只需简单的使用这些常量来代替流指向: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="shell" >php -r 'fwrite(STDERR, "stderr\n");'</PRE ></TD ></TR ></TABLE > 无需自己来关闭这些流,PHP 会自动完成这些操作。 </P ></LI ><LI ><P > <TT CLASS="literal" >CLI SAPI</TT > <SPAN CLASS="strong" ><B CLASS="emphasis" >不会</B ></SPAN >将当前目录改为已运行的脚本所在的目录。 </P ><P > 以下范例显示了本模块与 <TT CLASS="literal" >CGI SAPI</TT > 模块之间的不同: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> <font color="#0000BB"><?php<br /></font><font color="#FF8000">// 名为 test.php 的简单测试程序<br /></font><font color="#007700">echo </font><font color="#0000BB">getcwd</font><font color="#007700">(), </font><font color="#DD0000">"\n"</font><font color="#007700">;<br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > </P ><P > 在使用 <TT CLASS="literal" >CGI</TT > 版本时,其输出为 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ pwd /tmp $ php-cgi -f another_directory/test.php /tmp/another_directory</PRE ></TD ></TR ></TABLE > </P ><P > 明显可以看到 PHP 将当前目录改成了刚刚运行过的脚本所在的目录。 </P ><P > 使用 <TT CLASS="literal" >CLI SAPI</TT > 模式,得到: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ pwd /tmp $ php -q another_directory/test.php /tmp</PRE ></TD ></TR ></TABLE > 这使得在利用 PHP 编写外壳工具时获得了很大的便利。 </P ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 可以在命令行运行时给该 <TT CLASS="literal" >CGI SAPI</TT > 加上 <CODE CLASS="option" >-C</CODE > 参数,使其支持 <TT CLASS="literal" >CLI SAPI</TT > 的功能。 </P ></BLOCKQUOTE ></DIV ></LI ></UL > </P ><P > 以下是 PHP 二进制文件(即 <TT CLASS="filename" >php.exe</TT > 程序)提供的命令行模式的选项参数,随时可以运行带 <CODE CLASS="option" >-h</CODE > 参数的 PHP 命令来查询这些参数。 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >Usage: php [options] [-f] <file> [--] [args...] php [options] -r <code> [--] [args...] php [options] [-B <begin_code>] -R <code> [-E <end_code>] [--] [args...] php [options] [-B <begin_code>] -F <file> [-E <end_code>] [--] [args...] php [options] -- [args...] php [options] -a -a Run interactively -c <path>|<file> Look for php.ini file in this directory -n No php.ini file will be used -d foo[=bar] Define INI entry foo with value 'bar' -e Generate extended information for debugger/profiler -f <file> Parse <file>. -h This help -i PHP information -l Syntax check only (lint) -m Show compiled in modules -r <code> Run PHP <code> without using script tags <?..?> -B <begin_code> Run PHP <begin_code> before processing input lines -R <code> Run PHP <code> for every input line -F <file> Parse and execute <file> for every input line -E <end_code> Run PHP <end_code> after processing all input lines -H Hide any passed arguments from external tools. -s Display colour syntax highlighted source. -v Version number -w Display source with stripped comments and whitespace. -z <file> Load Zend extension <file>. args... Arguments passed to script. Use -- args when first argument starts with - or script is read from stdin</PRE ></TD ></TR ></TABLE > </P ><P > <TT CLASS="literal" >CLI SAPI</TT > 模块有以下三种不同的方法来获取要运行的 PHP 代码: <P ></P ><OL TYPE="1" ><LI ><P > 让 PHP 运行指定文件。 </P ><P > <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >php my_script.php php -f my_script.php</PRE ></TD ></TR ></TABLE > </P ><P > 以上两种方法(使用或不使用 <CODE CLASS="option" >-f</CODE > 参数)都能够运行给定的 <TT CLASS="filename" >my_script.php</TT > 文件。可以选择任何文件来运行,指定的 PHP 脚本并非必须要以 <TT CLASS="literal" >.php</TT > 为扩展名,它们可以有任意的文件名和扩展名。 </P ></LI ><LI ><P > 在命令行直接运行 PHP 代码。 </P ><P > <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >php -r 'print_r(get_defined_constants());'</PRE ></TD ></TR ></TABLE > </P ><P > 在使用这种方法时,请注意外壳变量的替代及引号的使用。 </P ><DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 请仔细阅读以上范例,在运行代码时没有开始和结束的标记符!加上 <CODE CLASS="option" >-r</CODE > 参数后,这些标记符是不需要的,加上它们会导致语法错误。 </P ></BLOCKQUOTE ></DIV ></LI ><LI ><P > 通过标准输入(<TT CLASS="literal" >stdin</TT >)提供需要运行的 PHP 代码。 </P ><P > 以上用法提供了非常强大的功能,使得可以如下范例所示,动态地生成 PHP 代码并通过命令行运行这些代码: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ some_application | some_filter | php | sort -u >final_output.txt</PRE ></TD ></TR ></TABLE > </P ></LI ></OL > </P ><P > 以上三种运行代码的方法不能同时使用。 </P ><P > 和所有的外壳应用程序一样,PHP 的二进制文件(<TT CLASS="filename" >php.exe</TT > 文件)及其运行的 PHP 脚本能够接受一系列的参数。PHP 没有限制传送给脚本程序的参数的个数(外壳程序对命令行的字符数有限制,但通常都不会超过该限制)。传递给脚本的参数可在全局变量 <CODE CLASS="varname" >$argv</CODE > 中获取。该数组中下标为零的成员为脚本的名称(当 PHP 代码来自标准输入获直接用 <CODE CLASS="option" >-r</CODE > 参数以命令行方式运行时,该名称为“<TT CLASS="literal" >-</TT >”)。另外,全局变量 <CODE CLASS="varname" >$argc</CODE > 存有 <CODE CLASS="varname" >$argv</CODE > 数组中成员变量的个数(而非传送给脚本程序的参数的个数)。 </P ><P > 只要传送给脚本的参数不是以 <TT CLASS="literal" >-</TT > 符号开头,就无需过多的注意什么。向脚本传送以 <TT CLASS="literal" >-</TT > 开头的参数会导致错误,因为 PHP 会认为应该由它自身来处理这些参数。可以用参数列表分隔符 <TT CLASS="literal" >--</TT > 来解决这个问题。在 PHP 解析完参数后,该符号后所有的参数将会被原样传送给脚本程序。 </P ><P > <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" ># 以下命令将不会运行 PHP 代码,而只显示 PHP 命令行模式的使用说明: $ php -r 'var_dump($argv);' -h Usage: php [options] [-f] <file> [args...] [...] # 以下命令将会把“-h”参数传送给脚本程序,PHP 不会显示命令行模式的使用说明: $ php -r 'var_dump($argv);' -- -h array(2) { [0]=> string(1) "-" [1]=> string(2) "-h" }</PRE ></TD ></TR ></TABLE > </P ><P > 除此之外,还有另一个方法将 PHP 用于外壳脚本。可以在写一个脚本,并在第一行以 <TT CLASS="literal" >#!/usr/bin/php</TT > 开头,在其后加上以 PHP 开始和结尾标记符包含的正常的 PHP 代码,然后为该文件设置正确的运行属性(例如:<B CLASS="command" >chmod +x test</B >)。该方法可以使得该文件能够像外壳脚本或 PERL 脚本一样被直接执行。 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> #!/usr/bin/php<br /><font color="#0000BB"><?php<br /> var_dump</font><font color="#007700">(</font><font color="#0000BB">$argv</font><font color="#007700">);<br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE > </P ><P > 假设改文件名为 <TT CLASS="filename" >test</TT > 并被放置在当前目录下,可以做如下操作: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ chmod +x test $ ./test -h -- foo array(4) { [0]=> string(6) "./test" [1]=> string(2) "-h" [2]=> string(2) "--" [3]=> string(3) "foo" }</PRE ></TD ></TR ></TABLE > </P ><P > 正如所看到的,在向该脚本传送以 <TT CLASS="literal" >-</TT > 开头的参数时,脚本仍然能够正常运行。 </P ><P > PHP 4.3.3 以来有效的长选项: <DIV CLASS="table" ><A NAME="AEN7985" ></A ><P ><B >表 43-3. 命令行选项</B ></P ><TABLE BORDER="1" CLASS="CALSTABLE" ><COL><COL><THEAD ><TR ><TH >选项名称</TH ><TH >长名称</TH ><TH >说明</TH ></TR ></THEAD ><TBODY ><TR ><TD >-a</TD ><TD >--interactive</TD ><TD > <P > 交互式运行 PHP。如果编译 PHP 时加入了 <A HREF="ref.readline.html" >Readline</A > 扩展(Windows 下不可用),那将会得到一个很好的外壳,包括一个自动完成的功能(例如可以在键入变量名的时候,按下 TAB 键,PHP 会自动完成该变量名)以及命令历史记录,可以用上下键来访问。历史记录存在 <TT CLASS="filename" >~/.php_history</TT > 文件中。 </P > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 通过 <A HREF="ini.core.html#ini.auto-prepend-file" >auto_prepend_file</A > 和 <A HREF="ini.core.html#ini.auto-append-file" >auto_append_file</A > 包含的文件在此模式下会被解析,但有些限制,例如函数必须在被调用之前定义。 </P ></BLOCKQUOTE ></DIV > </TD ></TR ><TR ><TD >-c</TD ><TD >--php-ini</TD ><TD > <P > 用该参数,可以指定一个放置 <TT CLASS="filename" >php.ini</TT > 文件的目录,或者直接指定一个自定义的 <TT CLASS="literal" >INI</TT > 文件(其文件名可以不是 <TT CLASS="filename" >php.ini</TT >),例如: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -c /custom/directory/ my_script.php $ php -c /custom/directory/custom-file.ini my_script.php</PRE ></TD ></TR ></TABLE > 如果不指定此选项,PHP 将在<A HREF="configuration.html#configuration.file" >默认位置</A >搜索文件。 </P > </TD ></TR ><TR ><TD >-n</TD ><TD >--no-php-ini</TD ><TD > <P > 完全忽略 <TT CLASS="filename" >php.ini</TT >。此参数在 PHP 4.3.0 以后有效。 </P > </TD ></TR ><TR ><TD >-d</TD ><TD >--define</TD ><TD > <P > 用该参数可以自行设置任何可以在 <TT CLASS="filename" >php.ini</TT > 文件中设置的配置选项的值,其语法为: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >-d configuration_directive[=value]</PRE ></TD ></TR ></TABLE > </P > <P > 例子(因版面原因而折行显示): <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" ># 取值部分被省略,将会把配置选项设为 "1" $ php -d max_execution_time -r '$foo = ini_get("max_execution_time"); var_dump($foo);' string(1) "1" # 取值部分为空白,将会把配置选项设为 "" php -d max_execution_time= -r '$foo = ini_get("max_execution_time"); var_dump($foo);' string(0) "" # 配置选项将被设置成为任何 '=' 字符之后的值 $ php -d max_execution_time=20 -r '$foo = ini_get("max_execution_time"); var_dump($foo);' string(2) "20" $ php -d max_execution_time=doesntmakesense -r '$foo = ini_get("max_execution_time"); var_dump($foo);' string(15) "doesntmakesense"</PRE ></TD ></TR ></TABLE > </P > </TD ></TR ><TR ><TD >-e</TD ><TD >--profile-info</TD ><TD > <P > 激活扩展信息模式,被用于调试/测试。 </P > </TD ></TR ><TR ><TD >-f</TD ><TD >--file</TD ><TD > <P > 解析并运行 <CODE CLASS="option" >-f</CODE > 选项给定的文件名。该参数为可选参数,可以省略,仅指明需要运行的文件名即可。 </P > </TD ></TR ><TR ><TD >-h and -?</TD ><TD >--help and --usage</TD ><TD > 使用该参数,可以得到完整的命令行参数的列表及这些参数作用的简单描述。 </TD ></TR ><TR ><TD >-i</TD ><TD >--info</TD ><TD > 该命令行参数会调用 <A HREF="function.phpinfo.html" ><B CLASS="function" >phpinfo()</B ></A > 函数并显示出结果。如果 PHP 没有正常工作,建议执行 <B CLASS="command" >php -i</B > 命令来查看在信息表格之前或者对应的地方是否有任何错误信息输出。请注意当使用 CGI 摸索时,输出的内容为 <TT CLASS="literal" >HTML</TT > 格式,因此输出的信息篇幅较大。 </TD ></TR ><TR ><TD >-l</TD ><TD >--syntax-check</TD ><TD > <P > 该参数提供了对指定 PHP 代码进行语法检查的方便的方法。如果成功,则向标准输出写入 <TT CLASS="literal" >No syntax errors detected in <filename></TT > 字符串,并且外壳返回值为 <TT CLASS="literal" >0</TT >。如果失败,则输出 <TT CLASS="literal" >Errors parsing <filename></TT > 以及内部解析器错误信息到标准输出,同时外壳返回值将别设置为 <TT CLASS="literal" >255</TT >。 </P > <P > 该参数将无法检查致命错误(如未定义函数),如果也希望检测致命错误,请使用 <CODE CLASS="option" >-f</CODE > 参数。 </P > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 该参数不能和 <CODE CLASS="option" >-r</CODE > 一同使用。 </P ></BLOCKQUOTE ></DIV > </TD ></TR ><TR ><TD >-m</TD ><TD >--modules</TD ><TD > <P > 使用该参数,PHP 将打印出内置以及已加载的 PHP 及 Zend 模块: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -m [PHP Modules] xml tokenizer standard session posix pcre overload mysql mbstring ctype [Zend Modules]</PRE ></TD ></TR ></TABLE > </P > </TD ></TR ><TR ><TD >-r</TD ><TD >--run</TD ><TD > <P > 使用该参数可以在命令行内运行单行 PHP 代码。<SPAN CLASS="strong" ><B CLASS="emphasis" >无需</B ></SPAN >加上 PHP 的起始和结束标识符(<TT CLASS="literal" ><?php</TT > 和 <TT CLASS="literal" >?></TT >),否则将会导致语法解析错误。 </P > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 使用这种形式的 PHP 时,应注意避免和外壳环境进行的命令行参数替换相冲突。 </P ><P > 显示语法解析错误的范例 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -r "$foo = get_defined_constants();" Command line code(1) : Parse error - parse error, unexpected '='</PRE ></TD ></TR ></TABLE > 这里的问题在于即使使用了双引号 <TT CLASS="literal" >"</TT >,sh/bash 仍然实行了参数替换。由于 <CODE CLASS="varname" >$foo</CODE > 没有被定义,被替换后它所在的位置变成了空字符,因此在运行时,实际被 PHP 读取的代码为: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -r " = get_defined_constants();"</PRE ></TD ></TR ></TABLE > 正确的方法是使用单引号 <TT CLASS="literal" >'</TT >。在用单引号引用的字符串中,变量不会被 sh/bash 还原成其原值。 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -r '$foo = get_defined_constants(); var_dump($foo);' array(370) { ["E_ERROR"]=> int(1) ["E_WARNING"]=> int(2) ["E_PARSE"]=> int(4) ["E_NOTICE"]=> int(8) ["E_CORE_ERROR"]=> [...]</PRE ></TD ></TR ></TABLE > 如果使用的外壳不是 sh/bash,可能会碰到更多问题。请将碰到的 Bug 向 <A HREF="http://bugs.php.net/" TARGET="_top" >http://bugs.php.net/</A > 报告。注意,当试图将 shell 变量用到代码中或者使用反斜线时仍然很容易碰到问题。 </P ></BLOCKQUOTE ></DIV > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > <CODE CLASS="option" >-r</CODE > 在 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >CLI</I ></SPAN > SAPI 中有效,在 <SPAN CLASS="emphasis" ><I CLASS="emphasis" >CGI</I ></SPAN > SAPI 中无效。 </P ></BLOCKQUOTE ></DIV > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 此选项只用于非常基本的用途。因此一些配置指令(例如 <A HREF="ini.core.html#ini.auto-prepend-file" >auto_prepend_file</A > 和 <A HREF="ini.core.html#ini.auto-append-file" >auto_append_file</A >)在此模式下被忽略。 </P ></BLOCKQUOTE ></DIV > </TD ></TR ><TR ><TD >-B</TD ><TD >--process-begin</TD ><TD > <P > 在处理 stdin 之前先执行 PHP 代码。PHP 5 新加。 </P > </TD ></TR ><TR ><TD >-R</TD ><TD >--process-code</TD ><TD > <P > 对每个输入行都执行 PHP 代码。PHP 5 新加。 </P > <P > 此模式下有两个特殊变量:<CODE CLASS="varname" >$argn</CODE > 和 <CODE CLASS="varname" >$argi</CODE >。<CODE CLASS="varname" >$argn</CODE > 包含 PHP 当前处理的行内容,而 <CODE CLASS="varname" >$argi</CODE > 则包含该行号。 </P > </TD ></TR ><TR ><TD >-F</TD ><TD >--process-file</TD ><TD > <P > 对每个输入行都执行 PHP 文件。PHP 5 新加。 </P > </TD ></TR ><TR ><TD >-E</TD ><TD >--process-end</TD ><TD > <P > 在处理完输入后执行的 PHP 代码。PHP 5 新加。 </P > <P > 使用 <CODE CLASS="option" >-B</CODE >,<CODE CLASS="option" >-R</CODE > 和 <CODE CLASS="option" >-E</CODE > 选项来计算一个项目总行数的例子。 <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ find my_proj | php -B '$l=0;' -R '$l += count(@file($argn));' -E 'echo "Total Lines: $l\n";' Total Lines: 37328</PRE ></TD ></TR ></TABLE > </P > </TD ></TR ><TR ><TD >-s</TD ><TD >--syntax-highlight and --syntax-highlight</TD ><TD > <P > 显示有语法高亮色彩的源代码。 </P > <P > 该参数使用内建机制来解析文件并为其生成一个 <TT CLASS="literal" >HTML</TT > 高亮版本并将结果写到标准输出。请注意该过程所做的只是生成了一个 <TT CLASS="literal" ><code> [...] </code></TT > 的 <TT CLASS="literal" >HTML</TT > 标记的块,并不包含任何的 <TT CLASS="literal" >HTML</TT > 头。 </P > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 该选项不能和 <CODE CLASS="option" >-r</CODE > 参数同时使用。 </P ></BLOCKQUOTE ></DIV > </TD ></TR ><TR ><TD >-v</TD ><TD >--version</TD ><TD > <P > 将 PHP,PHP SAPI 和 Zend 的版本信息写入标准输出。例如: <TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="screen" >$ php -v PHP 4.3.0 (cli), Copyright (c) 1997-2002 The PHP Group Zend Engine v1.3.0, Copyright (c) 1998-2002 Zend Technologies</PRE ></TD ></TR ></TABLE > </P > </TD ></TR ><TR ><TD >-w</TD ><TD >--strip</TD ><TD > <P > 显示除去了注释和多余空白的源代码。 </P > <DIV CLASS="note" ><BLOCKQUOTE CLASS="note" ><P ><B >注意: </B > 该选项不能和 <CODE CLASS="option" >-r</CODE > 参数同时使用。 </P ></BLOCKQUOTE ></DIV > </TD ></TR ><TR ><TD >-z</TD ><TD >--zend-extension</TD ><TD > <P > 加载 Zend 扩展库。如果仅给定一个文件名,PHP 将试图从当前系统扩展库的默认路径(在 Linux 系统下,该路径通常由 <TT CLASS="filename" >/etc/ld.so.conf</TT > 指定)加载该扩展库。如果用一个绝对路径指定文件名,则不会使用系统的扩展库默认路径。如果用相对路径指定的文件名,则 PHP 仅试图在当前目录的相对目录加载扩展库。 </P > </TD ></TR ></TBODY ></TABLE ></DIV > </P ><P > PHP 的命令行模式能使得 PHP 脚本能完全独立于 web 服务器单独运行。如果使用 Unix 系统,需要在 PHP 脚本的最前面加上一行特殊的代码,使得它能够被执行,这样系统就能知道用哪个程序去运行该脚本。在 Windows 平台下可以将 <TT CLASS="filename" >php.exe</TT > 和 <TT CLASS="literal" >.php</TT > 文件的双击属性相关联,也可以编写一个批处理文件来用 PHP 执行脚本。为 Unix 系统增加的第一行代码不会影响该脚本在 Windows 下的运行,因此也可以用该方法编写跨平台的脚本程序。以下是一个简单的 PHP 命令行程序的范例。 </P ><P > <TABLE WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" CLASS="EXAMPLE" ><TR ><TD ><DIV CLASS="example" ><A NAME="AEN8166" ></A ><P ><B >例 43-1. 试图以命令行方式运行的 PHP 脚本(script.php)</B ></P ><TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><code><font color="#000000"> #!/usr/bin/php<br /><font color="#0000BB"><?php<br /><br /></font><font color="#007700">if (</font><font color="#0000BB">$argc </font><font color="#007700">!= </font><font color="#0000BB">2 </font><font color="#007700">|| </font><font color="#0000BB">in_array</font><font color="#007700">(</font><font color="#0000BB">$argv</font><font color="#007700">[</font><font color="#0000BB">1</font><font color="#007700">], array(</font><font color="#DD0000">'--help'</font><font color="#007700">, </font><font color="#DD0000">'-help'</font><font color="#007700">, </font><font color="#DD0000">'-h'</font><font color="#007700">, </font><font color="#DD0000">'-?'</font><font color="#007700">))) {<br /></font><font color="#0000BB">?><br /></font><br />This is a command line PHP script with one option.<br /><br /> Usage:<br /> <font color="#0000BB"><?php </font><font color="#007700">echo </font><font color="#0000BB">$argv</font><font color="#007700">[</font><font color="#0000BB">0</font><font color="#007700">]; </font><font color="#0000BB">?></font> <option><br /><br /> <option> can be some word you would like<br /> to print out. With the --help, -help, -h,<br /> or -? options, you can get this help.<br /><br /><font color="#0000BB"><?php<br /></font><font color="#007700">} else {<br /> echo </font><font color="#0000BB">$argv</font><font color="#007700">[</font><font color="#0000BB">1</font><font color="#007700">];<br />}<br /></font><font color="#0000BB">?></font> </font> </code></TD ></TR ></TABLE ></DIV ></TD ></TR ></TABLE > </P ><P > 在以上脚本中,用第一行特殊的代码来指明该文件应该由 PHP 来执行。在这里使用 CLI 的版本,因此不会有 HTTP 头信息输出。在用 PHP 编写命令行应用程序时,可以使用两个参数:<CODE CLASS="varname" >$argc</CODE > 和 <CODE CLASS="varname" >$argv</CODE >。前面一个的值是比参数个数大 1 的整数(运行的脚本本身的名称也被当作一个参数)。第二个是包含有参数的数组,其第一个元素为脚本的名称,下标为数字 0(<CODE CLASS="varname" >$argv[0]</CODE >)。 </P ><P > 以上程序中检查了参数的个数是大于 1 个还是小于 1 个。此外如果参数是 <CODE CLASS="option" >--help</CODE >,<CODE CLASS="option" >-help</CODE >,<CODE CLASS="option" >-h</CODE > 或 <CODE CLASS="option" >-?</CODE > 时,打印出帮助信息,并同时动态输出脚本的名称。如果还收到了其它参数,将其显示出来。 </P ><P > 如果希望在 Unix 下运行以上脚本,需要使其属性为可执行文件,然后简单的运行 <B CLASS="command" >script.php echothis</B > 或 <B CLASS="command" >script.php -h</B >。在 Windows 下,可以为此编写一个批处理文件: </P ><P > <TABLE WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" CLASS="EXAMPLE" ><TR ><TD ><DIV CLASS="example" ><A NAME="AEN8182" ></A ><P ><B >例 43-2. 运行 PHP 命令行脚本的批处理文件(script.bat)</B ></P ><TABLE BORDER="0" BGCOLOR="#E0E0E0" CELLPADDING="5" ><TR ><TD ><PRE CLASS="shell" >@C:\php\php.exe script.php %1 %2 %3 %</PRE ></TD ></TR ></TABLE ></DIV ></TD ></TR ></TABLE > </P ><P > 假设将上述程序命名为 <TT CLASS="filename" >script.php</TT >,且 CLI 版的 <TT CLASS="filename" >php.exe</TT > 文件放置在 <TT CLASS="filename" >c:\php\cli\php.exe</TT >,该批处理文件会帮助将附加的参数传给脚本程序:<B CLASS="command" >script.bat echothis</B > 或 <B CLASS="command" >script.bat -h</B >。 </P ><P > 请参阅 <A HREF="ref.readline.html" >Readline</A > 扩展模块的有关文档,以获取更多的函数的信息。这些函数可以帮助完善 PHP 命令行应用程序。 </P ></DIV ><DIV CLASS="NAVFOOTER" ><HR ALIGN="LEFT" WIDTH="100%"><TABLE SUMMARY="Footer navigation table" WIDTH="100%" BORDER="0" CELLPADDING="0" CELLSPACING="0" ><TR ><TD WIDTH="33%" ALIGN="left" VALIGN="top" ><A HREF="features.safe-mode.functions.html" ACCESSKEY="P" >上一页</A ></TD ><TD WIDTH="34%" ALIGN="center" VALIGN="top" ><A HREF="index.html" ACCESSKEY="H" >起始页</A ></TD ><TD WIDTH="33%" ALIGN="right" VALIGN="top" ><A HREF="funcref.html" ACCESSKEY="N" >下一页</A ></TD ></TR ><TR ><TD WIDTH="33%" ALIGN="left" VALIGN="top" >被安全模式限制或屏蔽的函数</TD ><TD WIDTH="34%" ALIGN="center" VALIGN="top" ><A HREF="features.html" ACCESSKEY="U" >上一级</A ></TD ><TD WIDTH="33%" ALIGN="right" VALIGN="top" >函数参考</TD ></TR ></TABLE ></DIV ></BODY ></HTML >