zoukankan      html  css  js  c++  java
  • TCP/UDP套接字 java socket编程实例

    网络协议七层结构:

    什么是Socket?

      socket(套接字)是两个程序之间通过双向信道进行数据交换的端,可以理解为接口。使用socket编程也称为网络编程,socket只是接口并不是网络通信协议。

    HTTP协议和Socket的区别

      http协议是应用层,其模式是请求-应答,客户端发送请求,服务器端进行响应。传输的数据是原始格式的数据,eg :json、xml、text等数据格式。

      socket不是协议是接口,socket提供TCP/UDP socket 的实例,供java 或者其他语言操作数据的传输,socket是对传输层(TCP/UPD协议)的封装。

    Socket通信分为两种

      TCP Socket :使用流传输,提供inputStream 和 outputStream 方法对数据进行流操作。要理解TCP套接字首先要对TCP协议有所理解。

        1)TCP协议是传输层的协议,他的下一层是IP协议(网络层),IP协议在网络数据传输是通过ip寻址,将源地址和目的地址进行连接。TCP协议是在IP协议上多加一层端口寻址,光只通过IP寻址只能定位到主机,tcp通过端口找到对应的应用程序。

        2)TCP 建立连接需要三次握手,将源应用程序和目的应用程序之间搭建一个连接,所以源应用和目的应用程序之间必须是one by one。IP 协议只管数据的传输,不保证数据是否丢失,重复传,顺序是否正确,TCP会对这些问题做一些补偿机制,丢失数据重传,用队列保证数据的顺序。

        3) TCP 缺点:因为每个客户端和服务器端传输数据都要建立连接,三次握手是不传输数据并且有耗时,当有大量短连接的时候并且对数据的正确性要求不高的时候,将会占用带宽。

      UDP Socket:使用数据报文进行传输,创建UDP socket 发送和接收数据报文。

        1)UDP协议同TCP协议一样都是应用层协议,也是通过端口寻址,找到对应的应用程序。

        2)UDP传输数据报文不需要和目的应用程序建立连接,他在数据报文中指定目的主机和目的端口号,发送出的数据自动寻址到对应的主机和端口号。因为不用和目的主机建立连接,所以一个源应用程序可以以广播的形式将数据报文传输给多主机。因为不用建立连接,耗时和带宽占用量都比TCP协议更优秀

        3)UDP缺点:数据有可能丢失,丢失的数据不会重传

    java socket 实例

      TCP Socket client

    package socket.transmission.tcp;
    
    
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.OutputStream;
    import java.net.Socket;
    
    //TCP 套接字 客户端负责发送请求
    public class TcpClient {
        private static  final  int BUF_SIZE=32;
        /**
         * TCP客户端发送一个请求要有三个步骤:
         *  1.创建一个socket的实例,创建一个指向server主机ip和端口号的TCP连接
         *  2.通过套接字的输入和输出流进行通信
         *  3.使用socket close关闭
         */
        public static void main(String[] args){
          String ip="192.168.197.1";
            int port=8080;
            try {
                // 创建一个socket实例
                Socket socket=new Socket(ip,port); // 1 设置TCP SOCKET,初始化目的主机ip和端口号,建立和目的主机的连接,若目的主机没有开启服务,则会弹出server refused
                System.out.println("创建一个socket连接");
                InputStream inputStream=socket.getInputStream();  // 2 获取回馈服务器的输入流
                OutputStream outputStream=socket.getOutputStream(); // 3 将要传输的数据数据写入到输出流中,传输给目的主机
                //向socket中写入数据
                outputStream.write("this is a word".getBytes());  // 4 传输数据到目的主机
                int totalByrecive=0;  //到目前为止接收到的数据
                byte[] readBuff=new byte[BUF_SIZE];
                int lastReadByte;  //最后接收的字节
                System.out.println("从服务器中接收的数据:");
                int receiveMsgSize;
                while ((receiveMsgSize=inputStream.read(readBuff))!=-1){   // 5.从回馈服务器中获取数据,
                    System.out.println(new String(readBuff));
                }
                 socket.close();  //关闭
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    
    }

    tcp sokect server

    package socket.transmission.tcp;
    
    //TCP 服务器端进行接收请求
    
    import sun.java2d.pipe.OutlineTextRenderer;
    
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.OutputStream;
    import java.net.ServerSocket;
    import java.net.Socket;
    import java.net.SocketAddress;
    
    /**
     * TCP服务器对客户端发送的请求会进行以下处理
     *  1.创建serverSocket实例并且指定本机端口,功能:监听指定端口发送过来的连接
     *  2.重复执行:
     *      1).调用的serverSocket 的accept() 监听客户端发送过来的请求,并创建socket
     *      2).使用socket的inputStream 和  outputStream 进行通讯
     *      3).通信完使用socket.close() 方法将连接关闭
     */
    public class TcpServer {
    
        private static  final  int BUF_SIZE=32;
    
        public static void main(String[] args){
            int port=8080;
            Socket socket = null;
            InputStream inputStream = null;
            OutputStream outputStream = null;
            try {
                ServerSocket serverSocket=new ServerSocket(port);//创建一个socket实例用于监听客户端发送的连接,指定本服务器的端口号
                System.out.println("创建serverSocket 实例");
                int reviceMsgSize;   // 接收msg的大小
                byte[] receiveBuf=new byte[BUF_SIZE];  //创建一个信息接收的缓冲区
                    System.out.println("开始处理接收的数据");
                    while (true) {
                        socket = serverSocket.accept();  //接收客户端的连接,每接收一个数据都会创建一个连接,当没有数据的接收的时候会阻塞
                        SocketAddress socketAddress = socket.getRemoteSocketAddress(); //
                        System.out.println("访问的地址:" + socketAddress);
                        inputStream = socket.getInputStream();
                        outputStream = socket.getOutputStream();
                        while ((reviceMsgSize = inputStream.read(receiveBuf)) != -1) {
                            System.out.println(new String(receiveBuf));
                            outputStream.write("aaaaa".getBytes(), 0, 4);
                        }
                        outputStream.flush();
                        socket.close();
                    }
            } catch (IOException e) {
                e.printStackTrace();
            }finally {
                    try {
                        if(socket!=null){
                            socket.close();
                        }
                        if(inputStream!=null){
                            inputStream.close();
                        }
                        if(outputStream!=null){
                            outputStream.close();
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
            }
        }
    
    
    }

     UDP Socket client

    package socket.transmission.udp;
    
    
    import java.io.IOException;
    import java.io.InterruptedIOException;
    import java.net.*;
    
    //UDP 套接字传输的是数据报文
    
    /**
     * UDP 客户端发送数据报文的步骤:
     *  1.创建UPD 套接字 DataGramSocket ,使用UDP协议通信是不需要和服务器端创建的连接的,UPD协议只是在IP协议上
     *    多加了一层端口寻址,设置超时
     *  2.创建发送的数据报文实例DataGramPacket,若是单播的则要指明目的端的ip地址和port,IP地址通过创建InetAddress 的实例
     *  3.创建接收数据报文实例DataGramPacket ,指定接收数据的缓冲区
     *  4.调用socket的send方法将数据报文发送出去
     *  5.循环接收数据报文,当数据报文丢失的时候,发起重试。否则设置响应标志位true,将数据打印
     */
    public class UdpClient {
        /**
         * 当客户端发送给server端信息,收到回馈信息的时候,通过read读取数据,当没有数据返回(数据丢失)
         * read 方法会发生阻塞,若没有设置超时重发,则程序会一直阻塞
         */
        private static final int TIME_OUT=2000;  //设置超时重发时间
    
        private static final int MAX_RENTRY=3;  // 设置最大重试次数
    
        public  static void main(String[] args) throws IOException {
            try {
                int serverPort=8080; // 指定
                byte[] sendMsg="this is a test".getBytes();
                DatagramSocket socket=new DatagramSocket(); //创建一个数据报文
                socket.setSoTimeout(TIME_OUT);  //设置read阻塞超时时间
                byte[] ipByte={10,1,1,100};
                /**
                 * "10.1.1.100".getbytes()的方式不能正确的创建server端,调用InetAddress.getByAddress() 方法将会做两个长度判断,IPV4 的入参长度要==4
                 *  IPV6 的长度要== 16 而通过10.1.1.100 getbytes的方式获取的长度是10 抛出违法的长度
                 */
                InetAddress inetAddress= InetAddress.getByAddress(ipByte);  //创建server主机的ip地址
                DatagramPacket sendPacket=new DatagramPacket(sendMsg,sendMsg.length,inetAddress,8080);  //发送的数据报文
                DatagramPacket receivePacket=new DatagramPacket(new byte[sendMsg.length],sendMsg.length);  //接收的数据报文
                int tryTimes=0;  //数据报文可能丢失,设置重试计数器
                Boolean receiveResponse=false;
                socket.send(sendPacket);  //将数据报文发送出去
                do{
                    try {
                        socket.receive(receivePacket);   //尝试去循环接收数据报文
                        if (!receivePacket.getAddress().equals(inetAddress)) { //检查回馈过来的数据报文
                            throw new IOException("未知的Server端数据报文");
                        }
                        receiveResponse=true;
                    }catch (InterruptedIOException e){ //数据报文中断异常
                        tryTimes++;
                        System.out.println("超时还有"+(MAX_RENTRY-tryTimes)+"次重试机会");
                    }
                }while(!receiveResponse&&(tryTimes<MAX_RENTRY));
                if(receiveResponse){
                    System.out.println("从服务器端获取的数据:"+new String(receivePacket.getData()));
                }else{
                    System.out.println("没有获取到数据");
                }
                socket.close(); //关闭套接字
            } catch (SocketException e) {
                e.printStackTrace();
            }
        }
    }

    UPD Soceket server

    package socket.transmission.udp;
    
    import java.io.IOException;
    import java.io.InterruptedIOException;
    import java.net.*;
    
    //UDP 套接字接收客户端的数据报文
    public class UdpServer {
        private static final int ECHO_MAX=255;  //设置缓冲区的长度
    
        public static void main(String[] args)  {
            try {
                DatagramSocket datagramSocket=new DatagramSocket(8080);
                DatagramPacket reveiveMsg=new DatagramPacket(new byte[ECHO_MAX],ECHO_MAX);
                while(true){
                        datagramSocket.receive(reveiveMsg);
                    System.out.println("从客户端接收的来数据:"+new String(reveiveMsg.getData()));
                    //在服务器端将发送的信息修改
                    byte[] newData="啦啦啦啦".getBytes();
                   // reveiveMsg=new DatagramPacket(newData,newData.length);
                    //将转化后的数据发送
                    datagramSocket.send(reveiveMsg);
                    /**
                     * 重置接收包的长度,因为接收数据的时候已经接收包的长度设置为接收信息的长度,当下次再接收数据的时候,
                     * 新数据的长度大于上一次数据的长度时,多出的数据将被截断,所以要重置接收包缓冲区的长度
                     */
                    reveiveMsg.setLength(ECHO_MAX);
                }
            } catch (UnknownHostException e) {
                e.printStackTrace();
            }catch (SocketException se){
                se.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
    
        }
    }

     上面的代码还有一些未补足的:要在finally 中将所有的流关闭。

  • 相关阅读:
    leetcode 268. Missing Number
    leetcode 189. Rotate Array
    leetcode 118. Pascal's Triangle 、119. Pascal's Triangle II 、120. Triangle
    HDU高精度总结(java大数类)
    一起talk C栗子吧(第八十七回:C语言实例--使用管道进行进程间通信概述)
    iOS敏捷开发之道,经常使用的宏定义总结
    360 网络攻防 hackgame 解题报告(通关)
    打通B/S与C/S !让HTML5 WebSocket与.NET Socket公用同一个服务端!
    数列求和
    指尖上的电商---(11)Windows平台部署SolrCloud
  • 原文地址:https://www.cnblogs.com/blogxiao/p/9304107.html
Copyright © 2011-2022 走看看