【J2SE快速进阶】——Socket编程入门(TCP/UDP)

简介: 作品展时,我们的作品 “超级飞聊” 的主打功能就是聊天,包括局域网聊天、外网聊天等,虽然那时是用VB实现的(winsock控件),不过每种编程中的思想还是大同小异的,所以学习Java中的Socket编程时,倍感亲切啊。

      概念了解

      Socket又称“套接字”,用来向网络中的主机发出请求或者应答网络中发出的请求。


      文章开头先来简单了解一下TCP与UDP这两个协议:


      TCP


      TCP(Transmission Control Protocol传输控制协议)是一种面向连接的、可靠的、基于字节流的通信协议,位于传输层。这三个特点中,面向连接就如同打电话,双方的电话必须保持连接状态才能通话;可靠就如同QQ上的视频,一方发送视频请求,另一方必须同意后才能建立视频连接,也可以说安全性好;基于字节流,继续看下文就行。


      TCP最重要的思想就是大名鼎鼎的“三次握手”:


86.png

86.png


      客户端向服务端发送请求报文,服务端收到后向客户端回复,客户端确认收到服务端的回复。三次握手完成后,客户端就与服务端建立了可靠的链接。


     

       UDP


       UDP(User Datagram Protocol用户数据报协议),提供简单但不可靠的信息传送服务。很像生活中的写信或发邮件,不需要征得对方的同意,不需要与对方建立连接,就可以将数据发送出去,但是不能保证发送出去的数据能够确保无误地到达目的地。



      我们常用的ping命令就是基于UDP协议,telnet命令就是基于TCP协议。


      记得在VB中使用winsock,先往窗体上拖两个winsock控件(相当于实例化),然后给它们设置IP、端口、传输协议,就可以通信了。只不过在Java中,基于TCP协议的通信需要用ServerSocket和Socket来完成,基于UDP协议的通信需要用 DatagramSocket和DatagramPacket来完成。

      下面从TCP和UDP两个方面来举例说明Java中如何进行网络通信。    


       基于TCP协议的通信    

       服务端代码:      

import java.io.*;
import java.net.*;
public class Server { 
  public static void main(String[] args) {
    InputStream is=null; 
    OutputStream os=null;
    try{
      ServerSocket ss=new ServerSocket(5566);         //创建服务器套接字并绑定到5566端口
      Socket s=ss.accept();
      is=s.getInputStream();
      os=s.getOutputStream();
      DataInputStream dis=new DataInputStream(is);
      DataOutputStream dos=new DataOutputStream(os);
      String message=null;
      if((message=dis.readUTF())!=null){              //接收并输出此连接的客户端发来的信息
        System.out.println("Client:"+message);
      }
      dos.writeUTF("Hello, Client!");                //向此连接的客户端发送信息
      dis.close();
      dos.close();
      s.close();
    }catch(ConnectException ex){
      ex.printStackTrace();
    }catch(IOException ex){
      ex.printStackTrace();
    }
  }
}


       客户端代码:

import java.io.*;
import java.net.*;
public class Client {
  public static void main(String[] args) {
    InputStream is=null;
    OutputStream os=null;
    try{
      Socket s=new Socket("127.0.0.1",5566);         //创建一个套接字并将其连接到127.0.0.1地址(本机)的5566端口
      is=s.getInputStream();
      os=s.getOutputStream();
      DataInputStream dis=new DataInputStream(is);
      DataOutputStream dos=new DataOutputStream(os);
      dos.writeUTF("Hello, Server!");               //向服务端发送信息
      String message=null;
      if((message=dis.readUTF())!=null){
        System.out.println("Server:"+message);    //接收并输出服务端发来的信息
      }
      dis.close();
      dos.close();
      s.close();      
    }catch(ConnectException ex){
      ex.printStackTrace();
    }catch(IOException ex){
      ex.printStackTrace();
    }
  }
}


    执行结果:


         客户端: 87.png      

               服务端:88.png


       整个连接和交互的过程如下图:

89.png


       执行时,先启动服务端,服务端创建服务器套接字ServerSocket并绑定到指定的端口,当执行到Socket s=ss.accept()时,会产生“阻塞”(即让程序暂时停留在此处),但客户端启动,创建套接字Socket并向指定地址的指定端口发送请求时,ServerSocket接受服务端的请求并返回客户端的socket实例,与之建立连接。socket原意为插座,还真的挺形象的,整个过程就如同把客户端的插头插到服务端的插座上,就建立起了通道。通信完毕后,双方都关闭连接。


       例子中,Socket的getInputStream()方法可以获得网络连接输入,返回此套接字的字节输入流对象;getOutputStream()方法返回此套接字的字节输出流对象,用于向连接对象写入数据。这里体现了上文中TCP的第三个特点“基于字节流”。


       基于UDP协议的通信


       服务端代码:    

import java.net.*;
class UDPServer{
    public static void main(String[] args)throws Exception{
      //接收信息
        DatagramSocket  server = new DatagramSocket(5050);
        byte[] recvBuf = new byte[1000];
        DatagramPacket recvPacket = new DatagramPacket(recvBuf , recvBuf.length);
        server.receive(recvPacket);
        String recvStr = new String(recvPacket.getData() , 0 , recvPacket.getLength());
        System.out.println("What server has received is:" + recvStr);
        //发送信息
        int port = recvPacket.getPort();
        InetAddress addr = recvPacket.getAddress();
        String sendStr = "Hello ! I'm Server";
        byte[] sendBuf = new String("Hello Client! I'm Server!").getBytes();    
        DatagramPacket sendPacket 
            = new DatagramPacket(sendBuf , sendBuf.length , addr , port );
        server.send(sendPacket);
        server.close();
    }
}


      客户端代码:


import java.net.*;
class UDPClient{
    public static void main(String[] args)throws Exception{
      //发送信息
        DatagramSocket client = new DatagramSocket();       
        byte[] sendBuf=new String("Hello Server! I'm Client").getBytes();       
        InetAddress addr = InetAddress.getByName("127.0.0.1");
        int port = 5050;
        DatagramPacket sendPacket 
            = new DatagramPacket(sendBuf ,sendBuf.length , InetAddress.getByName("127.0.0.1") , port);
        client.send(sendPacket);
        //接收信息
        byte[] recvBuf = new byte[100];
        DatagramPacket recvPacket = new DatagramPacket(recvBuf , recvBuf.length);
        client.receive(recvPacket);
        String recvStr = new String(recvPacket.getData() , 0 ,recvPacket.getLength());
        System.out.println("What client has received is:" + recvStr);
        client.close();
    }
}


       整个流程如下图:

90.png

       服务端启动后,准备一个包含字节数组buf的数据报包DatagramPacket用于接收客户端发来的数据报,客户端启动后,将目的地址、端口、发送内容等信息封装在DatagramPacket中,通过DatagramSocket的send()方法发往目的地(服务端),服务端将接收到的数据报放在DatagramPackage中。服务端向客户端发送数据报的过程亦然。




      文章前面说了TCP三次握手,客户端发送请求、服务端接受请求并返回确认信息,客户端收到确认信息后再发送确认收到确认信息,简单的三个步骤,过程却非常复杂(而且TCP断开连接时还需要四次挥手)。因为UDP协议不需要连接的过程,所以它的效率高,速度快,同时也因如此,UDP的可靠性不如TCP高。


相关文章
|
28天前
|
网络协议 算法 网络性能优化
|
16天前
|
网络协议 SEO
TCP连接管理与UDP协议IP协议与ethernet协议
TCP、UDP、IP和Ethernet协议是网络通信的基石,各自负责不同的功能和层次。TCP通过三次握手和四次挥手实现可靠的连接管理,适用于需要数据完整性的场景;UDP提供不可靠的传输服务,适用于低延迟要求的实时通信;IP协议负责数据包的寻址和路由,是网络层的重要协议;Ethernet协议定义了局域网的数据帧传输方式,广泛应用于局域网设备之间的通信。理解这些协议的工作原理和应用场景,有助于设计和维护高效可靠的网络系统。
27 4
|
22天前
|
缓存 负载均衡 网络协议
面试:TCP、UDP如何解决丢包问题
TCP、UDP如何解决丢包问题。TCP:基于数据块传输/数据分片、对失序数据包重新排序以及去重、流量控制(滑动窗口)、拥塞控制、自主重传ARQ;UDP:程序执行后马上开始监听、控制报文大小、每个分割块的长度小于MTU
|
1月前
|
网络协议 前端开发 物联网
TCP和UDP区别?
本文首发于微信公众号“前端徐徐”,详细介绍了TCP和UDP两种传输层协议的核心概念、连接性和握手过程、数据传输和可靠性、延迟和效率、应用场景及头部开销。TCP面向连接、可靠、有序,适用于网页浏览、文件传输等;UDP无连接、低延迟、高效,适用于实时音视频传输、在线游戏等。
49 1
TCP和UDP区别?
|
30天前
|
Web App开发 缓存 网络协议
不为人知的网络编程(十八):UDP比TCP高效?还真不一定!
熟悉网络编程的(尤其搞实时音视频聊天技术的)同学们都有个约定俗成的主观论调,一提起UDP和TCP,马上想到的是UDP没有TCP可靠,但UDP肯定比TCP高效。说到UDP比TCP高效,理由是什么呢?事实真是这样吗?跟着本文咱们一探究竟!
53 10
|
30天前
|
Java
[Java]Socket套接字(网络编程入门)
本文介绍了基于Java Socket实现的一对一和多对多聊天模式。一对一模式通过Server和Client类实现简单的消息收发;多对多模式则通过Server类维护客户端集合,并使用多线程实现实时消息广播。文章旨在帮助读者理解Socket的基本原理和应用。
23 1
|
1月前
|
网络协议 网络性能优化 C#
C# 一分钟浅谈:UDP 与 TCP 协议区别
【10月更文挑战第8天】在网络编程中,传输层协议的选择对应用程序的性能和可靠性至关重要。本文介绍了 TCP 和 UDP 两种常用协议的基础概念、区别及应用场景,并通过 C# 代码示例详细说明了如何处理常见的问题和易错点。TCP 适用于需要可靠传输和顺序保证的场景,而 UDP 适用于对延迟敏感且可以容忍一定数据丢失的实时应用。
38 1
|
1月前
|
网络协议 Linux 网络性能优化
Linux C/C++之TCP / UDP通信
这篇文章详细介绍了Linux下C/C++语言实现TCP和UDP通信的方法,包括网络基础、通信模型、编程示例以及TCP和UDP的优缺点比较。
38 0
Linux C/C++之TCP / UDP通信
|
2月前
|
存储 网络协议 算法
UDP 协议和 TCP 协议
本文介绍了UDP和TCP协议的基本结构与特性。UDP协议具有简单的报文结构,包括报头和载荷,报头由源端口、目的端口、报文长度和校验和组成。UDP使用CRC校验和来检测传输错误。相比之下,TCP协议提供更可靠的传输服务,其结构复杂,包含序列号、确认序号和标志位等字段。TCP通过确认应答和超时重传来保证数据传输的可靠性,并采用三次握手建立连接,四次挥手断开连接,确保通信的稳定性和完整性。
93 1
UDP 协议和 TCP 协议
|
3月前
|
消息中间件 网络协议 算法
UDP 和 TCP 哪个更好?
【8月更文挑战第23天】
214 0
下一篇
无影云桌面