前言
本文主要給大家介紹了關(guān)于php中文亂碼問題的終極解決方案,分享出來供大家參考學習,在開始之前,我們先來談談為什么會出現(xiàn)中文亂碼?
很多新手朋友學習PHP的時候,發(fā)現(xiàn)程序中的中文在輸出的時候會出現(xiàn)亂碼的問題,那么為什么會出現(xiàn)這種亂碼的情況呢?一般來說,亂碼的出現(xiàn)有2種原因,一種是由于編碼(charset) 設(shè)置錯誤,導致瀏覽器以錯誤的編碼來解析,從而出現(xiàn)了滿屏亂七八糟的“天書”,第二種就是文件被以錯誤的編碼打開,然后保存,比如一個文本文件原先是GB2312編碼的,卻以UTF-8編碼打開再保存,就會出現(xiàn)亂碼的問題。本篇文章,就帶大家了解一下,怎么解決php中亂碼的問題。
我們將亂碼情況分為以下幾種,有需要的可以對照下面的幾種情況有針對性的解決亂碼問題
第一種:解決HTML中中文亂碼問題方法
1. php文件本身的編碼與網(wǎng)頁的編碼應匹配
a. 如果欲使用gb2312編碼,那么php要輸出頭:header(“Content-Type: text/html; charset=gb2312"),靜態(tài)頁面添加<meta http-equiv="Content-Type" content="text/html; charset=gb2312">,所有文件的編碼格式為ANSI,可用記事本打開,另存為選擇編碼為ANSI,覆蓋源文件。
b. 如果欲使用utf-8編碼,那么php要輸出頭:header(“Content-Type: text/html; charset=utf-8"),靜態(tài)頁面添加<meta http-equiv="Content-Type" content="text/html; charset=utf-8">,所有文件的編碼格式為utf-8。保存為utf-8可能會有點麻煩,一般utf-8文件開頭會有BOM,如果使用session就會出問題,可用editplus來保存,在editplus中,工具->參數(shù)選擇->文件->UTF-8簽名,選擇總是刪除,再保存就可以去掉BOM信息了。
2. php本身不是Unicode的,所有substr之類的函數(shù)得改成mb_substr(需要裝mbstring擴展);或者用iconv轉(zhuǎn)碼。
如果你的HTML文件文件出現(xiàn)了亂碼問題,那么你可以在head標簽里面加入UTF8編碼(國際化編碼):UTF-8是沒有國家的編碼,也就是獨立于任何一種語言,任何語言都可以使用的。
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
示例
我們現(xiàn)在的HTML5文件,設(shè)置編碼更為簡單,像下面這樣
第二種、HTML和PHP混合的頁面解決方案
如何是HTML和PHP混編,除了按照第一個方法所說的操作之外,還需要在PHP文件的最上面加入這句代碼:
<?php header("content-type:text/html;charset=utf-8"); //設(shè)置編碼 ?>
第三種、純PHP頁面的中文亂碼問題(數(shù)據(jù)是靜態(tài)的)
如果你的PHP頁面出現(xiàn)了亂碼,只需要在頁面的開始處加入下面代碼就可以了。
<?php header("content-type:text/html;charset=utf-8"); //設(shè)置編碼 ?>
第四種、PHP+Mysql中文亂碼問題
這個除了按照第三種所說的操作之外,還要在你的數(shù)據(jù)查詢/修改/增加之前加入數(shù)據(jù)庫編碼。而且,值得注意的是,這里的UTF8和之前的不一樣,中間是沒有橫線的。
<?php mysql_query('SET NAMES UTF8'); //接下來的就是查出數(shù)據(jù)或者修改,增加 ?>
如何你使用的MySQL版本在 4.1或更高版本,可以在鏈接數(shù)據(jù)庫操作后,設(shè)置一個字符編碼,像下面這樣
UTF-8編碼只是其中一種編碼,如果不想使用utf-8編碼,也可以使用其他編碼,只需將UTF-8換成你想使用的編碼就可以,目前中文網(wǎng)站開發(fā)中主要用的是GB2312和UTF-8 兩種編碼。
有一點要注意:在需要做數(shù)據(jù)庫操作的php程序前加的 mysql_query("set names '編碼'");編碼,一定要和php編碼一致,如果php編碼是gb2312那mysql編碼就是gb2312,如果是utf-8那mysql編碼就是 utf8,這樣插入或檢索數(shù)據(jù)時就不會出現(xiàn)亂碼了
第五種.PHP與操作系統(tǒng)相關(guān)
Windows和Linux的編碼是不一樣的,在Windows環(huán)境下,調(diào)用PHP的函數(shù)時參數(shù)如果是utf-8編碼會出現(xiàn)錯誤,比如move_uploaded_file()、filesize()、readfile()等,這些函數(shù)在處理上傳、下載時經(jīng)常會用到,調(diào)用時可能會出現(xiàn)下面的錯誤:
Warning: move_uploaded_file()[function.move-uploaded-file]:failed to open stream: Invalid argument in ...
Warning: move_uploaded_file()[function.move-uploaded-file]:Unable to move '' to '' in ...
Warning: filesize() [function.filesize]: stat failed for ... in ...
Warning: readfile() [function.readfile]: failed to open stream: Invalid argument in ..
在Linux環(huán)境下用gb2312編碼雖然不會出現(xiàn)這些錯誤,但保存后的文件名出現(xiàn)亂碼導致無法讀取文件,這時可先將參數(shù)轉(zhuǎn)換成操作系統(tǒng)識別的編碼,編碼轉(zhuǎn)換可用mb_convert_encoding(字符串,新編碼,原編碼)或iconv(原編碼,新編碼,字符串),這樣處理后保存的文件名就不會出現(xiàn)亂碼,也可以正常讀取文件,實現(xiàn)中文名稱文件的上傳、下載。
其實還有更好的解決方法,徹底與系統(tǒng)脫離,也就不用考慮系統(tǒng)是何編碼??梢陨梢粋€只有字母和數(shù)字的序列作為文件名,而將原來帶有中文的名字保存在數(shù)據(jù)庫中,這樣調(diào)用move_uploaded_file()就不會出現(xiàn)問題,下載的時候只需將文件名改為原來帶有中文的名字。實現(xiàn)下載的代碼如下
header("Pragma: public");
header("Expires: 0");
header("Cache-Component: must-revalidate, post-check=0, pre-check=0");
header("Content-type: $file_type");
header("Content-Length: $file_size");
header("Content-Disposition: attachment; filename=\"$file_name\"");
header("Content-Transfer-Encoding: binary");
readfile($file_path);
$file_type是文件的類型,$file_name是原來的名字,$file_path是保存在服務上文件的地址。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對365建站網(wǎng)的支持
如對本文有疑問,請?zhí)峤坏浇涣髡搲?,廣大熱心網(wǎng)友會為你解答??! 點擊進入論壇