网络编程之 Socket 套接字(使用数据报套接字和流套接字分别实现一个小程序(附源码))

简介: 1. 什么是网络编程2. 网络编程中的基本概念1)发送端和接收端2)请求和响应3)客户端和服务端4)常见的客户端服务端模型3. Socket 套接字1)Socket 的分类2)Java 数据报套接字通信模型3)Java 流套接字通信模型4. UDP 数据报套接字编程1)DatagramSocket API2)DatagramPacket API3)示例5. TCP 流套接字编程1)ServerSocket API2)Socket API3)示例a. 短连接版本b. 长连接并发版本

1. 什么是网络编程

网络编程是指网络上的主机,通过不同的进程,以编程的方式实现 网络通信(或称为网络数据传输)


66.png


只要满足不同的进程就可以进行通信,所以即便是在同一个主机,只要不同的进程,基于网络传输数据,也属于网络编程


2. 网络编程中的基本概念

1)发送端和接收端

在一次网络传输中:


发送端: 数据的 发送方进程,称为发送端。发送端主机即网络通信中的源主机


接收端: 数据的 接收方进程,称为接收端。接收端主机即网络通信中的目的主机


收发端: 发送端和接收端两端,简称为收发端


注意: 发送端和接收端只是相对的,只是一次网络数据传输产生数据流向后的概念


67.png


2)请求和响应

一般来说,获取一个网络资源时,涉及到两次网络数据传输

  • 第一次:请求数据的发送
  • 第二次:响应数据的发送

就好比在饭店里点了一碗面

顾客先发起请求:来一碗面

饭店再做出响应:提供一碗面


68.png


3)客户端和服务端

服务端: 在常见的网络传输场景下,把 提供服务 的一方进程,称为服务端,可以对外提供服务

客户端:获取服务 的一方进程,称为客户端

对于服务来说,一般是提供:

  • 客户端获取服务资源

69.png


  • 客户端保存资源到服务端
  • 70.png

4)常见的客户端服务端模型

在常见的场景中,客户端是指给用户使用的程序,服务端是指提供用户服务的程序,具体流程如下:

  1. 客户端发送请求到服务端
  2. 服务端根据请求数据,执行相应的业务处理
  3. 服务端返回响应:发送业务处理结果
  4. 客户端根据响应数据,展示处理结果

71.png


3. Socket 套接字

Socket 套接字时由系统提供用于网络通信的技术,是基于 TCP/IP 协议的网络通信的基本操作单元。基于 Socket 套接字的网络程序开发就是网络编程


1)Socket 的分类

常用的 Socket 套接字有以下两类:


流套接字: 使用 TCP 协议


传输数据基于 IO 流,流式数据的特征就是在 IO 流没有关闭的情况下,是无边界的数据,可以多次发送数据,也可以分开多次接收


数据报套接字: 使用 UDP 协议


传输数据是一块一块的,每一块数据必须一次性发送,接受也必须一次性接受,不能分开发送或接收


2)Java 数据报套接字通信模型

UDP 协议具有无连接、面向数据报的特征,即每次传输都是没有建立连接,并且一次发送全部数据,一次接收全部数据


Java 中使用 UDP 协议通信时,主要基于 DatagramSocket 类来创建数据报套接字,并使用 DatagramPacket 作为发送或者接受的数据报。


具体流程如下:

72.png


3)Java 流套接字通信模型

TCP 协议具有有连接、面向字节流的特征,即传输数据之前要先建立连接,并且数据的传输是基于 IO 流的

具体流程如下:


73.png


4. UDP 数据报套接字编程

1)DatagramSocket API

DatagramSocket 是 UDP Socket,用于发送和接收 UDP 数据报

构造方法:

1683887483814.png



实例方法:

1683887509558.png



2)DatagramPacket API

DatagramPacket 是 UDP Socket 发送和接收的数据报

构造方法:

1683887529427.png

实例方法:

1683887540077.png


3)示例

使用 UDP Socket 套接字完成一个简单的翻译小程序

客户端向服务端发起翻译请求,服务端接收到请求后,对请求进行处理,再向客户端做出响应

服务端

服务端接收到请求之后,在我们人为规定的 map 中进行查询,再对客户端做出响应


public class Server {
    private final static HashMap<String, String> map = new HashMap<>();
    static {
        map.put("苹果", "apple");
        map.put("香蕉", "banana");
        map.put("梨", "pear");
        map.put("电脑", "computer");
        map.put("手机", "phone");
    }
    public static void main(String[] args) throws IOException {
        // 得到 socket 对象,并对其绑定端口号
        Log.println("服务器开启并且绑定了 8080 端口");
        DatagramSocket socket = new DatagramSocket(8080);
        // 循环接受和处理请求
        while (true) {
            // 准备接收数据的容器
            byte[] buf = new byte[1024];
            // 包装数据包
            DatagramPacket received = new DatagramPacket(buf, 0, 1024);
            // 接收请求
            Log.println("服务器准备接受请求");
            socket.receive(received);
            Log.println("服务器接受到了请求");
            // 处理请求
            InetAddress address = received.getAddress();    // 得到发起请求方的 IP
            Log.println("对方的地址:" + address);
            int port = received.getPort();      // 得到发起请求方的端口号
            Log.println("对方的端口:" + port);
            int length = received.getLength();      // 得到请求中真实有效的数据长度
            Log.println("真实的数据长度:" + length);
            byte[] realData = Arrays.copyOf(buf, length);   // 得到请求中真实有效的数据
            // 将请求数据转换成 String
            String request = new String(realData, 0, length, StandardCharsets.UTF_8);
            Log.println("请求中的数据:\r\n" + request);
            // 人为规定请求和相应的格式
            // 请求格式必须以 “我是请求:\r\n” 开始,以 “\r\n” 结束
            if (!request.startsWith("我是请求:\r\n")) {
                Log.println("请求格式出错");
                // 请求出错
                continue;
            }
            if (!request.endsWith("\r\n")) {
                Log.println("请求格式出错");
                // 请求出错
                continue;
            }
            // 得到请求中的英文单词
            String EnglishWord = request.substring("我是请求:\r\n".length(), request.length() - 2);
            Log.println("请求中的英文单词:" + EnglishWord);
            // 进行翻译
            String ChineseWord = map.getOrDefault(EnglishWord, "我不知道");
            Log.println("翻译后的中文:" + ChineseWord);
            // 将翻译后的结果进行包装
            String response = String.format(ChineseWord, "%s\r\n");
            byte[] bytes = response.getBytes(StandardCharsets.UTF_8);
            // 包装数据包
            DatagramPacket send = new DatagramPacket(bytes, 0, bytes.length, address, port);
            // 发起响应
            Log.println("准备发送响应");
            socket.send(send);
            Log.println("相应发送成功");
        }
    }
}

客户端

public class Client {
    public static void main(String[] args) throws IOException {
        Scanner sc = new Scanner(System.in);
        DatagramSocket socket = new DatagramSocket(9999);
        while (sc.hasNextLine()) {
            String str = sc.nextLine();
            String send = "我是请求:\r\n" + str + "\r\n";
            byte[] bytes = send.getBytes();
            DatagramPacket request = new DatagramPacket(bytes, 0, bytes.length, InetAddress.getLoopbackAddress(), 8080);
            socket.send(request);
            // 接收响应
            byte[] buf = new byte[1024];
            DatagramPacket datagramPacket = new DatagramPacket(buf, 0, 1024);
            socket.receive(datagramPacket);
            int n = datagramPacket.getLength();
            String response = new String(buf, 0, n, StandardCharsets.UTF_8);
            Log.println(response);
        }
    }
}

74.png


5. TCP 流套接字编程

1)ServerSocket API

用于创建 TCP 服务端 Socket 的 API

构造方法:


方法签名 方法说明
ServerSocket(int port) 创建一个服务端流套接字 Socket,并绑定到指定端口


实例方法:

方法签名 方法说明
Socket accept() 等待客户端发起连接,连接成功后返回一个 Socket 对象
void close() 关闭 Socket


2)Socket API

这里的 Socket 是客户端 Socket,或服务端中接收到客户端连接请求后,accept 方法返回的服务端 Socket

不管是客户端还是服务端 Socket,都是双方建立连接后,保存对端信息,以及用来与对方收发数据的

构造方法:


方法签名 方法说明
Socket(String host, int port) 创建一个客户端流套接字,并与对应 IP 的主机,对应端口的进程建立连接


实例方法:

方法签名 方法说明
InetAddress getInetAddress() 返回套接字所连接的 IP 地址
InputStream getInputStream() 返回套接字的输入流
OutputStream getOutputStream() 返回套接字的输出流


3)示例

使用 TCP Socket 套接字完成一个简单的翻译小程序

完成的效果和上文 UDP 的小程序效果一样

不过在 TCP 这里有长短连接的区分,所以有两个版本的代码,在长连接版本中加入了并发编程,使得同一个服务端可以被多个客户端所连接


a. 短连接版本

客户端:

public class 短连接版本Client {
    public static void main(String[] args) throws IOException {
        for (int i = 0; i < 3; i++) {
            Socket socket = new Socket("127.0.0.1", 8080);      // 拨号
            OutputStream os = socket.getOutputStream();
            OutputStreamWriter writer = new OutputStreamWriter(os, "UTF-8");
            PrintWriter printWriter = new PrintWriter(writer);
            printWriter.printf("我是请求\r\n%s\r\n", "苹果");
            printWriter.flush();
            InputStream is = socket.getInputStream();
            Scanner scanner = new Scanner(is, "UTF-8");
            String header = scanner.nextLine();
            String word = scanner.nextLine();
            System.out.println(word);
            socket.close();     // 挂电话
        }
    }
}

服务端:

public class 短连接版本Server {
    private final static HashMap<String, String> map = new HashMap<>();
    static {
        map.put("苹果", "apple");
        map.put("香蕉", "banana");
        map.put("梨", "pear");
        map.put("电脑", "computer");
        map.put("手机", "phone");
    }
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(8080);
        while(true) {
            Log.println("等待客户端建立连接");
            Socket socket = serverSocket.accept();
            Log.println("客户端连接成功");
            InetAddress address = socket.getInetAddress();
            Log.println("客户端 IP 地址:" + address);
            int port = socket.getPort();
            Log.println("客户端端口号:" + port);
            InputStream is = socket.getInputStream();
            Scanner sc = new Scanner(is, "UTF-8");
            String header = sc.nextLine();
            String EnglishWord = sc.nextLine();
            Log.println("请求中的英文单词为:" + EnglishWord);
            String ChineseWord = map.getOrDefault(EnglishWord, "我不知道");
            Log.println("翻译后的中文为:" + ChineseWord);
            Log.println("准备响应");
            OutputStream os = socket.getOutputStream();
            OutputStreamWriter writer = new OutputStreamWriter(os);
            PrintWriter printWriter = new PrintWriter(writer);
            printWriter.printf("OK!\r\n%s\t\n", ChineseWord);
            printWriter.flush();
            Log.println("响应成功");
            socket.close();
            Log.println("断开连接");
        }
    }
}

75.png

b. 长连接并发版本

在长连接版本中,必须服务端和客户端同时支持长连接才可以进行通信

如果不清楚长连接和短连接,可以去看看我上篇文章,里面有介绍长连接和短连接,HTTP 和 HTTPS,有介绍 HTTP 的长短连接,实质上就是 TCP 长短连接

客户端:


public class Client {
    public static void main(String[] args) throws IOException {
        Scanner sc = new Scanner(System.in);
        Socket socket = new Socket("127.0.0.1", 8080);
        OutputStream os = socket.getOutputStream();
        OutputStreamWriter writer = new OutputStreamWriter(os, StandardCharsets.UTF_8);
        PrintWriter printWriter = new PrintWriter(writer);
        InputStream is = socket.getInputStream();
        Scanner scanner = new Scanner(is, "UTF-8");
        while (sc.hasNextLine()) {
            String str = sc.nextLine();
            printWriter.printf("我是请求:\r\n%s\r\n", str);
            printWriter.flush();
            // 接收响应
            String header = scanner.nextLine();
            String response = scanner.nextLine();
            Log.println(header + "\r\n" + response);
        }
        socket.close();
    }
}

服务端:

public class Server {
    private final static HashMap<String, String> map = new HashMap<>();
    static {
        map.put("苹果", "apple");
        map.put("香蕉", "banana");
        map.put("梨", "pear");
        map.put("电脑", "computer");
        map.put("手机", "phone");
    }
    private final static class ServerTask implements Runnable {
        private final Socket socket;
        private ServerTask(Socket socket) {
            this.socket = socket;
        }
        @Override
        public void run() {
            InetAddress address = socket.getInetAddress();
            Log.println("客户端 IP 地址为:" + address);
            int port = socket.getPort();
            Log.println("客户端端口号为:" + port);
            try {
                InputStream is = socket.getInputStream();
                // 请求格式必须以 “我是请求:\r\n” 开始,以 “\r\n” 结束
                Scanner sc = new Scanner(is, "UTF-8");
                // 进行响应
                OutputStream os = socket.getOutputStream();
                OutputStreamWriter writer = new OutputStreamWriter(os, StandardCharsets.UTF_8);
                PrintWriter printWriter = new PrintWriter(writer);
                while (sc.hasNextLine()) {
                    String header = sc.nextLine();
                    String EnglishWord = sc.nextLine();
                    Log.println("请求中的词为:" + EnglishWord);
                    // 对请求进行处理
                    String ChineseWord = map.getOrDefault(EnglishWord, "我不知道");
                    Log.println("翻译后的词:" + ChineseWord);
                    Log.println("准备发起响应");
                    printWriter.printf("OK!\r\n%s\r\n", ChineseWord);
                    printWriter.flush();
                    Log.println("响应成功");
                }
                socket.close();
                Log.println("连接已关闭");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    public static void main(String[] args) throws IOException {
        // 使用多线程完成多用户同时在线的功能
        ExecutorService threadPool = Executors.newFixedThreadPool(3);
        ServerSocket serverSocket = new ServerSocket(8080);
        Log.println("绑定端口到 8080");
        while (true) {
            Log.println("等待客户端连接");
            Socket socket = serverSocket.accept();
            Log.println("客户端连接成功");
            ServerTask serverTask = new ServerTask(socket);
            threadPool.execute(serverTask);
        }
    }
}

76.png


目录
相关文章
|
6月前
|
小程序 前端开发
2025商业版拓展校园圈子论坛网络的创新解决方案:校园跑腿小程序系统架构
校园跑腿小程序系统是一款创新解决方案,旨在满足校园配送需求并拓展校友网络。跑腿员可接单配送,用户能实时跟踪订单并评价服务。系统包含用户、客服、物流、跑腿员及订单模块,功能完善。此外,小程序增设信息咨询发布、校园社区建设和活动组织等功能,助力校友互动、经验分享及感情联络,构建紧密的校友网络。
217 1
2025商业版拓展校园圈子论坛网络的创新解决方案:校园跑腿小程序系统架构
|
2月前
|
缓存 小程序 前端开发
商城/点餐/家政类小程序源码合集_微信抖音小程序源码开发从入门到精通实战
本文系统讲解如何利用现有源码快速开发商城、点餐、家政类微信/抖音小程序,涵盖环境搭建、核心功能实现、多平台部署与优化,提供完整技术方案。实战导向,助力开发者高效入门与落地。
|
2月前
|
小程序 PHP 图形学
热门小游戏源码(Python+PHP)下载-微信小程序游戏源码Unity发实战指南​
本文详解如何结合Python、PHP与Unity开发并部署小游戏至微信小程序。涵盖技术选型、Pygame实战、PHP后端对接、Unity转换适配及性能优化,提供从原型到发布的完整指南,助力开发者快速上手并发布游戏。
|
2月前
|
存储 小程序 Java
热门小程序源码合集:微信抖音小程序源码支持PHP/Java/uni-app完整项目实践指南
小程序已成为企业获客与开发者创业的重要载体。本文详解PHP、Java、uni-app三大技术栈在电商、工具、服务类小程序中的源码应用,提供从开发到部署的全流程指南,并分享选型避坑与商业化落地策略,助力开发者高效构建稳定可扩展项目。
|
5月前
|
消息中间件 缓存 小程序
婚恋交友相亲公众号app小程序系统源码「脱单神器」婚恋平台全套代码 - 支持快速二次开发
这是一套基于SpringBoot + Vue3开发的婚恋交友系统,支持微信公众号、Uniapp小程序和APP端。系统包含实名认证、智能匹配、视频相亲、会员体系等功能,适用于婚恋社交平台和相亲交友应用。后端采用SpringBoot 3.x与MyBatis-Plus,前端使用Vue3与Uniapp,支持快速部署和二次开发。适合技术团队或有经验的个人创业者使用。
378 8
|
4月前
|
小程序 Java 关系型数据库
圈子系统公众号app小程序系统源码圈子系统带即时通讯 多级圈子系统源码 兴趣小组系统开源 私密圈子系统代码 会员制社区系统
本圈子系统解决方案提供即时通讯、多级圈子、兴趣小组、私密社区及会员制管理功能。支持开源与商业方案,推荐ThinkSNS+、EasyClub及OpenFire等系统,并提供前后端技术选型建议,助力快速搭建社交平台。
213 0
|
7月前
|
监控 前端开发 小程序
陪练,代练,护航,代打小程序源码/前端UNIAPP-VUE2.0开发 后端Thinkphp6管理/具备家政服务的综合型平台
这款APP通过技术创新,将代练、家政、娱乐社交等场景融合,打造“全能型生活服务生态圈”。以代练为切入点,提供模块化代码支持快速搭建平台,结合智能匹配与技能审核机制,拓展家政服务和商业管理功能。技术架构具备高安全性和扩展性,支持多业务复用,如押金冻结、录屏监控等功能跨领域应用。商业模式多元,包括交易抽成、增值服务及广告联名,同时设计跨领域积分体系提升用户粘性,实现生态共生与B端赋能。
659 12
|
7月前
|
小程序 Java 关系型数据库
weixin163基于微信小程序的校园二手交易平台系统设计与开发ssm(文档+源码)_kaic
本文介绍了一款基于微信小程序的校园二手物品交易平台的开发与实现。该平台采用Java语言开发服务端,使用MySQL数据库进行数据存储,前端以微信小程序为载体,支持管理员和学生两种角色操作。管理员可管理用户、商品分类及信息、交易记录等,而学生则能注册登录、发布购买商品、参与交流论坛等。系统设计注重交互性和安全性,通过SSM框架优化开发流程,确保高效稳定运行,满足用户便捷交易的需求,推动校园资源共享与循环利用。
|
8月前
|
小程序 Java 关系型数据库
weixin116大学生就业平台微信小程序+ssm(文档+源码)_kaic
本文介绍了一款大学生就业平台微信小程序的开发过程,涵盖开发环境、系统设计、实现与测试等方面。该小程序基于微信平台特性,采用MYSQL数据库存储数据,确保系统稳定与安全,同时满足学生、企业和管理员不同权限用户的功能需求。通过简化操作流程,实现了招聘信息查看、简历投递等实用功能,旨在为用户提供便捷高效的求职体验,符合“操作简单,功能实用”的设计理念。
|
8月前
|
网络协议 物联网
VB6网络通信软件上位机开发,TCP网络通信,读写数据并处理,完整源码下载
本文介绍使用VB6开发网络通信上位机客户端程序,涵盖Winsock控件的引入与使用,包括连接服务端、发送数据(如通过`Winsock1.SendData`方法)及接收数据(利用`Winsock1_DataArrival`事件)。代码实现TCP网络通信,可读写并处理16进制数据,适用于自动化和工业控制领域。提供完整源码下载,适合学习VB6网络程序开发。 下载链接:[完整源码](http://xzios.cn:86/WJGL/DownLoadDetial?Id=20)
299 12