Socket编程--使用Tcp实现简单的聊天程序

Socket

在Android聊天程序的实现中,通常是通过http请求拉取最新聊天信息,由于http请求是无状态(Stateless)的,无法随时获知新消息的到达,通常采用推送的方式告知客户端有新的消息。除了这种http+推送的方式外,也可以通过Socket编程实现聊天程序。本文将对网络层次进行简单的讲解,并结合一个简单的demo讲解Socket编程的实现。

本文Github Demo地址

计算机网络基础知识

计算机网络用于连接不同的计算机,并实现计算机间的数据传输。整个数据传输是个很复杂的过程,需要很多步骤进行,这些步骤被划分成了不同的层级,所以计算机网络有标准的层级结构,从底层到高层依次是物理层,数据链路层,网络层,传输层,应用层,如下图:

计算机网络五层协议

分层的优点在于各层级之间是独立的,灵活性好,在结构上可分隔开,易于实现和维护,同时能促进每层的标准化工作。每个层次都有运行在该层的一系列网络协议,如网络层的IP协议,传输层的TCP/UDP协议,应用层的Http/SMTP/DNS协议,如下图:

每层包含的协议

其中Http协议运行在应用层,典型的应用是网络浏览器。Http协议是无状态的,同一个客户多次访问同一个服务器上的页面时,服务器的响应时间跟第一次访问时是相同的,因为服务器并不记得曾经访问过的这个客户,也不记得上次访问的内容。Http协议的无状态特性简化了服务器的设计,使得服务器更容易支持大量并发的Http请求。

因为Http协议完成一次数据交互后就断开连接,所以服务器无法通过Http请求主动告知客户端某条新消息的到达。而Tcp/Udp协议可以做到了这一点。Tcp/Udp协议是传输层协议,可以进行全双工通信。这样就使得服务器端可以在有新消息到达的时候主动通知客户端,让客户端拉取最新消息。

Tcp和Udp两个协议的区别在于,Tcp是面向连接的,在传输数据区必须先建立连接,数据传输结束后要释放连接。能保证数据的可靠传输,建立连接的过程较复杂,占用较多的处理机资源。而Udp是无连接的,不需要建立连接,不提供可靠交付,但是处理机开销小,效率高。基于这些特点,Udp协议通常用于对传输数据容错性高、时效性强的场景,如语音电话数据的传输;而Tcp协议通常用于对传输数据完整性和顺序要求高的场景,如文件传输。

在我们的聊天程序中,我们将利用Tcp协议进行聊天内容的传输。

使用Tcp协议的Socket编程

使用Tcp协议的Socket编程主要用到两个类,Socket和ServerSocket。ServerSocket本身也是一个Socket,只是它同时包含了一些额外的服务器终端的功能,比如监听端口,等待客户端Socket前来建立连接等。通过accept方法一旦和客户端建立起连接,就会返回一个普通的Socket和客户端Socket进行对等的通信。

本文的Demo实现的功能很简单,用户输入消息,发送给服务器,服务器原封不动返回给客户端。原理是:启动一个服务器线程和两个客户端线程。服务器线程监听2000端口,等待客户端进程的连接。客户端线程包括一个发送线程和一个接收线程,发送线程连接服务器的2000端口,将用户输入的消息发送给服务器线程,接收线程监听2001端口,等待服务器返回的数据。服务器通过2000端口收到客户端发来的数据后,将数据原封不动发送到客户端的2001端口,完成通信。

实现原理图

由于通过Socket进行收发消息是阻塞式的,也就是说在等待接收消息的时候不能同时发送消息,所以客户端启用了两个线程分别进行收消息和发消息的操作。代码结构如下图:

代码结构

服务器收发线程关键代码:

@Override    
public void run() {        
    try {            
        // 服务器线程通过2000端口监听客户端发来的消息            
        serverSocket = new ServerSocket(2000);            
        serverReceiveSocket = serverSocket.accept();            
        // 和客户端2001端口进行通信,向客户端发送消息            
        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();    
}

客户端发送线程关键代码:

@Override    
public void run() {        
    try {            
        // 客户端连接服务器端2000端口,通过该端口向服务器发消息            
        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();    
}

客户端收消息关键代码:

@Override    
public void run() {       
     try {            
        // 客户端监听2001端口,等待服务器的连接,接收服务器发来的消息            
        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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容