首页 > 其他分享 >【Netty入门和实践】1.传统的socket分析

【Netty入门和实践】1.传统的socket分析

时间:2023-03-19 11:04:48浏览次数:37  
标签:Netty Socket read 入门 socket import 服务端 客户端


我们知道,使用Java进行TCP/UDP协议的网络通信一般使用Java的Net包下的Socket服务进行编写,有Server服务端和Client客户端,服务端用于监听客户端的连接和接收客户端发来的信息,而客户端可以接收服务端发送的信息,形成多个客户端与一个服务端连接,各个客户端可以相互通信的机制。



下面我们就编写一个简单的基于Socket服务的TCP协议传输信息的服务端,然后来进行分析。



在Eclipse中新建一个Java工程,名为“IOServer”即是服务端:


【Netty入门和实践】1.传统的socket分析_客户端


然后在src下面创建一个cn.com.io.old包,下面创建一个名为“Old_IO_Server”的类:


【Netty入门和实践】1.传统的socket分析_客户端_02


在该类中编写服务端代码:

package cn.com.io.old;

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

/***
* 传统socket服务端
*/
public class Old_IO_Server {
public static void main(String[] args) throws IOException {
//创建Socket服务,监听10010端口
ServerSocket server = new ServerSocket(10010);
System.out.println("服务端启动!");
while(true){
//获取一个套接字(阻塞)
final Socket socket = server.accept();
System.out.println("出现一个新客户端!");
//业务处理
handler(socket);
}
}

/**
* 读取数据
* @param socket
* @throws Exception
* */
private static void handler(Socket socket){
try {
byte [] bytes = new byte[1024];
InputStream input = socket.getInputStream();

int read = 0;
while(read!=-1){
//读取数据(阻塞)
read = input.read(bytes);
System.out.println(new String(bytes,0,read));
}
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
System.out.println("socket 关闭");
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

我们在mian函数的第一行打一个断点,使用Debug模式运行该程序:


【Netty入门和实践】1.传统的socket分析_客户端_03


可以看到在客户端没有连接进来的时候,服务端会在server.accept();的位置阻塞等待。


然后使用Windows的CMD控制台的telnet进行客户端连接:


【Netty入门和实践】1.传统的socket分析_NIO_04


可以看到程序向下走了:


【Netty入门和实践】1.传统的socket分析_服务端_05


当程序走到read = input.read(bytes);发生了阻塞等待,等待客户端传输信息:


【Netty入门和实践】1.传统的socket分析_NIO_06


此时在阻塞过程中,意味着这条线程是被这个Socket一直占用着的,其它的Socket不能进来。(再次打开一个CMD控制台使用telnet,可以发现进不来)




当我们在CMD中输入信息(Ctrl+]进入客户端,send发信息),可以看到服务端收到了信息:


【Netty入门和实践】1.传统的socket分析_Netty_07


想要服务端处理多个客户端的信息,就需要为每一个客户端分配一个线程。下面我们修改一下服务端:


package cn.com.io.old;

import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/***
* 传统socket服务端
*/
public class Old_IO_Server {
public static void main(String[] args) throws IOException {
//创建一个缓存线程池
ExecutorService newCachedThreadPool = Executors.newCachedThreadPool();
//创建Socket服务,监听10010端口
ServerSocket server = new ServerSocket(10010);
System.out.println("服务端启动!");
while(true){
//获取一个套接字(阻塞)
final Socket socket = server.accept();
System.out.println("出现一个新客户端!");
//在线程池为新客户端开一个线程
newCachedThreadPool.execute(new Runnable() {
@Override
public void run() {
//业务处理
handler(socket);
}
});
}
}


/**
* 读取数据
* @param socket
* @throws Exception
* */
private static void handler(Socket socket){
try {
byte [] bytes = new byte[1024];
InputStream input = socket.getInputStream();

int read = 0;
while(read!=-1){
//读取数据(阻塞)
read = input.read(bytes);
System.out.println(new String(bytes,0,read));
}
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
System.out.println("socket 关闭");
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

可以看到我们创建了一个缓存线程池,当服务端新连接了一个客户端的时候,就创建一个新的线程为客户端进行服务。



现在使用两个客户端连接10010端口,可以就看到两个客户端的信息都可以接收到:


【Netty入门和实践】1.传统的socket分析_服务端_08


那么我们是不是就可以使用这种结构实现一个长连接服务器呢?这里还是有一个很大的弊端,因为目前我们每个客户端都为其分配了一个线程去运行,如果有一万个客户端进来,我们就要分配一万个线程给客户端使用,这样的资源消耗是十分巨大的。就好像一个饭店每进来一个客人,就为其分配一个服务员给他服务一样,对于饭店来说服务员的开销也是巨大的。



基于以上几点,传统的Socket服务实现长连接服务是不合适的,但是可以实现短连接的服务器,如老版本的tomcat。



总结:


传统IO传输的特点:


(1)至少两处阻塞点(等待客户端连接,等待客户端发送信息)


(2)单线程下只能有一个客户端连接


(3)使用线程池可以连接多个客户端,但是十分消耗性能。



使用NIO就可以解决传统IO的网络传输问题,我们下篇进行介绍。

标签:Netty,Socket,read,入门,socket,import,服务端,客户端
From: https://blog.51cto.com/u_16012040/6130998

相关文章

  • 快速带你入门css
    css复习笔记1.css样式值1.1文字样式1p{2font-size:30px;/*设置文字大小*/3font-weight:bold;/*文字加粗*/4font-style:ital......
  • 【RSocket】使用 RSocket(三)——服务端主动调用客户端方法
    1.编写客户端接收请求的逻辑我们可以在初始化Rsocket实例的时候指定客户端可以被调用的方法,使用acceptor()指定可被调用的方法和方法使用的通信模型类型:通信类型为......
  • Vue2入门之超详细教程三-初识模板语法
    1、简介模板语法就是按照固定的模板去书写代码,分为插值语法和指令语法。差值语法:功能:用于解析标签体内容写法:{{xxxx}},xxx是js表达式,且可以读取......
  • 20230317软件测试入门
    https://zh.wikipedia.org/wiki/软件测试https://www.ibm.com/cn-zh/topics/software-testinghttps://www.softwaretestingmaterial.com/software-testing/https......
  • 新手如何入门性能测试?一文4个章节带你学会性能测试
    本文介绍一下性能测试的基础内容和一些学习经验,主要讲针对服务器端的性能测试。其他代码级性能测试、前端性能测试等属于比较细分的领域,建议大家有需要的时候针对性得去学......
  • 树状数组入门
    前言树状数组作为维护序列区间修改与查询的利器是每一个“OIer”都应该要掌握的知识点今天,我们来详细的整理一下树状数组的知识脉络目录一.树状数组简介二.树状数......
  • day01入门
    java入门常识快捷方式:本质上链接到了真正的程序上,使用方便;环境变量:环境变量是操作系统中的一个配置,专门用来配置路径的,配置到环境变量中的路径,可以在任何地方访问或使用......
  • 【Python从入门到进阶】4、pycharm的安装及使用
    接上篇《​​3、运行python代码​​》上一篇我们学习了如何使用终端和执行文件运行python代码,本篇我们来学习python编程工具pycharm的安装及基本使用。一、IDE的概念上一篇......
  • 【Python从入门到进阶】2、Python环境的安装
    接上篇《​​1、初识Python​​》上一篇我们对Python这门编程语言进行了一个基本的了解,本篇我们来学习如何下载安装Python编程环境,以及如何使用pip管理Python包。本篇讲解......
  • 【Python从入门到进阶】10、流程控制语句-循环语句(for-while)
    接上篇《9、流程控制语句-条件语句(if-else)》上一篇我们学习了Python的控制流语句的概念,以及其中的条件语句(if/else),本篇我们来学习控制流语句中的循环语句(for/while)。......