首页 > 编程语言 >[Java]Socket套接字(网络编程入门)

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

时间:2024-05-23 21:07:45浏览次数:29  
标签:Java Socket System BufferedReader client println new 接字

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)
https://www.cnblogs.com/cnb-yuchen/p/18032037
出自【进步*于辰的博客

参考笔记二,P61。

注:“一对一”、“多对多”是相对于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类。

最后

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

原由

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

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

本文完结。

标签:Java,Socket,System,BufferedReader,client,println,new,接字
From: https://www.cnblogs.com/cnb-yuchen/p/18032037

相关文章

  • DdddOcr 带带弟弟OCR通用验证码和 JAVA调用
    本文主要参考DdddOcr发布的最新版本启动服务端,以及JAVA如何和服务端对接。DdddOcr,其由作者与kerlomz共同合作完成,通过大批量生成随机数据后进行深度网络训练,本身并非针对任何一家验证码厂商而制作,本库使用效果完全靠玄学,可能可以识别,可能不能识别。DdddOcr、最简依赖......
  • [Java]反射
    【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)https://www.cnblogs.com/cnb-yuchen/p/17960654出自【进步*于辰的博客】参考笔记二,P75.3;笔记三,P15.2、P43.2、P44.2、P64.3、P69.1。1、什么是“反射”?关于类加载,详述可查阅博文《[Java]知识点》中的【类加载......
  • Java.年月日正则表达式
    表达式:yyyy-MM:^([1-9]{1}[0-9]{3}[\\-]{1}){1}((1[0-2]{1}){1}|(0[1-9]{1})|([1-9]{1})){1}$yyyy-MM-dd:^((((19|20)\\d{2})(-)(0?[13578]|1[02])-(0?[1-9]|[12]\\d|3[01]))|(((19|20)\\d{2})(-)(0?[469]|11)-(0?[1-9]|[12]\\d|30))|(((19|20)\\d{2})(-)(0......
  • JavaScript中reduce()详解及使用方法。
    一、定义和用法reduce()方法接收一个函数作为累加器,数组中的每个值(从左到右)开始缩减,最终计算为一个值。reduce()可以作为一个高阶函数,用于函数的compose。reduce()方法为归并类方法,最常用的场景就是,计算数组中的每一项的总和。注意:reduce()对于空数组是不会执行回调函数的......
  • Spring AI 抢先体验,5 分钟玩转 Java AI 应用开发
    SpringAI是Spring官方社区项目,旨在简化JavaAI应用程序开发,让Java开发者像使用Spring开发普通应用一样开发AI应用。SpringCloudAlibabaAI以SpringAI为基础,并在此基础上提供阿里云通义系列大模型全面适配,让用户在5分钟内开发基于通义大模型的JavaAI应用。......
  • java 通过 microsoft graph 调用outlook(三)
    这次会添加一个Reply接口,并且使用6.10.0版本 直接上代码一,POM<!--office365--><dependency><groupId>com.microsoft.graph</groupId><artifactId>microsoft-graph</artifactId><......
  • Pairwise实现(Java篇)
    importjava.util.HashMap;/***PairWise(成对)测试方法*author:likeqc*date:2021-4-411:06:59*/classPairWise{/***@paramstrString[][],二维数组,一维数组str[i]中存放第i个因素的因子*/privatestaticvoidsolution(String[][]s......
  • 位运算符在 Javascript 中的运用
    零、资料JavaScript中的位运算和权限设计javascript位运算技巧巧用JS位运算JavaScript位运算及其妙用聊聊JavaScript中的二进制数一、权限在权限设计时,每一个基础权限单元都是二进制数形式,有且只有一位值是1,其余全部是0,即权限码是 2^n 。所以,在这套设......
  • 关于如何使用JNI将C语言接口打包成可供java环境调用的so库文件
    一、环境检查在linux下打包.so文件,首先需要确认是否有安装java环境,可通过在终端中输入指令java的方式来进行查看。如下图所示,则为已安装java环境。  若当前未安装java环境,则可通过在终端中输入如下指令进行安装,我这里使用的java环境为1.8.0版本。sudoapt-getinstallo......
  • 使用-HTML5-和-JavaScript-开发-Windows-商店应用-全-
    使用HTML5和JavaScript开发Windows商店应用(全)原文:zh.annas-archive.org/md5/8F13EC8AC7BDB8535E7218C5DDB48475译者:飞龙协议:CCBY-NC-SA4.0序言使用HTML5和JavaScript开发WindowsStore应用是一本实践性强的指南,涵盖了WindowsStore应用的基本重要特性以及......