Java简单实现UDP和TCP

TCP实现

TCP协议需要在双方之间建立连接,通过输入输出流来进行数据的交换,建立需要通过三次握手,断开需要四次挥手,保证了数据的完整性,但传输效率也会相应的降低。

简单的TCP实现

//服务端
public class TcpServer {
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(8886);  // 建立服务端,ip为本机ip,端口为8886
        Socket accept = serverSocket.accept();  // 监听客户端的连接,一旦有客户端连接,则会返回客户端对应的accept
        
        InputStream in = accept.getInputStream();   //获取到客户端的输出流
        byte b[] = new byte[1024];
        int len = in.read(b);
        System.out.println("接受到客户端数据,返回数据"+new String(b,0,len));  
        
        OutputStream out = accept.getOutputStream();  // 给客户端发送消息
        out.write("服务端已经接受".getBytes());
        
        serverSocket.close();
    }
}

// 客户端
public class TcpClient {
    public static void main(String[] args) throws IOException {
        Socket socket = new Socket("127.0.0.1", 8886);   // 通过Socket来建立和服务端的连接
        OutputStream out = socket.getOutputStream();   // 获取输出流(客户端输出流即向服务端输出信息)
        out.write("hello tcp Server".getBytes());  // 输出信息
        
        InputStream in = socket.getInputStream();   // 接受服务端的消息
        byte b[] = new byte[1024];
        int len = in.read(b);
        System.out.println("接受到服务器消息 : "+new String(b,0,len));   // 输出
        out.write("返回的的数据已经收到 ".getBytes());  // 向服务器返回消息
        socket.close();
    }
}

改进服务端,启用多线程来接受客户端的数据

// server
 public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(8886); // 建立服务端,ip为本机ip,端口为8886
        int i=4;
        while(i>2){
            Socket accept = serverSocket.accept(); // 监听客户端的连接,一旦有客户端连接,则会返回客户端对应的accept
            
            ServerThread st = new ServerThread(accept);   // 启动线程
            Thread th = new Thread(st);
            th.start();
        }
        serverSocket.close();
    }

// thread

public class ServerThread implements Runnable {
    private Socket accept;
    public ServerThread(Socket s) {
        this.accept = s;
    }
    public void run(){
        InputStream in;
        try {
            in = accept.getInputStream();
            byte b[] = new byte[1024];
            int len = in.read(b);
            System.out.println("接受到客户端数据,返回数据" + new String(b, 0, len));
            OutputStream out = accept.getOutputStream(); // 给客户端发送消息
            out.write("服务端已经接受".getBytes());
        } catch (IOException e) {
            e.printStackTrace();
        } 
    }
}

传递图片

// 服务端

public class Server {
    public static void main(String[] args) throws IOException {

        ServerSocket serverSocket = new ServerSocket(5555);
        Socket ss = serverSocket.accept();
        
        BufferedInputStream br = new BufferedInputStream(ss.getInputStream());

        BufferedOutputStream bw = new BufferedOutputStream(new FileOutputStream("copy.jpg"));  // 写出文件流
       
        byte b[] = new byte[1024];
        int len = 0;
 
        while ((len = br.read(b)) != -1) {   // 写出文件
            bw.write(b, 0, len);
            bw.flush();  // 别忘了刷新,要不然最后一块缓冲区字符串会缺失
        }

        BufferedWriter bw2 = new BufferedWriter(new OutputStreamWriter(ss.getOutputStream())); // 输出
        bw2.write("图片传输成功");
        bw2.flush();
        
        bw.close();
        ss.close();
    }

}


// 客户端

public class Client {
    public static void main(String[] args) throws UnknownHostException, IOException {
        Socket socket = new Socket("127.0.0.1", 5555);
        BufferedInputStream in = new BufferedInputStream(new FileInputStream("c.jpg"));

        BufferedOutputStream out = new BufferedOutputStream(socket.getOutputStream());

        byte b[] = new byte[1024];
        int len = 0;

        while ((len = in.read(b)) != -1) {
            out.write(b, 0, len);
            out.flush();    // 刷新缓冲区 要不然最后一块缓冲区字符串会缺失
        }
        
        socket.shutdownOutput();   // 关闭流以后Server段才会接收道结束字符结束接受

        BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String line;
        while ((line = br.readLine()) != null) {
            System.out.println(line);
        }
        
        in.close();
        socket.close();

    }

}

UDP实现

UDP是将数据打成数据包向对方发送,只关系是否发送成功,而不关心是否接收成功,传输速度快,但是可靠性低。

udp代码实现

// 发送端
public class SendDemo {
    public static void main(String[] args) throws IOException {
        DatagramSocket ds = new DatagramSocket(); // 此类表示用来发送和接收数据报包的套接字。
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in)); // 键盘输入
        String line = null;
        while ((line = br.readLine()) != null) {
            byte[] bytes = line.getBytes();
            DatagramPacket dp = new DatagramPacket(bytes, bytes.length, InetAddress.getByName("127.0.0.1"), 10005); // 数据包对象
            ds.send(dp);
            if ("886".equals(line)) { // 当输入886时结束发送
                break;
            }
        }
        ds.close();
    }

}


// 接收端

public class ReciveDemo {
    public static void main(String[] args) throws IOException {
        DatagramSocket ds = new DatagramSocket(10005); // 建立服务端
        byte bytes[] = new byte[1024];
        DatagramPacket dp = new DatagramPacket(bytes, bytes.length); // 建立数据包对象
        while (true) {
            ds.receive(dp); // 接受数据包
            byte[] data = dp.getData(); // 获取数据
            String str = new String(data, 0, dp.getLength());
            if ("886".equals(str)) {
                break;
            }
            System.out.println(str);
        }
        ds.close();
    }
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏分布式系统进阶

ReplicaManager源码解析2-LeaderAndIsr 请求响应

其中最主要的操作调用ReplicaManager.becomeLeaderOrFollower来初始化Partition

501
来自专栏分布式系统进阶

KafkaController分析4-Partition选主

491
来自专栏尚国

S2-057远程代码执行漏洞复现过程

https://github.com/vulhub/vulhub/tree/master/struts2/s2-048

2173
来自专栏Jerry的SAP技术分享

Opportunity的chance of success的赋值逻辑

该字段的值和另外两个字段Sales Stage和Status都相关。

2128
来自专栏Kubernetes

原 荐 Kubernetes Resourc

更多关于kubernetes的深入文章,请看我csdn或者oschina的博客主页。 ResoureQuota介绍 关于ResoureQuota和Resourc...

4859
来自专栏比原链

Derek解读Bytom源码-P2P网络 地址簿

Gitee地址:https://gitee.com/BytomBlockchain/bytom

651
来自专栏积累沉淀

CXF 框架拦截器

CXF的拦截器 •为什么设计拦截器? 1.为了在webservice请求过程中,能动态操作请求和响应数据, CXF设计了拦截器. •拦截器分类: 1...

1896
来自专栏Java帮帮-微信公众号-技术文章全总结

Web-第二十一天 Web商城实战一【悟空教程】

public class BaseServlet extends HttpServlet {

1434
来自专栏技术博文

excel导入与导出

基本上导出的文件分为两种: 1:类Excel格式,这个其实不是传统意义上的Excel文件,只是因为Excel的兼容能力强,能够正确打开而已。修改这种文件后再保存...

2636
来自专栏人工智能LeadAI

通过UDP广播实现Android局域网Peer Discovering

本文将介绍以下知识点: 1、TCP与UDP的区别; 2、单播、多播、广播; 3、Java中实现UDP的重要的类; 4、Peer Discovering方案 01...

3974

扫码关注云+社区