搭建Tomcat(一)---SocketServerSocket

news2024/12/16 5:53:29

目录

引入1

引入2--socket 

流程

Socket(应用程序之间的通讯保障)

网卡(计算机之间的通讯保障)

端口

端口号

实例

 client端

解析

server端

解析

相关方法

问题1:ServerSocket和Socket有什么关系?

ServerSocket

Socket

关系和作用总结

 问题2:ServerSocket和Socket的区别?

一、构造方法与参数

二、核心方法

三、通信过程


引入1

Tomcat是一个开源的Java Web服务器和Servlet容器,它遵循Servlet和JavaServer Pages(JSP)规范。Tomcat的架构主要包含两个核心组件:连接器(Connector)和容器(Container)。

那么Tomcat究竟是做什么的呢?---服务器开发

简而言之,一句话总结,tomcat就是一个用来承载项目的web容器

众所周知,我们自己创建的项目是不能随意访问的。

要想让网络能够随意访问这些项目,就要把它们丢进tomcat中。

上面将项目丢进tomcat的操作,就是实现其中配置的项目接受http请求

使用tomcat的过程,就是实现以下路径的转变:

经过上述简单概念,对tomcat的作用和概念有了简单的了解,接下来引入另一个重要概念:Socket。

引入2--socket 

如下的关系图是两个应用程序(或两台计算机)之间传输数据的构造图:

流程

简述不同计算机程序之间是如何互传信息的流程:

本台应用程序A输出一些数据信息,并且通过socket传输给本台操作系统对数据进行处理后变成由0和1组成的二进制数据流信号,进而被本台的网卡通过计算机网络(WiFi等)传输给另一台计算机的网卡接收,经另一台计算机的操作系统处理信息后,通过socket传输给应用程序B上展示。

以上就是一个完整的工作流程。

Socket(应用程序之间的通讯保障)

Socket(套接字)是网络通信中的一个抽象接口,它提供了进程间通信的端点。Socket允许同一台计算机或不同计算机上的应用程序通过网络进行通信,实现了跨网络的进程间通信。Socket不仅包含了IP地址和端口号等通信信息,还提供了数据传输和接收的接口。通过Socket,应用程序可以建立网络连接、发送和接收数据,实现网络通信的各种功能。

(每个Socket都对应一个来源,多个socket对应多个来源;Socket会一直保存来源线程,我们只需要通过Response直接返回到socket上即可!)

网卡(计算机之间的通讯保障)

网卡通常安装在计算机的主板插槽中,或者通过USB、PCIe等接口与计算机连接。它的主要功能是实现计算机与网络之间的数据交换,即将计算机内部的数据通过网络传输到外部,同时将外部网络的数据传输到计算机内部。

简而言之,网卡就是相互之间通过计算机网络接收和发送0101这种信息的一种物理硬件。

端口

端口是网络通信中的逻辑接口,用于区分不同的服务和应用进程。每个端口都有一个唯一的标识符,即端口号,用于标识特定的服务或应用进程。当计算机通过网络发送数据时,数据会被封装成数据包,并标记上目标端口号,以便接收端能够正确识别和处理。端口的作用在于确保数据能够准确地被发送到目标服务或应用进程,实现网络通信的精确性和可靠性。

端口号

  1. 作用:端口号用于唯一标识一个网络服务或应用程序。当计算机接收到一个数据包时,它会检查数据包中的目标端口号,并将数据包发送到监听该端口的服务或应用程序。
  2. 范围:端口号的范围通常是0到65535。其中,0到1023是系统保留端口,通常被用于常见的网络服务,如HTTP(80端口)、HTTPS(443端口)、FTP(21端口)等。这些端口通常被系统或管理员保留,不建议普通用户随意使用。1024到65535是用户端口或动态端口,可以由用户根据需要自由分配。

 如下,就是一个对端口号应用的例子:

通过端口号实现一台计算机上不同网络服务或应用程序的区分,以保证数据信息能够准确无误的传输。

实例

下面来实现客户端向服务端发送信息,以及服务端接收到并且输出的代码:

 client端

import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.net.Socket;
import java.util.Scanner;

public class Client {
    public static void main(String[] args) throws IOException {
        //客户端只需要从本地IP地址向指定的端口去发送
        Socket socket=new Socket("127.0.0.1",8080);
        //打开输出流--准备传送数据
        OutputStream outputStream=socket.getOutputStream();
        //创建输出流类型的流数据类型参数
        PrintStream printStream=new PrintStream(outputStream);
        Scanner in=new Scanner(System.in);
        String x=in.nextLine();//获取一行的数据
        printStream.println(x);
        printStream.close();
        in.close();
    }
}

解析

接下来开始解析上述的代码:

Socket socket=new Socket("127.0.0.1",8080);

 "127.0.0.1"是本地IP,8080是一个端口号,这一条表示声明一个Socket对象,用于建立本地客户端与服务器之间的连接,通过8080端口。

 OutputStream outputStream=socket.getOutputStream();

通过Socket对象的getOutputStream()方法获取输出流;这个输出流用于向服务器发送数据 

 PrintStream printStream=new PrintStream(outputStream);

创建一个PrintStream对象,将OutputStream作为参数传入(将将 OutputStream 包装成 PrintStream

为什么使用PrintStream?

PrintStream提供了丰富的打印方法,使得向输出流写入数据变得更加方便,(如 print()println()printf() 等),这些方法可以自动将参数转换为字符串并写入到输出流中,同时,PrintStream还提供自动刷新功能,每次写入后都会自动调用 flush() 方法,确保数据被及时发送到输出流。

客户端的输出流一定要使用PrintStream吗?

  1. 如果数据是纯字节数据(如图像、音频文件等),那么直接使用 OutputStream 的 write() 方法就足够了,不需要将 OutputStream 包装成 PrintStream

  2. 如果数据是文本数据,并且需要格式化输出(如添加换行符、格式化数字等),那么使用 PrintStream 会更加便捷。

  3. 在客户端向服务器发送文本数据时,虽然可以直接使用 OutputStream 的 write(byte[] b, int off, int len) 方法来发送字符串的字节表示(通过字符串的 getBytes() 方法获取字节数组),但这样做需要手动处理字符串的编码和转换,而且不方便添加换行符等文本格式。因此,使用 PrintStream 可以简化这些操作。

Scanner in=new Scanner(System.in);
String x=in.nextLine();//获取一行的数据

键盘输出一行作为传送信息

 printStream.println(x);
 printStream.close();
 in.close();

使用PrintStream的println()方法将用户输入的字符串发送到服务器;其中println()方法会在字符串末尾自动添加换行符,并将其写入到输出流中。

完成这一切后,关闭scanner和printStream。

server端

import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class Server {
    public static void main(String[] args) throws IOException {
        run();
    }
    public static void run() throws IOException {
        ServerSocket serverSocket=new ServerSocket(8080); //0~65535监听接口
        while(true){//一直等待客户端连接(无限循环)
            //无限循环就意味着需要做出解除方法
            Socket socket= serverSocket.accept();//接收到client信息后,将获取的请求初始化给Socket(阻塞监听)
            //输出客户端给我们发来的程序
            InputStream inputStream=socket.getInputStream(); //打开输入流-接收输入的信息--并赋值给inputStream
            int count=0;
            while(count==0){
                count= inputStream.available(); //获取可读出的字节数
            }
            byte[] bytes=new byte[count]; //开辟适合大小的byte数组
            inputStream.read(bytes);// 从输入流中读取数据到byte数组
            String context=new String(bytes);//将输入流转化为字符串输出
            System.out.println(context);
        }
    }
}

解析

接下来开始解析上述的代码:

ServerSocket serverSocket = new ServerSocket(8080);

创建一个监听端口8080的ServerSocket实例。端口号8080是任选的,但必须是0到65535之间的有效端口。

while (true)服务器持续运行,等待客户端连接。

Socket socket = serverSocket.accept();

accept方法会阻塞(暂停执行),直到一个客户端连接到服务器。一旦有连接,它就返回一个Socket实例,用于与该客户端通信。

 //输出客户端给我们发来的程序
            InputStream inputStream=socket.getInputStream(); //打开输入流-接收输入的信息--并赋值给inputStream
            int count=0;
            while(count==0){
                count= inputStream.available(); //获取可读出的字节数
            }
            byte[] bytes=new byte[count]; //开辟适合大小的byte数组
            inputStream.read(bytes);// 从输入流中读取数据到byte数组
            String context=new String(bytes);//将输入流转化为字符串输出
            System.out.println(context);
  • InputStream inputStream = socket.getInputStream();:从客户端连接获取输入流,用于读取客户端发送的数据(创建输入流)。
  • int count = 0; while (count == 0) { count = inputStream.available(); }:这段代码读取输入流中可用的字节数。
  • byte[] bytes = new byte[count];:根据available()返回的字节数创建一个字节数组。
  • inputStream.read(bytes);:从输入流中读取数据到字节数组。
  • String context = new String(bytes);:将字节数组转换为字符串。方便下一步输出。
  • System.out.println(context);:在控制台上打印从客户端接收到的字符串。

运行1:

在client端借用socket输出流时(这个方法毋庸置疑,毕竟这两段代码的本质目的就是为了这个)

我在客户端输入“你好!”

将在服务器端接收到“你好!”

运行2

我在浏览器输入:

 

也能在服务端拦截到这个信息(这是因为服务器一直在拦截识别8080端口的信息,不管出自哪里,所用通路是一样的):

 

相关问题

通过上述的讲解和应用,罗列了以下问题:

问题1:ServerSocket和Socket有什么关系?

ServerSocket

ServerSocket 类位于 java.net 包中,它用于创建一个服务器端套接字,该套接字可以监听来自客户端的连接请求。服务器通过绑定到一个特定的端口来等待客户端的连接。一旦有客户端尝试连接到该端口,ServerSocket 就会接受这个连接,并返回一个新的 Socket 实例,该实例代表了服务器与客户端之间的通信链路。

ServerSocket 的主要作用包括:

  1. 监听端口:服务器绑定到一个特定的端口,等待客户端的连接请求。
  2. 接受连接:通过 accept() 方法阻塞等待,直到一个客户端连接请求到达。
  3. 返回通信链路:一旦连接建立,accept() 方法会返回一个 Socket 对象,用于后续的通信。

Socket

Socket 类同样位于 java.net 包中,它代表了一个客户端套接字,或者是一个服务器与客户端之间已经建立的通信链路(在服务器端,这个 Socket 对象是由 ServerSocket 的 accept() 方法返回的)。Socket 提供了输入流和输出流,用于在客户端和服务器之间发送和接收数据。

Socket 的主要作用包括:

  1. 建立连接:客户端通过 Socket 类的构造函数或 connect() 方法尝试连接到服务器。
  2. 获取输入/输出流:通过 getInputStream() 和 getOutputStream() 方法获取用于读取和写入数据的流。
  3. 通信:使用输入/输出流与服务器进行通信,发送和接收数据。
  4. 关闭连接:通过 close() 方法关闭与服务器的连接。

关系和作用总结

  • ServerSocket 用于服务器端,负责监听端口和接受客户端的连接请求。
  • Socket 用于客户端(或者服务器端与客户端之间的通信链路),负责建立连接、发送和接收数据。
  • 服务器端通过 ServerSocket 接受连接后,会返回一个 Socket 对象,该对象用于与客户端进行后续的通信。
  • 客户端通过 Socket 连接到服务器后,同样获得一个 Socket 对象(这个对象在客户端创建时就已经获得),用于与服务器进行通信。

 问题2:ServerSocket和Socket的区别?

一、构造方法与参数

  1. Socket

    • 常见的构造方法有:
      • Socket(String host, int port):创建一个Socket并连接到指定主机上的指定端口。
      • Socket(InetAddress address, int port):创建一个Socket并连接到指定IP地址和端口。
  2. ServerSocket

    • 常见的构造方法有:
      • ServerSocket(int port):创建一个绑定到指定端口的ServerSocket实例。
      • ServerSocket(int port, int backlog):创建一个绑定到指定端口的ServerSocket实例,并设置客户端连接请求队列的长度。
      • ServerSocket(int port, int backlog, InetAddress bindAddr):创建一个绑定到指定端口和IP地址的ServerSocket实例,并设置客户端连接请求队列的长度。

二、核心方法

  1. Socket

    • connect(SocketAddress endpoint):连接到服务器。
    • getInputStream():获取输入流,用于从服务器接收数据。
    • getOutputStream():获取输出流,用于向服务器发送数据。
  2. ServerSocket

    • accept():监听并接受客户端的连接。此方法会阻塞,直到有客户端连接为止。一旦连接建立,它会返回一个新的Socket对象,表示与客户端的连接。
    • getLocalPort():获取服务器套接字绑定的本地端口。
    • isClosed():检查ServerSocket是否已关闭。

三、通信过程

  1. Socket

    • 客户端创建一个Socket对象,指定服务器地址和端口。
    • 客户端通过Socket对象的输出流向服务器发送数据。
    • 客户端通过Socket对象的输入流接收服务器的响应。
  2. ServerSocket

    • 服务器创建一个ServerSocket对象,监听指定的端口。
    • 服务器调用ServerSocket对象的accept()方法,等待客户端的连接请求。
    • 一旦有客户端连接到服务器,ServerSocket会创建一个Socket对象与客户端进行通信。
    • 服务器通过Socket对象的输入流接收客户端发送的数据。
    • 服务器通过Socket对象的输出流发送响应给客户端。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2260333.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot快速使用

一些名词的碎碎念: 1> 俩种网络应用设计模式 C/S 客户端/服务器 B/S 浏览器/服务器 俩者对比: 2> 集群和分布式的概念 集群: 分布式: 例子: 一个公司有一个人身兼多职 集群: 招聘N个和上面这个人一样身兼多职 分布式: 招聘N个人,分担上面这个人的工作,进行工作的拆分. 工…

【含开题报告+文档+PPT+源码】基于SpringBoot的开放实验管理平台设计与实现

开题报告 设计开放实验管理平台的目的在于促进科学研究与教学的融合。传统实验室常常局限于特定地点和时间,而开放平台可以为学生、教师和研究人员提供一个便捷的交流与共享环境。通过在线平台,他们可以分享实验资源、交流经验,从而促进科学…

分布式 漏桶算法 总结

前言 相关系列 《分布式 & 目录》《分布式 & 漏桶算法 & 总结》《分布式 & 漏桶算法 & 问题》 概述 简介 LBA Leaky Bucket Algorithm 漏桶算法是一种流行于网络通信领域的流量控制/频率限制算法。漏桶算法的核心原理是通过一个概念上的“漏桶”来…

linux glances vs top

一、安装 apt-get install glances glances top显示效果:

CTF知识集-PHP特性

title: CTF知识集-PHP特性 写在开头可能会用到的提示 call_user_func 调用的函数可以不区分大小写preg_match过滤存在长度溢出,长度超过100w检测失效。str_repeat(‘show’,250000); 生成100w个字符preg_match是无法处理数组的,例如:preg_match( n u m…

Hadoop运行Mapreduce问题集锦——Ubuntu虚拟机配置

一、端口访问问题 问题描述 运行任务前一直重连。具体来说,错误发生在尝试从czs-virtual-machine虚拟机的127.0.1.1地址连接到同一台机器的8032端口时,连接被拒绝。 如下: 2024-11-17 23:05:45,800 INFO retry.RetryInvocationHandler: java…

【经验分享】搭建本地训练环境知识点及方法

最近忙于备考没关注,有次点进某小黄鱼发现首页出现了我的笔记还被人收费了 虽然我也卖了一些资源,但我以交流、交换为主,笔记都是免费给别人看的 由于当时刚刚接触写的并不成熟,为了避免更多人花没必要的钱,所以决定公…

流程引擎Activiti性能优化方案

流程引擎Activiti性能优化方案 基于关系型数据库层面优化 MySQL建表语句优化 Activiti在MySQL中创建默认字符集为utf8(即utf8mb3)格式,本文将默认字符集设置为utf8mb4,排序规则为utf8mb4_general_ci,并修改变量等类…

Unix 传奇 | 谁写了 Linux | Unix birthmark

注:本文为 “左耳听风”陈皓的 unix 相关文章合辑。 皓侠已走远,文章有点“年头”,但值得一阅。 文中部分超链已沉寂。 Unix 传奇 (上篇) 2010 年 04 月 09 日 陈皓 了解过去,我们才能知其然,更知所以然。总结过去…

TimerPickerDialog组件的用法

文章目录 1 概念介绍2 使用方法3 示例代码我们在上一章回中介绍了Snackbar Widget相关的内容,本章回中将介绍TimePickerDialog Widget.闲话休提,让我们一起Talk Flutter吧。 1 概念介绍 我们在这里说的TimePickerDialog是一种弹出窗口,只不过窗口的内容固定显示为时间,它主…

大模型系列4--开源大模型本地部署到微调(WIP)

背景 一直想真正了解大模型对硬件资源的需求,于是准备详细看一篇视频,将核心要点总结记录下。本文内容参考视频:保姆级教程:6小时掌握开源大模型本地部署到微调,感谢up主 训练成本 训练 > 微调 > 推理训练GPT…

现代密码学总结(上篇)

现代密码学总结 (v.1.0.0版本)之后会更新内容 基本说明: ∙ \bullet ∙如果 A A A是随机算法, y ← A ( x ) y\leftarrow A(x) y←A(x)表示输入为 x x x ,通过均匀选择 的随机带运行 A A A,并且将输出赋给 y y y。 ∙ \bullet …

Python中opencv的一些函数及应用

Sobel 算子函数 功能: Sobel 算子用于计算图像的梯度(变化率),常用于边缘检测。它通过对图像应用一个基于一阶导数的滤波器来强调图像中的边缘部分,特别是水平和垂直方向上的边缘。通过计算图像的梯度,可以…

【docker】springboot 服务提交至docker

准备docker (不是docker hub或者harbor,就是可以运行docker run的服务),首先确保docker已经安装。 本文以linux下举例说明: systemctl stats docker ● docker.service - Docker Application Container EngineLoaded…

XDOJ 877 图的深度优先遍历

题目:图的深度优先遍历 问题描述 已知无向图的邻接矩阵,以该矩阵为基础,给出深度优先搜索遍历序列,并且给出该无向图的连通分量的个数。在遍历时,当有多个点可选时,优先选择编号小的顶点。(即…

大屏开源项目go-view二次开发1----环境搭建(C#)

最近公司要求做一个大屏的程序用于展示公司的产品,我以前也没有相关的经验,最糟糕的是公司没有UI设计的人员,领导就一句话要展示公司的产品,具体展示的内容细节也不知道,全凭借自己发挥。刚开始做时是用wpf做的&#x…

摆脱B端UI框架的桎梏,首先从布局开始

在 B 端开发中,UI 框架虽带来便利,但也可能形成桎梏。要摆脱这种束缚,首先从布局着手是个明智之举。传统的 B 端 UI 框架布局可能较为固定,缺乏灵活性。我们可以尝试创新的布局方式,如响应式设计,适应不同屏…

认识异常吧

在 Java 中,将程序执行过程中发生的不正常行为称为异常 。 异常的体系结构 1. Throwable : 是异常体系的顶层类,其派生出两个重要的子类 , Error(错误) 和 Exception(异常) 2. Error &…

Repo管理

文章目录 前言Repo介绍清单仓库清单仓库的组成 初始化Repo同步远程仓库Repo实际应用 前言 我们知道,Git是用来管理某一个仓库,那当一个项目用到了多个仓库时,怎么来同步管理这些仓库呢?这个时候就可以引入Repo管理。 Repo介绍 …

Spring Security6.3 自定义AuthorizationManager问题

项目环境: Springboot3.3.5, 对应的SpringFrameWork6.1,Security为6.3 问题:我想自定义AuthorizationManager接口实现类,在里面判断如果角色为amdin则放行请求; 在AdminAuthorizationManager类的check()方法中pass变量…