其他功能

NO IMAGE

轉自php中文使用者5. 其他雜項
5.1 生成影象

PHP可以操作處理影象。如果你已經安裝了GD庫,你甚至可以利用PHP生成影象。
<?
Header("Content-type: image/gif");
$string=implode($argv," ");
$im = imagecreatefromgif("images/button1.gif");
$orange = ImageColorAllocate($im, 220, 210, 60);
$px = (imagesx($im)-7.5*strlen($string))/2;
ImageString($im,3,$px,9,$string,$orange);
ImageGif($im);
ImageDestroy($im);
?>
(譯者注:以上程式碼段缺少註釋,請讀者參考PHP Manual的影象處理函式部分)
這段程式碼在其他頁面中通過以下標記<img src="button.php3?text">呼叫,然後以上的那段button.php3程式碼取得text值並在另外取得的影象檔案中加上該值–在以上的程式碼中該影象檔案是images/button1.gif–最後輸出到瀏覽器。假如你想在表單域中使用影象按鈕,但是又不希望在每次按鈕上的文字改變後不得不重新生成新的影象,就可以利用這樣簡單的方法動態生成影象檔案。

5.2 Cookies

PHP支援基於HTTP的cookies。在需要時你可以像使用一般變數一樣方便的使用cookie。Cookies是瀏覽器儲存於客戶端的一些資訊片段,由此你可以知道是否一臺特定PC上的任何人都訪問過你的站點,瀏覽者者在你的站點上的蹤跡等等。使用cookies的典型例子就是對瀏覽者偏好的甄別。Cookies由函式setcookie()設定。與輸出HTTP標頭的函式header()一樣,setcookie()必須在任何實際內容杯輸出到瀏覽器之前呼叫。以下是一個簡單例子:
<?
if (empty($VisitedBefore))
{
// 如果沒有設定cookie,為cookie賦上當前時間值
// 函式中的最後一個引數宣告瞭該cookie儲存的時間
// 在這個例子中是1年
// time()函式返回自1970年1月1日以來的以秒數計的時間
SetCookie("VisitedBefore",time(), time() (60*60*24*365));
}
else
{
// 歡迎瀏覽者再次光臨
echo "Hello there, welcome back<BR>";
// 讀取cookie並判斷
if ( (time() – $VisitedBefore) >= "(60*60*24*7)" )
echo "Why did you take a week to come back. You should be here more often!? ";
}
?>

5.3 基於HTTP驗證

基於HTTP驗證當PHP以CGI模式執行時不能實現。我們可以使用函式header()傳送HTTP標頭強制驗證,客戶端瀏覽器則彈出供輸入使用者名稱和密碼的對話方塊。這兩個變數被儲存在$PHP_AUTH_USER和$PHP_AUTH_PW中,你可以使用這兩個變數驗證合法並允許進入。以下的例子通過使用者名稱稱/密碼對為tnc/nature的驗證一名使用者的登入:
<?
if(!isset($PHP_AUTH_USER))
{
Header("WWW-Authenticate: Basic realm=\"My Realm\"");
Header("HTTP/1.0 401 Unauthorized");
echo "Text to send if user hits Cancel button\n";
exit;
}
else
{
if ( !($PHP_AUTH_USER=="tnc" && $PHP_AUTH_PW=="nature") )
{
// 如果是錯誤的使用者名稱稱/密碼對,強制再驗證
Header("WWW-Authenticate: Basic realm=\"My Realm\"");
Header("HTTP/1.0 401 Unauthorized");
echo "ERROR : $PHP_AUTH_USER/$PHP_AUTH_PW is invalid.";
exit;
}
else
{
echo "Welcome tnc!";
}
?>
事實上再實際引用中不大可能如上面使用程式碼段明顯的使用者名稱稱/密碼對,而是利用資料庫或者加密的密碼檔案存取它們。

5.4 檔案上傳

你可以利用PHP實現檔案的功能,注意客戶端的瀏覽器應該是Netscape3以上或者IE3以上。以下就是該功能的簡單演示:
( upload.html ):
<HTML>
<HEAD>
<TITLE>Upload Your File</TITLE>
</HEAD>
<BODY>
<FORM ACTION="receiver.php3"
ENCTYPE="multipart/form-data" METHOD=POST>
<INPUT TYPE="HIDDEN"
NAME="MAX_FILE_SIZE" VALUE="2000000">
<INPUT TYPE="FILE"
NAME="uploadfile" SIZE="24" MAXLENGTH="80">
<BR><BR>
<INPUT TYPE="SUBMIT" VALUE="Upload File!"
NAME="sendit">
<INPUT TYPE="SUBMIT" VALUE="Cancel"
NAME="cancelit"><BR>
</FORM>
<I><FONT SIZE="2">(You may notice a slight
delay while we upload your file.)</FONT></I>
</BODY>
</HTML>

下面是處理上傳的檔案:
( receiver.php3 ):
<?
function do_upload ()
{
global $uploadfile, $uploadfile_size;
global $local_file, $error_msg;
if ( $uploadfile == "none" )
{
$error_msg = "You did not specify a file for uploading.";
return;
}
if ( $uploadfile_size > 2000000 )
{
$error_msg = "Sorry, your file is too large.";
return;
}
$the_time = time ();
// 你需要對以下目錄有寫許可權
$upload_dir = "/local/uploads";
$local_file = "$upload_dir/$the_time";
if ( file_exists ( ‘$local_file’ ) )
{
$seq = 1;
while ( file_exists ( "$upload_dir/$the_time$seq" ) ) { $seq ; }
$local_file = "$upload_dir/$the_time$seq";
};
rename ( $uploadfile, $local_file );
display_page ();
}
function display_page ()
{
// 這裡是你的頁面內容
}
<HTML>
<HEAD>
<TITLE>php3 Receiving Script</TITLE>
</HEAD>
<BODY>
<?
if ( $error_msg ) { echo "<B>$error_msg</B><BR><BR>"; }
if ( $sendit )
{
do_upload ();
}
elseif ( $cancelit )
{
header ( "Location: $some_other_script" );
exit;
}
else
{
some_other_func ();
}
?>
</BODY>
</HTML>

5.5 常用函式

我們簡單來看看一些常用的函式。

陣列

array – 生成陣列
count – 陣列元素個數
sort – 陣列排序,另有其他幾種排序函式可供使用
list – 列出陣列元素
each – 返回下一個key/value對
current – 返回當前陣列元素
next,prev – 傳回當前陣列元素前後指標

日期和時間

checkdate – 驗證日期/時間格式
date – 生成日期/時間格式
time – 當前時間資訊
strftime – 格式化日期/時間

目錄、檔案系統

chdir – 改變目錄
dir – 目錄類別
opendir, readdir, closedir – 開啟、讀取、關閉目錄
fopen, fclose – 開啟、關閉檔案
fgets, fgetss – 逐行讀取內容
file – 將整個檔案讀入一個陣列變數中

正規表示式

ereg – 匹配正規表示式
eregi – 大小寫非敏感匹配正規表示式
ereg_replace -匹配正規表示式並替換
eregi_replace -大小寫非敏感匹配正規表示式並替換
split – 依規則切開字串並以陣列形勢儲存

字串

AddSlashes – 加上斜槓後使用字串
echo – 輸出一個或多個字串
join, implode – 將陣列元素合併為字串
htmlentities, htmlspecialchars – 將HTML特殊字元轉換為HTML標記形式
split – 依規則切開字串並以陣列形勢儲存
5.6 擴充套件我們的範例主頁

我們將使用以上提到的一些函式和思想為我們的範例主頁新增更多的動態內容。我們可以在每個頁面的頂部加上導航欄,同時使得當前頁自動的不被連結顯示;同時還可以新增一個使用者驗證表單以便上傳音樂、影象等檔案並自動更新頁面。

導航欄

實際上就是在footer.inc檔案中加上一段程式碼。假設你的web站點中所有字尾為.php3的檔案都會出現在導航欄中,以下就是被存為include/navbar.inc的程式碼:
<?
/* 輸出該導航欄,連結所有除當前頁的站內.php3檔案 */
# 讀取目錄
$d = dir("./");
echo "<P ALIGN=\"CENTER\"> | \n";
while($entry = $d->read())
{
// 忽略無檔案情況
if ( !is_file($entry) )
continue;
/* 將檔名與副檔名分開。由於.是正規表示式特殊字元,應該用\引出 */
list($filenm, $fileext) = split("\.",$entry, 2);
// 忽略非.php3檔案情況
if( $fileext != "php3" )
continue;
/* 現在我們已經把.php3檔案都選出,下面搜尋檔案中的第一行(標題)
類似$title="something";
並將以上標題內容分開,用作連結文字 */
$linknm = "";
$fp=fopen($entry,"r");
while($buffer=fgets($fp, 4096))
{
$buffer = trim($buffer);
// 我們已經把每個檔案的標題放在檔案的第一行以便搜尋
// 但是當你改變變數名稱時可能會帶來大麻煩
if (ereg("title *= *\"", $buffer))
{
/* 我們已經取得了標題內容並可以在此基礎上
進行去除空格等處理。
必須以PHP程式碼方式處理,比如$title = "blah blah" */
eval($buffer);
// 然後將連結文字顯示為標題文字
$linknm = $title;
break;
}
}
fclose($fp);
if ( $entry == basename($PHP_SELF) )
echo "$linknm";
else
echo "<A HREF=\"$entry\">$linknm</A>";
echo " | ";
}
$d->close();
echo " </P>\n";
?>

照片收藏夾

我們將引用基於HTTP的驗證、檔案系統函式和檔案上傳功能維護放置影象檔案的目錄。
同時我們需要建立一個可以列出在該目錄下所有照片的頁面。

檔案上傳
<?
include("include/common.inc");
// 我們在這裡再做一次使用者驗證
if(!isset($PHP_AUTH_USER))
{
Header("WWW-Authenticate: Basic realm=\"$MySiteName\"");
Header("HTTP/1.0 401 Unauthorized");
echo "Sorry, you are not authorized to upload files\n";
exit;
}
else
{
if ( !($PHP_AUTH_USER==$MyName && $PHP_AUTH_PW==$MyPassword ) )
{
// 如果是錯誤的使用者名稱稱/密碼對,強制再次認證
Header("WWW-Authenticate: Basic realm=\"My Realm\"");
Header("HTTP/1.0 401 Unauthorized");
echo "ERROR : $PHP_AUTH_USER/$PHP_AUTH_PW is invalid.<P>";
exit;
}
}
if ( $cancelit )
{
// 當瀏覽者按下"取消"按鈕則轉向首頁面
header ( "Location: front_2.php3" );
exit;
}
function do_upload () {
global $userfile, $userfile_size, $userfile_name, $userfile_type;
global $local_file, $error_msg;
global $HTTP_REFERER;
if ( $userfile == "none" ) {
$error_msg = "You did not specify a file for uploading.";
return;
}
if ( $userfile_size > 2000000 )
{
$error_msg = "Sorry, your file is too large.";
return;
}
// Wherever you have write permission below…
$upload_dir = "photos";
$local_file = "$upload_dir/$userfile_name";
if ( file_exists ( $local_file ) ) {
$error_msg = "Sorry, a file with that name already exists";
return;
};
// 你還可以由此檢查檔名稱/型別對以確定是何種檔案:gif,jpg,mp3…
rename($userfile, $local_file);
echo "The file is uploaded<BR>\n";
echo "<A HREF=\"$HTTP_REFERER\">Go Back</A><BR>\n";
}
$title = "Upload File";
include("include/header.inc");
if (empty($userfile) || $userfile=="none")
{
// 輸出以下表單
?>
<FORM ACTION="<? echo "$PHP_SELF"; ?>" ENCTYPE="multipart/form-data" METHOD=POST>
<INPUT TYPE="HIDDEN" NAME="MAX_FILE_SIZE" VALUE="2000000">
<INPUT TYPE="FILE" NAME="userfile" SIZE="24" MAXLENGTH="80">
<BR><BR>
<INPUT TYPE="SUBMIT" VALUE="Upload File!" NAME="sendit">
<INPUT TYPE="SUBMIT" VALUE="Cancel" NAME="cancelit"><BR>
</FORM>
<I><FONT SIZE="2">(You may notice a slight delay while we upload your file.)</FONT></I>
<?
} else {
if ( $error_msg ) { echo "<B>$error_msg</B><BR><BR>"; }
if ( $sendit ) {
do_upload ();
}
}
include("include/footer.inc");
?>

照片相簿

<?
include("include/common.inc");
$title = "Gallery";
include("include/header.inc");
?>
<P>
Here are some of our family photos. This PHP script can really
be made better, by splitting into multiple pages.
</P>
<?
$d = dir("photos");
while($entry = $d->read())
{
if (is_file("photos/$entry"))
echo "<IMG SRC=\"photos/$entry\">\n";
}
$d->close();
?>
<?
include("include/footer.inc");
?>

另外,你可以在檔案上傳的表單中加上一個輸入元素去描述該上傳的檔案。這個元素將被儲存在檔案中,然後被以上的照片相簿的那段程式碼所讀出並顯示出來。