PHP聊天室技術(shù)(1)
PHP聊天室技術(shù) 黃國輝
1.前言 上網(wǎng)聊天是時(shí)下最流行的交友方式。各大網(wǎng)站推出的聊天室都各具特色。 聊天室主要分為WebChat、BBSChat兩種。BBSChat是基于Telnet的Tcp協(xié)議,是BBS的附設(shè)功能,需要客戶端Telnet程式。WebChat則采用瀏覽器方式,實(shí)際上是個(gè)多人一起使用的CGI程式。其基本原理是把每個(gè)用戶的發(fā)言通過瀏覽器傳給系統(tǒng),再由系統(tǒng)收集處理后分發(fā)給特定用戶。 WebChat一般采用Server Push或Client Pull技術(shù)。兩種技術(shù)的區(qū)別在于使用不同的方式將數(shù)據(jù)分發(fā)給用戶,Server Push是由服務(wù)器將數(shù)據(jù)以多重MIME編碼,推給(push)使用者端,現(xiàn)在較少網(wǎng)站使用這種方式。Client Pull則是用戶從服務(wù)器拉(pull)所要的數(shù)據(jù)。 最常用的Client Pull就是利用Html語言的Meta標(biāo)簽http-equiv='Refresh' 的屬性,每隔一段時(shí)間就檢查服務(wù)器上是否有新的數(shù)據(jù)。例如 ,每隔5秒鐘就會刷新一次頁面。這種方法簡單有效,缺點(diǎn)是刷新時(shí)會產(chǎn)生閃爍的現(xiàn)象;而且為了保持效率,每次刷新都會把舊的聊天內(nèi)容清除,用戶想查看或保留對話內(nèi)容都很不方便。為此,采用JavaApplet作為聊天室的前端,利用刷新把從服務(wù)器Pull的數(shù)據(jù)通過JavaApplet來顯示也是一種解決方案。更有就是本文要介紹的使聊天程式保持連線的方案。方法一,把聊天程式配置為無限大,就能使瀏覽器不停地保持下載的連線狀態(tài);方法二,聊天程式中存在著無限循環(huán),因?yàn)槟軌蚍奖愕嘏渲酶呒壍墓δ埽栽诖诉x擇采用。 Web服務(wù)器采用FreeBSD+Apache,原因是兩者的組合具備最強(qiáng)的性能,而且花費(fèi)為零。還需要考慮的是用什么方式來存放數(shù)據(jù)呢。用文檔相對而言容易實(shí)現(xiàn),但是多人使用,頻繁地對同一文檔進(jìn)行IO操作,難免會影響效率,況且FreeBSD的IO性能不佳。能夠考慮使用RamDisk,將文檔整個(gè)放進(jìn)內(nèi)存,以提高速度。或在內(nèi)存中劃分出一塊高速空間用以數(shù)據(jù)存放。筆者使用的是數(shù)據(jù)庫方式:MySQL。因?yàn)樵摂?shù)據(jù)庫就是為大批量用戶同時(shí)使用而設(shè)計(jì),利用他能夠省去設(shè)計(jì)高速空間操作的復(fù)雜編寫,速度也可得到確保。如將整個(gè)數(shù)據(jù)庫放進(jìn)內(nèi)存,效果更好。 程式用PHP+Html+JavaScript編寫。聊天室主要是對Html的Form中的各種元素進(jìn)行操作。JavaScript是基于對象的語言,對Html中的各種元素皆當(dāng)作對象看待,所以每個(gè)元素的方法和屬性都很豐富,操作比較方便。而PHP只有在Form經(jīng)過Post后,將Form中的元素轉(zhuǎn)成對應(yīng)的變量,才能處理用戶輸入的數(shù)據(jù)。從交互性來說比較差,這也是使用JavaScript的原因。采用PHP的原因是因?yàn)楸绕鹌渌鸆GI語言,其速度和安全性都較佳,研發(fā)也比較容易。
2. 不斷刷新的聊天室 一個(gè)標(biāo)準(zhǔn)的聊天室頁面由三個(gè)Frame組成,分別是顯示在線用戶的Online、用戶發(fā)言及功能配置的Say和顯示聊天內(nèi)容的List。用戶在Say Frame中敲入發(fā)言內(nèi)容后按發(fā)送,數(shù)據(jù)經(jīng)過處理后保存在Mysql數(shù)據(jù)庫,同時(shí)被保存的還包括發(fā)言人、聊天對象及發(fā)言的時(shí)間。用戶一進(jìn)入聊天室,List Frame從Mysql數(shù)據(jù)庫中把發(fā)言時(shí)間大于用戶進(jìn)入時(shí)間的發(fā)言提出顯示出來。而后續(xù)顯示新的發(fā)言內(nèi)容的關(guān)鍵,在于顯示聊天內(nèi)容的那段程式是無限循環(huán)的。 List Frame程式概要: $db=mysql_pconnect(localhost,root);;#Mysql數(shù)據(jù)庫連接 mysql_select_db(chat,$db); 顯示歡迎進(jìn)入聊天室 配置$init為數(shù)據(jù)庫中發(fā)言時(shí)間比 進(jìn)入時(shí)間大的第一個(gè)數(shù)據(jù)的ID號#是數(shù)據(jù)提取的標(biāo)志位 while (1==1) {;;#無限循環(huán)開始 提取從$init到最后的數(shù)據(jù); while (每一個(gè)數(shù)據(jù)$text) { $emote=strip_tags($text) #防止用戶直接輸入Html語言 if (eregi('^/',$emote)) #判斷發(fā)言是否系統(tǒng)命令(以/開頭) 系統(tǒng)特別處理 else 顯示發(fā)言 } 配置$init為最后數(shù)據(jù)的ID號 flush();;;;;#清出輸出緩沖,使發(fā)言馬上顯示 sleep(2);;;;;#使程式暫停2秒,節(jié)省系統(tǒng)資源 mysql_free_result($result);;;;;#釋放數(shù)據(jù)庫結(jié)果占用的內(nèi)存 } 因?yàn)槌淌綗o限循環(huán),在每次循環(huán)中輸出的發(fā)言是先置入輸出緩沖區(qū)內(nèi)。通過flush()馬上把緩沖區(qū)的內(nèi)容向用戶的List Frame送出,達(dá)到了實(shí)時(shí)的聊天效果。循環(huán)最后要釋放Mysql結(jié)果集占用的內(nèi)存,否則因?yàn)闊o限循環(huán)的緣故,系統(tǒng)資源很快會被耗盡。 用戶在Login后將建立一個(gè)Online的表用于在線用戶的統(tǒng)計(jì),主要是為了防止User表過大,程式中頻繁用到的檢索會拖慢系統(tǒng)的運(yùn)作。這其中用的最多的是Online Frame,在采用Client Pull的Refresh Meta,每隔一段時(shí)間就會查詢Online表,以刷新在線的用戶。假如用戶不發(fā)言超過規(guī)定時(shí)間,系統(tǒng)會調(diào)用自定義函數(shù)將用戶設(shè)為TimeOut,強(qiáng)制將其退出聊天室。
