Socket編程--使用Tcp實(shí)現(xiàn)簡(jiǎn)單的聊天程序

Socket

在Android聊天程序的實(shí)現(xiàn)中趾代,通常是通過http請(qǐng)求拉取最新聊天信息,由于http請(qǐng)求是無狀態(tài)(Stateless)的撒强,無法隨時(shí)獲知新消息的到達(dá),通常采用推送的方式告知客戶端有新的消息胚想。除了這種http+推送的方式外,也可以通過Socket編程實(shí)現(xiàn)聊天程序浊服。本文將對(duì)網(wǎng)絡(luò)層次進(jìn)行簡(jiǎn)單的講解胚吁,并結(jié)合一個(gè)簡(jiǎn)單的demo講解Socket編程的實(shí)現(xiàn)牙躺。

本文Github Demo地址

計(jì)算機(jī)網(wǎng)絡(luò)基礎(chǔ)知識(shí)

計(jì)算機(jī)網(wǎng)絡(luò)用于連接不同的計(jì)算機(jī)腕扶,并實(shí)現(xiàn)計(jì)算機(jī)間的數(shù)據(jù)傳輸。整個(gè)數(shù)據(jù)傳輸是個(gè)很復(fù)雜的過程乓搬,需要很多步驟進(jìn)行,這些步驟被劃分成了不同的層級(jí)进肯,所以計(jì)算機(jī)網(wǎng)絡(luò)有標(biāo)準(zhǔn)的層級(jí)結(jié)構(gòu),從底層到高層依次是物理層江掩,數(shù)據(jù)鏈路層,網(wǎng)絡(luò)層策泣,傳輸層抬吟,應(yīng)用層萨咕,如下圖:

計(jì)算機(jī)網(wǎng)絡(luò)五層協(xié)議

分層的優(yōu)點(diǎn)在于各層級(jí)之間是獨(dú)立的危队,靈活性好,在結(jié)構(gòu)上可分隔開茫陆,易于實(shí)現(xiàn)和維護(hù)擎析,同時(shí)能促進(jìn)每層的標(biāo)準(zhǔn)化工作。每個(gè)層次都有運(yùn)行在該層的一系列網(wǎng)絡(luò)協(xié)議揍魂,如網(wǎng)絡(luò)層的IP協(xié)議,傳輸層的TCP/UDP協(xié)議讨盒,應(yīng)用層的Http/SMTP/DNS協(xié)議步责,如下圖:

每層包含的協(xié)議

其中Http協(xié)議運(yùn)行在應(yīng)用層,典型的應(yīng)用是網(wǎng)絡(luò)瀏覽器蔓肯。Http協(xié)議是無狀態(tài)的,同一個(gè)客戶多次訪問同一個(gè)服務(wù)器上的頁面時(shí)秉扑,服務(wù)器的響應(yīng)時(shí)間跟第一次訪問時(shí)是相同的,因?yàn)榉?wù)器并不記得曾經(jīng)訪問過的這個(gè)客戶舟陆,也不記得上次訪問的內(nèi)容。Http協(xié)議的無狀態(tài)特性簡(jiǎn)化了服務(wù)器的設(shè)計(jì)秦躯,使得服務(wù)器更容易支持大量并發(fā)的Http請(qǐng)求。

因?yàn)镠ttp協(xié)議完成一次數(shù)據(jù)交互后就斷開連接倡缠,所以服務(wù)器無法通過Http請(qǐng)求主動(dòng)告知客戶端某條新消息的到達(dá)茎活。而Tcp/Udp協(xié)議可以做到了這一點(diǎn)。Tcp/Udp協(xié)議是傳輸層協(xié)議载荔,可以進(jìn)行全雙工通信。這樣就使得服務(wù)器端可以在有新消息到達(dá)的時(shí)候主動(dòng)通知客戶端懒熙,讓客戶端拉取最新消息。

Tcp和Udp兩個(gè)協(xié)議的區(qū)別在于,Tcp是面向連接的泌豆,在傳輸數(shù)據(jù)區(qū)必須先建立連接,數(shù)據(jù)傳輸結(jié)束后要釋放連接蔬浙。能保證數(shù)據(jù)的可靠傳輸贞远,建立連接的過程較復(fù)雜,占用較多的處理機(jī)資源蓝仲。而Udp是無連接的,不需要建立連接袱结,不提供可靠交付,但是處理機(jī)開銷小溢吻,效率高果元∠耍基于這些特點(diǎn),Udp協(xié)議通常用于對(duì)傳輸數(shù)據(jù)容錯(cuò)性高阅畴、時(shí)效性強(qiáng)的場(chǎng)景题翰,如語音電話數(shù)據(jù)的傳輸;而Tcp協(xié)議通常用于對(duì)傳輸數(shù)據(jù)完整性和順序要求高的場(chǎng)景豹障,如文件傳輸。

在我們的聊天程序中昵仅,我們將利用Tcp協(xié)議進(jìn)行聊天內(nèi)容的傳輸累魔。

使用Tcp協(xié)議的Socket編程

使用Tcp協(xié)議的Socket編程主要用到兩個(gè)類,Socket和ServerSocket垦写。ServerSocket本身也是一個(gè)Socket,只是它同時(shí)包含了一些額外的服務(wù)器終端的功能梯投,比如監(jiān)聽端口,等待客戶端Socket前來建立連接等尔艇。通過accept方法一旦和客戶端建立起連接么鹤,就會(huì)返回一個(gè)普通的Socket和客戶端Socket進(jìn)行對(duì)等的通信。

本文的Demo實(shí)現(xiàn)的功能很簡(jiǎn)單蒸甜,用戶輸入消息,發(fā)送給服務(wù)器昧辽,服務(wù)器原封不動(dòng)返回給客戶端。原理是:?jiǎn)?dòng)一個(gè)服務(wù)器線程和兩個(gè)客戶端線程搅荞。服務(wù)器線程監(jiān)聽2000端口,等待客戶端進(jìn)程的連接咕痛。客戶端線程包括一個(gè)發(fā)送線程和一個(gè)接收線程塞栅,發(fā)送線程連接服務(wù)器的2000端口腔丧,將用戶輸入的消息發(fā)送給服務(wù)器線程,接收線程監(jiān)聽2001端口愉粤,等待服務(wù)器返回的數(shù)據(jù)。服務(wù)器通過2000端口收到客戶端發(fā)來的數(shù)據(jù)后衣厘,將數(shù)據(jù)原封不動(dòng)發(fā)送到客戶端的2001端口,完成通信错邦。

實(shí)現(xiàn)原理圖

由于通過Socket進(jìn)行收發(fā)消息是阻塞式的型宙,也就是說在等待接收消息的時(shí)候不能同時(shí)發(fā)送消息,所以客戶端啟用了兩個(gè)線程分別進(jìn)行收消息和發(fā)消息的操作妆兑。代碼結(jié)構(gòu)如下圖:

代碼結(jié)構(gòu)

服務(wù)器收發(fā)線程關(guān)鍵代碼:

@Override    
public void run() {        
    try {            
        // 服務(wù)器線程通過2000端口監(jiān)聽客戶端發(fā)來的消息            
        serverSocket = new ServerSocket(2000);            
        serverReceiveSocket = serverSocket.accept();            
        // 和客戶端2001端口進(jìn)行通信,向客戶端發(fā)送消息            
        serverSendSocket = new Socket("127.0.0.1", 2001);            
        dataInputStream = new DataInputStream(serverReceiveSocket.getInputStream());            
        printStream = new PrintStream(serverSendSocket.getOutputStream());            
        bufferedReader = new BufferedReader(new InputStreamReader(dataInputStream, "UTF-8"));            
        while (!Thread.currentThread().isInterrupted()) {                
            try {      
                String line = bufferedReader.readLine();                    
                printStream.println(line);                
            } catch (IOException e) {                    
                e.printStackTrace();                
            }            
        }        
    } catch (IOException e) {            
        e.printStackTrace();            
        return;        
    }        
    cleanUpJob();    
}

客戶端發(fā)送線程關(guān)鍵代碼:

@Override    
public void run() {        
    try {            
        // 客戶端連接服務(wù)器端2000端口,通過該端口向服務(wù)器發(fā)消息            
        clientSocket = new Socket("127.0.0.1", 2000);            
        outputStream=new PrintStream(clientSocket.getOutputStream());            
        while (!Thread.currentThread().isInterrupted()) {                
            if (toSendList != null && toSendList.size() > 0) {                    
                outputStream.println(toSendList.get(0));                    
                toSendList.clear();                
            }            
        }        
    } catch (UnknownHostException e) {            
        e.printStackTrace();        
    } catch (IOException e) {            
        e.printStackTrace();        
    }    
    
    cleanUpJob();    
}

客戶端收消息關(guān)鍵代碼:

@Override    
public void run() {       
     try {            
        // 客戶端監(jiān)聽2001端口,等待服務(wù)器的連接,接收服務(wù)器發(fā)來的消息            
        server = new ServerSocket(2001);            
        socket = server.accept();            
        inputStream = new DataInputStream(socket.getInputStream());                  
        bufferedReader = new BufferedReader(new InputStreamReader(inputStream, "UTF-8"));            
        while (!Thread.currentThread().isInterrupted()) {
            String line = bufferedReader.readLine();                
            if (!TextUtils.isEmpty(line)) {                    
                strList.add(line);                    
                handler.sendEmptyMessage(-1);                
            }           
         }        
    } catch (IOException e) {            
    e.printStackTrace();            
    return;      
    }        
    cleanUpJob();   
 }

參考:
http://stackoverflow.com/questions/2004531/what-is-the-difference-between-socket-and-serversocket
http://stackoverflow.com/questions/5764065/the-difference-between-inputstream-and-inputstreamreader-when-reading-multi-byte

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末潭千,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子刨晴,更是在濱河造成了極大的恐慌,老刑警劉巖狈癞,帶你破解...
    沈念sama閱讀 216,692評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蝶桶,死亡現(xiàn)場(chǎng)離奇詭異慨绳,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)厌小,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門战秋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人脂信,你說我怎么就攤上這事≌粒” “怎么了?”我有些...
    開封第一講書人閱讀 162,995評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵秉撇,是天一觀的道長(zhǎng)秋泄。 經(jīng)常有香客問我,道長(zhǎng)恒序,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,223評(píng)論 1 292
  • 正文 為了忘掉前任滋饲,我火速辦了婚禮,結(jié)果婚禮上屠缭,老公的妹妹穿的比我還像新娘崭参。我一直安慰自己,他們只是感情好何暮,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,245評(píng)論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著跨新,像睡著了一般。 火紅的嫁衣襯著肌膚如雪域帐。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,208評(píng)論 1 299
  • 那天肖揣,我揣著相機(jī)與錄音,去河邊找鬼阳欲。 笑死,一個(gè)胖子當(dāng)著我的面吹牛球化,可吹牛的內(nèi)容都是我干的瓦糟。 我是一名探鬼主播筒愚,決...
    沈念sama閱讀 40,091評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼巢掺,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了劲蜻?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,929評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤轧苫,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后含懊,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,346評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡岔乔,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,570評(píng)論 2 333
  • 正文 我和宋清朗相戀三年滚躯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片剿配。...
    茶點(diǎn)故事閱讀 39,739評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡阅束,死狀恐怖茄唐,靈堂內(nèi)的尸體忽然破棺而出息裸,到底是詐尸還是另有隱情,我是刑警寧澤年扩,帶...
    沈念sama閱讀 35,437評(píng)論 5 344
  • 正文 年R本政府宣布访圃,位于F島的核電站厨幻,受9級(jí)特大地震影響腿时,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜批糟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,037評(píng)論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望盛末。 院中可真熱鬧否淤,春花似錦悄但、人聲如沸石抡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,677評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至挖藏,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間膜眠,已是汗流浹背溜嗜。 一陣腳步聲響...
    開封第一講書人閱讀 32,833評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留炸宵,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,760評(píng)論 2 369
  • 正文 我出身青樓捎琐,卻偏偏與公主長(zhǎng)得像会涎,于是被迫代替她去往敵國(guó)和親瑞凑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,647評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容