PHP CLI SAPI 允許您開發 PHP 支持的 shell 腳本,甚至是基於桌面的腳本。事實上,可以用PHP命令行運行的工具。采用這種方式,PHP 開發人員可以像 Perl、AWK、Ruby 或 shell 程序員一樣高效。本文探究構建到 PHP 中的工具,讓您了解 PHP 運行的底層 shell 環境和文件系統。PHP 為執行外部命令提供大量函數,其中包括 shell_exec()、exec()、passthru() 和 system()。這些命令是相似的,但為您運行的外部程序提供不同的界面。所有這些命令都衍生一個子進程,用於運行您指定的命令或腳本,並且每個子進程會在命令輸出寫到標准輸出 (stdout) 時捕捉它們。
PHP CLI SAPI 允許您開發 PHP 支持的 shell 腳本,甚至是基於桌面的腳本。事實上,可以用PHP命令行運行的工具。采用這種方式,PHP 開發人員可以像 Perl、AWK、Ruby 或 shell 程序員一樣高效。本文探究構建到 PHP 中的工具,讓您了解 PHP 運行的底層 shell 環境和文件系統。PHP 為執行外部命令提供大量函數,其中包括 shell_exec()、exec()、passthru() 和 system()。這些命令是相似的,但為您運行的外部程序提供不同的界面。所有這些命令都衍生一個子進程,用於運行您指定的命令或腳本,並且每個子進程會在命令輸出寫到標准輸出 (stdout) 時捕捉它們。
shell_exec()
清單 1. 使用反撇號計算單詞數量
復制代碼 代碼如下:
#! /bin/sh
number_of_words=`wc -w *.txt`
echo $number_of_words#result would be something like:
#165 readme.txt 388 results.txt 588 summary.txt
#and so on....
在您的 PHP 腳本中,您可以在 shell_exec() 中運行這個簡單的命令,如清單 2 所示,並獲取想要的結果。這裡假設在同一個目錄下有一些文本文件。
清單 2. 在 shell_exec() 中運行相同的命令
復制代碼 代碼如下:
<?php教程
$results = shell_exec('wc -w *.txt');
echo $results;
?>
在圖 1 中可以看到,獲得的結果與從 shell 腳本得到的一樣。這是因為 shell_exec() 允許您通過 shell 運行外部程序,然後以字符串的形式返回結果。
圖 1. 通過 shell_exec() 運行 shell 命令的結果
注意,僅使用後撇號操作符也會得到相同的結果,如下所示。
清單 3. 僅使用後撇號操作符
復制代碼 代碼如下:
<?php
$results = `wc -w *.txt`;
echo $results;
?>
清單 4 給出了一種更加簡單的方法。
清單 4. 更加簡單的方法
復制代碼 代碼如下:
<?php
echo `wc -w *.txt`;
?>
通過 UNIX 命令行和 shell 腳本能夠完成很多東西,知道這點很重要。例如,您可以使用豎線將命令連接起來。您甚至可以使用操作符在其中創建 shell 腳本,並且僅調用 shell 腳本(根據需要使用或不使用參數)。
例如,如果您僅希望計算該目錄下的前 5 個文本文件的單詞數,那麼可以使用豎線 (|) 將 wc 和 head 命令連接起來。另外,您還可以將輸出結果放到 pre 標記內部,讓它能夠更美觀地呈現在 Web 浏覽器中,如下所示。
清單 5. 更加復雜的 shell 命令
復制代碼 代碼如下:
<?php
$results = shell_exec('wc -w *.txt | head -5');
echo "<code lang="php">".$results . "</code>";
?>
圖 2 演示了運行清單 5 的腳本得到的結果。
圖 2. 從 shell_exec() 運行更復雜的 shell 命令得到的結果
在本文的後面部分,您將學習如何使用 PHP 為這些腳本傳遞參數。現在您可以將它看作運行 shell 命令的一種方法,但要記住您只能看到標准輸出。如果命令或腳本出現錯誤,您將看不到標准的錯誤 (stderr),除非您通過豎線將它添加到 stdout。
官網參考
<?php
define("__USED_CHARS__", "abcdefghijklmnopqrstuvwxyz0123456789");
define("__CASE_SENSITIVE__", true); // Use string above or use uppercase / lowercase variant$bf = new chargen(2); // new chargen object, length 2
$bf->generate("whois"); // generate chars and call whois functionfunction whois($str)
{
$domain = $str.".com";$retval = shell_exec("whois $domain");
if (eregi("no match", $retval))
echo $domain." ist availablen";
else
echo $domain." is unavailablen";
}class chargen
{
private $chars = NULL;
private $maxlength = NULL;protected $buffer = NULL;
function generate($mycallback = false)
{
foreach ($this->buffer as $char)
{
foreach ($this->chars as $nextchar)
{
$retval = $char.$nextchar;
$this->buffer[$retval] = $retval;if ($mycallback && function_exists($mycallback))
$mycallback($retval);
else
echo $retval."n";
}
}if (strlen($retval) == $this->maxlength)
return;$this->generate($mycallback);
}function __construct($maxlength = 8)
{
$chars = array();$this->buffer = array();
array_push($this->buffer, "");for ($i = 0; $i < strlen(__USED_CHARS__); $i++)
{
$index = substr(__USED_CHARS__, $i, 1);if (__CASE_SENSITIVE__)
{
$this->chars[$index] = $index;
}
else
{
$this->chars[strtolower($index)] = strtolower($index);
$this->chars[strtoupper($index)] = strtoupper($index);
}
}$this->maxlength = $maxlength;
}
}
?>