使用 PHP

本节汇集了你在写 PHP 脚本时可能碰到的大多数普通错误。

  1. 我忘了PHP函数的参数顺序,它们是随机的吗?
  2. 我想写一个可以处理任何表单来的数据的通用 PHP 脚本。我怎么知道哪个 POST 方法变量可用呢?
  3. 我需要在所有的单引号(')前加一个反斜线(\),使它们变成(\'),我如何能够通过正则表达式来实现?我同样希望能够将(")转换成(\"),将(\)转换成(\\)。
  4. 我所有的(")和(')都被变成了(\")和(\'),我如何才能去掉这些不必要的反斜线?它们为什么及如何出现?
  5. PHP 选项 register_globals 对我有什么影响?
  6. 当我这样做时,输出显示的次序是错的: <?php function myfunc($argument) { echo $argument + 10; } $variable = 10; echo "myfunc($variable) = " . myfunc($variable); ?> 这是怎么回事?
  7. 下面代码怎么没有分成两行显示? <pre> <?php echo "This should be the first line."; ?> <?php echo "This should show up after the new line above."; ?> </pre>
  8. 我得到消息“Warning: Cannot send session cookie - headers already sent...”或者“Cannot add header information - headers already sent...”。
  9. 我需要直接访问请求报头中的信息,怎么能办到?
  10. 当我用 IIS 进行 HTTP 认证时得到“No Input file specified”消息。
  11. Windows:不能访问另一台电脑上用 IIS 共享的文件。
  12. 我怎样混合使用 XML 和 PHP?它不认我的 <?xml 标记!
  13. 怎样用 FrontPage 或者其它非要把我的代码移来移去的 HTML 编辑器来编辑 PHP?
  14. 哪里可以找到所有可用的 PHP 预定义变量的完整列表?
  15. 怎样才能不用非免费的商业库(例如 PDFLib) 来生成 PDF 文档?我想要个免费的并且不需要再连接别的 PDF 库。
  16. 我试着在用户自定义函数中访问一个标准的 CGI 变量(例如 $DOCUMENT_ROOT 或 $HTTP_REFERER),但是找不到,哪里出了错?
  17. 有些 PHP 选项可以接受缩写的字节值,与仅能接受 integer字节值相反。都有哪些缩写字节值?可以将其用于 php.ini 之外吗?
  18. Windows: I keep getting connection timeouts when using localhost, whereas "127.0.0.1"works?
我忘了PHP函数的参数顺序,它们是随机的吗?

PHP is a glue that brings together hundreds of external libraries, so sometimes this gets messy. However, a simple rule of thumb is as follows:

Array functionparameters are ordered as " needle, haystack" whereas String functionsare the opposite, so " haystack, needle".

我想写一个可以处理任何表单来的数据的通用 PHP 脚本。我怎么知道哪个 POST 方法变量可用呢?

PHP 提供很多" 预定义变量",例如超全局变量 $_POST。可以遍历 $_POST变量,因为它是一个和所有通过 POST 方法传递数据相联系的数组。例如,可以用 foreach简单地遍历它,检查 empty()值,以及将它们输出。

<?php
$empty 
$post = array();
foreach (
$_POST as $varname => $varvalue) {
    if (empty(
$varvalue)) {
        
$empty[$varname] = $varvalue;
    } else {
        
$post[$varname] = $varvalue;
    }
}

print 
"<pre>";
if (empty(
$empty)) {
    print 
"None of the POSTed values are empty, posted:\n";
    
var_dump($post);
} else {
    print 
"We have " count($empty) . " empty values\n";
    print 
"Posted:\n"var_dump($post);
    print 
"Empty:\n";  var_dump($empty);
    exit;
}
?>

Note: Superglobal 可用性说明:

自 PHP 4.1.0 起可以使用 Superglobal 数组,例如 $_GET$_POST,和 $_SERVER,等等。更多信息请阅读手册中的 superglobals 章节。

我需要在所有的单引号(')前加一个反斜线(\),使它们变成(\'),我如何能够通过正则表达式来实现?我同样希望能够将(")转换成(\"),将(\)转换成(\\)。

函数 addslashes()能够实现这种操作。请参阅函数 mysql_escape_string()。另外,还可以用函数 stripslashes()来去掉反斜线。

Note: 设置选项说明: magic_quotes_gpc

设置选项magic_quotes_gpc默认值为 on。相当于对所有 GET, POST, and COOKIE 数据使用了 addslashes()。可以用 stripslashes() 来去掉它们。

我所有的(")和(')都被变成了(\")和(\'),我如何才能去掉这些不必要的反斜线?它们为什么及如何出现?

PHP 函数 stripslashes()能够从 string中去掉所有的反斜线。这些反斜线出现最有可能的原因是 PHP 设置项 magic_quotes_gpc被开启。

Note: 设置选项说明: magic_quotes_gpc

设置选项magic_quotes_gpc默认值为 on。相当于对所有 GET, POST, and COOKIE 数据使用了 addslashes()。可以用 stripslashes() 来去掉它们。

PHP 选项 register_globals 对我有什么影响?

首先要理解这个设置选项的作用。假如我们使用以下URL: http://example.com/foo.php?animal=cat,那么在 foo.php中我们可能会使用以下代码:

<?php
// 建议使用这种访问变量的方式
echo $_GET['animal'];

// 如果想直接访问$animal变量,就要把register_globals选项设置为on
// 强烈建议不要这么做!!
echo $animal;

// 这个选项的值会影响到所有变量,包括$_SERVER
// 以下是该选项设置为off时的正确写法
echo $_SERVER['PHP_SELF'];

// 同样,要使$PHP_SELF变量自动生效,register_globals选项必须设置为on
// 强烈建议不要这么做!!
echo $PHP_SELF;
?>

上面的代码解释了register_globals的作用,就是自动生成变量。这种编程方式被很多人所不喜,所以在PHP当中register_globals默认设置为off。在PHP6,这个选项被删除了。所以目前绝大部分虚拟主机都默认把register_globals禁用。请注意,你在阅读过时的文章、教程时可能需要把该选项开启,否则可能示例代码不能通过。但在实际开发当中,强烈建议将其设置为off,否则可能会导致安全问题。

请参阅以下资源进一步了解:

Note:

以上示例中,我们使用了一个 URL,其中包含了一个QUERY_STRING。PHP会把QUERY_STRING中的信息通过GET HTTP请求传递,所以我们可以通过超级全局变量(superglobal) $_GET来访问其中的变量。

当我这样做时,输出显示的次序是错的:
<?php
function myfunc($argument)
{
    echo 
$argument 10;
}
$variable 10;
echo 
"myfunc($variable) = " myfunc($variable);
?>
这是怎么回事?

要在一个表达式中(例如在上面的例子中和其它字符串连接)使用函数的结果,需要 return这个值,而不是 echo它。

下面代码怎么没有分成两行显示?
<pre>
<?php echo "This should be the first line."?>
<?php 
echo "This should show up after the new line above."?>
</pre>

在 PHP 中,一段代码的结束标记要么是"?>"要么是"?>\n"(\n 表示换行)。因此在上面的例子中,输出的句子将显示在同一行中,因为 PHP 忽略了代码结束标记后面的换行。这意味着如果要输出一个换行符,需要在每段 PHP 代码的结束标记后面多加一个换行。

PHP 为什么这么做呢?因为在格式化正常的 HTML 时,这样通常会更容易。假如输出了换行而你不需要这个换行时,就不得不用一个非常长的行来达到这样的效果,或者让产生的 HTML 页面的源文件的格式很难读。

我得到消息"Warning: Cannot send session cookie - headers already sent..."或者"Cannot add header information - headers already sent..."。

函数 header()setcookie()session 函数需要在输出流中添加头信息。但是头信息只能在其它任何输出内容之前发送。在使用这些函数前不能有任何(如 HTML)的输出。函数 headers_sent()能够检查脚本是否已经发送了头信息。请参阅 输出控制函数

我需要直接访问请求报头中的信息,怎么能办到?

如果以 Apache 的模块方式运行 PHP,那么函数 getallheaders()可以做这件事。因此下面的代码可以显示所有的请求报头:

<?php
$headers 
getallheaders();
foreach (
$headers as $name => $content) {
    echo 
"headers[$name] = $content<br />\n";
}
?>

请参阅函数 apache_lookup_uri()apache_response_headers()fsockopen()

当我用 IIS 进行 HTTP 认证时得到"No Input file specified"消息。

IIS 的安全模型这里有毛病。这是所有 IIS 下运行的 CGI 程序所共有的问题。一个解决办法是建立一个纯 HTML 文件(不经过 PHP 解析)作为要进入认证目录的登录页面,然后用 META 标记来重定向到 PHP 页面,或者用一个连接到 PHP 页面。然后 PHP 就可以正确识别认证信息了。如果是用 ISAPI 模块,那没有这个问题。其它 NT 下的 web 服务器不受此影响。更多信息见 » http://support.microsoft.com/kb/q160422/HTTP 认证一章。

Windows:不能访问另一台电脑上用 IIS 共享的文件。

必须要做些修改。打开 Internet 信息服务,找到你的 PHP 文件并打开属性页,选择 文件安全性标签,点击 匿名访问和身份验证控制的"编辑"按钮。

解决此问题有两个方法,一是不要选中 匿名访问并且选中 集成 Windows 身份验证,二是选中 匿名访问并编辑匿名访问使用的帐户,改成一个有权限的。

我怎样混合使用 XML 和 PHP?它不认我的 <?xml 标记!

要能够在 PHP 代码中直接嵌入 <?xml,需要将 PHP 设置项 short_open_tags设置为 0以关闭短标记格式。无法通过函数 ini_set()来更改这项设置。不管 short_open_tags是开或者关,都可以用类似于 <?php echo '<?xml'; ?>的方法达到目的。该项设置的默认值为开。

怎样用 FrontPage 或者其它非要把我的代码移来移去的 HTML 编辑器来编辑 PHP?

最简单的方法是让 PHP 代码也能使用 ASP 标记。这可以让你用 ASP 风格的 <% 和 %> 代码定界符。一些流行的 HTML 编辑器在处理此格式上更加智能化一些(目前如此)。要使用 ASP 风格的标记,需要在 php.ini 中打开 asp_tags,或者用相应的 Apache 配置选项。

哪里可以找到所有可用的 PHP 预定义变量的完整列表?

请阅读手册中的 预定义变量一章,该部分的文档已经包含了一部分可以用于你的脚本的预定义变量的列表。可用变量的完整列表(及更多信息)可以通过调用 phpinfo()函数来查阅。请务必阅读手册中 来自 PHP 之外的变量一节,这部分文档阐述了外部变量的概念,例如来自 HTML 表单、Cookie 和 URL 的变量。

Note: register_globals 重要说明:

自 PHP 4.2.0 起,PHP 指令 register_globals 的默认值为 off。PHP 社区鼓励开发者不要依赖于此指令, 用其他手段替代,例如superglobals

怎样才能不用非免费的商业库(例如 PDFLib) 来生成 PDF 文档?我想要个免费的并且不需要再连接别的 PDF 库。

有几个选择,例如 » FPDF» TCPDF模块。

There is also the Haru extension that uses the free libHaru external library.

我试着在用户自定义函数中访问一个标准的 CGI 变量(例如 $DOCUMENT_ROOT 或 $HTTP_REFERER),但是找不到,哪里出了错?

首先非常重要的一点是 PHP 设置项 register_globals同样会对服务器端和环境变量产生影响。当 register_globals = off (从 PHP 4.2.0 开始其默认值为 off),变量 $DOCUMENT_ROOT将不会存在,而只有 $_SERVER['DOCUMENT_ROOT']。如果 register_globals = on 则变量 $DOCUMENT_ROOT$GLOBALS['DOCUMENT_ROOT']将同时存在。

如果确认 register_globals = on 但不知道为什么 $DOCUMENT_ROOT在函数内部不可用,这是因为它和其它的变量一样需要在函数中执行 global $DOCUMENT_ROOT。请参阅手册中的 变量范围一节。建议在 register_globals = off 的情况下进行编码。

Note: Superglobal 可用性说明:

自 PHP 4.1.0 起可以使用 Superglobal 数组,例如 $_GET$_POST,和 $_SERVER,等等。更多信息请阅读手册中的 superglobals 章节。

有些 PHP 选项可以接受缩写的字节值,与仅能接受 integer字节值相反。都有哪些缩写字节值?可以将其用于 php.ini 之外吗?

可用的选择有 K(对应 Kilobytes),M(对应 Megabytes)和 G(对应 Gigabytes;自 PHP 5.1.0 起可用),区分大小写。其余的都认为是字节。 1M等于一个 Megabyte,即 1048576字节。 1K等于一个 Kilobyte,即 1024字节。不能在 php.ini 之外使用这些符号,最好用整数的 integer字节值。参见 ini_get()文档中的转换示例。

Note: kilobyte versus kibibyte

The PHP notation describes one kilobyte as equalling 1024 bytes, whereas the IECstandard considers this to be a kibibyte instead. Summary: k and K = 1024 bytes.

Windows: I keep getting connection timeouts when using localhost, whereas "127.0.0.1"works?

Prior to PHP 5.3.4, there was a bug in the network resolving code inside PHP that caused localhostin all stream related situations to fail if IPv6 was enabled. To work around this issue you can either use "127.0.0.1"or disable IPv6 resolving in the hostsfile.

User Contributed Notes

robjbrain at gmail dot com 10-Oct-2016 04:37
There is no error message given if you accidentally use KB, MB or GB instead of K, M or G. From what I can tell the alphabetic characters will be ignored and it will be treated as bytes, so 1GB actually equals 1 Byte. phpinfo() and ini_get() will report what you have written e.g. 1GB, so it will not be clear that there is anything wrong with your setting even though there is.