[Java]Socket套接字(网络编程入门)

简介: [Java]Socket套接字(网络编程入门)

1、概述

两种聊天模式的共同点:ClientServer都是以管道(IO流)的方式进行一对一连接、通信,故在"多对多"聊天模式的实现中,需要循环接收多个Socket(客户端,见第2.2项)。(大家可能云里雾里,继续看)

从IO流的角度:

  1. Client,读取input流是获取回复,output流用于发送请求;
  2. Server:读取input是获取请求,output流用于响应。

ClientServer 发送消息的实现都基于阻塞,实现“聊天”的本质就是循环“发送-接收”消息的过程。

2、二种聊天模式

2.1 “一对一”

看下述代码:

1、服务端类Server

/**
 * 功能:Server 与 Client 一对一聊天
 */
public class Server {
    public static void main(String[] args) throws Exception {
        ServerSocket server = new ServerSocket(8844);
        System.out.println("等待:");
        Socket client = server.accept();// 阻塞,当有 Socket 访问时,返回此 Socket,阻塞放开
        System.out.println("连接成功");
        BufferedReader req = new BufferedReader(new InputStreamReader(client.getInputStream()));// 缓冲输入流,用于获取请求
        PrintWriter resp = new PrintWriter(client.getOutputStream(), true);// 缓冲输出流,用于响应。true → 自动刷新
        BufferedReader in = new BufferedReader(new InputStreamReader(System.in));// 用于获取键盘输入
        while (true) {
            String reqStr = req.readLine();// 读取 Socket 输入流,因为 Client 还未输入,阻塞。当 Client 输入后,阻塞放开
            System.out.println("请求 = " + reqStr);
            System.out.print("input:");
            String respStr = in.readLine();// 获取键盘输入,阻塞
            resp.println(respStr);// 输出响应(发送回复)
        }
    }
}

2、客户端类·Client·。

public class Client {
    public static void main(String[] args) throws Exception {
        Socket client = new Socket("127.0.0.1", 8844);
        BufferedReader in = new BufferedReader(new InputStreamReader(System.in));// 缓冲输入流,用于获取键盘输入
        PrintWriter req = new PrintWriter(new OutputStreamWriter(client.getOutputStream()), true);// 缓冲输出流,用于发送请求
        BufferedReader resp = new BufferedReader(new InputStreamReader(client.getInputStream()));// 用于获取响应
        while (true) {
            System.out.print("input:");
            String reqStr = in.readLine();// 获取键盘输入,阻塞
            req.println(reqStr);// 发送请求
            String respStr = resp.readLine();// 读取输入流(获取响应)
            System.out.println("响应 = " + respStr);
        }
    }
}

127.0.0.1是本地ip,如果服务端不在本地,可以使用cmd → ipconfig 或 ipconfig/all查询本机ip

2.2 “多对多”

看下述代码:

1、服务端类Server

/**
 * 业务:负责将各个客户端发送的信息广播推送
 */
public class Server {
    private static Set<Socket> set = Collections.synchronizedSet(new HashSet<>());
    public static void main(String[] args) throws Exception {
        ServerSocket server = new ServerSocket(8844);
        System.out.println("等待:");
        while (true) {
            Socket client = server.accept();
            System.out.println("连接成功");
            set.add(client);
            /**
             * 为什么使用线程?
             *      1、实时接收各个客户端消息;
             *      2、实时推送;
             *      3、为了能获取各个客户端信息。
             * 这些工作只能在后台运行,而不能在主线程执行,因为一个线程(主线程)会因阻塞、导致后续其他工作一并终止。
             * 因此,需要线程,并且为每个客户端都创建一个。
             */
            new Thread(() -> {
                /**
                 * 为什么在线程内获取输入流等信息,而不是线程外?
                 *      因为线程外属于主线程,如 req 这些变量都是局部变量,会被后续覆盖,
                 *      因此置于线程内,这样每个线程都独立保存着当前客户端的信息。
                 */
                try (BufferedReader req = new BufferedReader(new InputStreamReader(client.getInputStream()))) {
                    while (true) {
                        String reqStr = req.readLine();
                        for (Socket current : set) {
                            if (current == client)// 向所有客户端推送消息,排除发送者
                                continue;
                            // 根据当前客户端对象获取输出管道(输出流)推送消息
                            PrintWriter resp = new PrintWriter(current.getOutputStream(), true);
                            resp.println(reqStr);
                        }
                        System.out.println(new Date());
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }
}

2、客户端类Client

public class Client {
    public static void main(String[] args) throws Exception {
        Socket client = new Socket("127.0.0.1", 8844);
        BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
        PrintWriter req = new PrintWriter(new OutputStreamWriter(client.getOutputStream()), true);
        BufferedReader resq = new BufferedReader(new InputStreamReader(client.getInputStream()));
        /**
         * 为什么使用线程?
         *      为了实时获取推送消息
         */
        new Thread(() -> {
            try {
                while (true) {
                    String respStr = resq.readLine();
                    System.out.println("响应 = " + respStr);
                }
            }catch (Exception e) {
                e.printStackTrace();
            }
        }).start();
        // 发送请求
        while (true) {
            System.out.println("input:");
            String reqStr;
            if (!(reqStr = in.readLine()).isEmpty())
                req.println(reqStr);
            else
                System.out.println("info can't empty");
        }
    }
}

若要生成多个客户端进行聊天,正确的做法不是重启类,因为那样的结果是覆盖,结果还是一个客户端。因此,需要再创建Client类。

3、最后

本文的例子是为了阐述Socket套接字的使用、方便大家理解而简单举出的,不一定有实用性。并且,示例功能比较粗糙。为什么我不添加一些功能?比如:使用swinghtml生成一个聊天界面、完善聊天功能,等等。

原由: \color{green}{原由:}原由:

  1. 本文的核心是阐述Socket套接字的使用
  2. Socket套接字只是网络编程的开端;
  3. 单一使用Socket难以开发性能优良、功能完善的聊天功能。

因此,后续我会使用WebSocket实现。

本文完结。

相关文章
|
9天前
|
应用服务中间件 nginx Docker
【与时俱进】网络工程师必备技能:Docker基础入门指南,助你轻松应对新时代挑战!
【8月更文挑战第22天】随着容器技术的发展,Docker已成为开发与运维的关键工具。本文简要介绍Docker——一种开源容器化平台,能让应用程序及依赖项被打包成轻量级容器,在任何Linux或Windows机器上运行。文中涵盖Docker的安装步骤、基础命令操作如启动服务、查看版本、拉取与运行容器等。并通过实例演示了如何运行Nginx服务器和基于Dockerfile构建Python Flask应用镜像的过程。这些基础知识将助力网络工程师理解Docker的核心功能,并为实际应用提供指导。
38 2
|
8天前
|
网络协议 安全 网络安全
网络编程:基于socket的TCP/IP通信。
网络编程:基于socket的TCP/IP通信。
61 0
|
1天前
|
运维 安全 应用服务中间件
自动化运维的利器:Ansible入门与实战网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享
【8月更文挑战第30天】在当今快速发展的IT时代,自动化运维已成为提升效率、减少错误的关键。本文将介绍Ansible,一种流行的自动化运维工具,通过简单易懂的语言和实际案例,带领读者从零开始掌握Ansible的使用。我们将一起探索如何利用Ansible简化日常的运维任务,实现快速部署和管理服务器,以及如何处理常见问题。无论你是运维新手还是希望提高工作效率的资深人士,这篇文章都将为你开启自动化运维的新篇章。
|
3天前
|
编解码 网络协议 Oracle
java网络编程入门以及项目实战
这篇文章是Java网络编程的入门教程,涵盖了网络编程的基础知识、IP地址、端口、通讯协议(TCP和UDP)的概念与区别,并提供了基于TCP和UDP的网络编程实例,包括远程聊天和文件传输程序的代码实现。
java网络编程入门以及项目实战
|
4天前
|
数据采集 机器学习/深度学习 人工智能
Python爬虫入门指南探索AI的无限可能:深度学习与神经网络的魅力
【8月更文挑战第27天】本文将带你走进Python爬虫的世界,从基础的爬虫概念到实战操作,你将学会如何利用Python进行网页数据的抓取。我们将一起探索requests库和BeautifulSoup库的使用,以及反爬策略的应对方法。无论你是编程新手还是有一定经验的开发者,这篇文章都将为你打开一扇通往数据抓取世界的大门。
|
15天前
|
分布式计算 网络协议 Python
Python网络编程:socket编程
Socket 编程是网络编程的重要部分,主要用于在不同计算机之间进行通信。Python 提供了一个非常强大的 socket 库,使得网络编程变得简单和灵活。本篇博文将详细介绍 Python 的 socket 编程,包括基础概念、核心组件、常用功能,并附上一个综合的示例及其运行结果。
|
13天前
|
网络协议 Java
一文讲明TCP网络编程、Socket套接字的讲解使用、网络编程案例
这篇文章全面讲解了基于Socket的TCP网络编程,包括Socket基本概念、TCP编程步骤、客户端和服务端的通信过程,并通过具体代码示例展示了客户端与服务端之间的数据通信。同时,还提供了多个案例分析,如客户端发送信息给服务端、客户端发送文件给服务端以及服务端保存文件并返回确认信息给客户端的场景。
一文讲明TCP网络编程、Socket套接字的讲解使用、网络编程案例
|
1天前
|
Kubernetes Cloud Native 网络安全
云原生入门指南:Kubernetes和容器化技术云计算与网络安全:技术融合的新篇章
【8月更文挑战第30天】在云计算的浪潮中,云原生技术如Kubernetes已成为现代软件部署的核心。本文将引导读者理解云原生的基本概念,探索Kubernetes如何管理容器化应用,并展示如何通过实践加深理解。
|
2天前
|
机器学习/深度学习 PyTorch 测试技术
深度学习入门:使用 PyTorch 构建和训练你的第一个神经网络
【8月更文第29天】深度学习是机器学习的一个分支,它利用多层非线性处理单元(即神经网络)来解决复杂的模式识别问题。PyTorch 是一个强大的深度学习框架,它提供了灵活的 API 和动态计算图,非常适合初学者和研究者使用。
14 0
|
5天前
|
网络协议 Linux 应用服务中间件
Socket通信之网络协议基本原理
【8月更文挑战第27天】网络协议定义了机器间通信的标准格式,确保信息准确无损地传输。主要分为两种模型:OSI七层模型与TCP/IP模型。
下一篇
云函数