一、阻塞

服务器端

public class BIOServer {

    public static void main(String[] args) throws Exception {
ServerSocket sc = new ServerSocket(9093);
System.out.println("服务器启动成功!");
while (!sc.isClosed()) {
Socket request = sc.accept(); // 阻塞
System.out.println("收到新连接:" + request.toString());
try {
InputStream is = request.getInputStream(); // net+i/o
BufferedReader reader = new BufferedReader(new InputStreamReader(is, "utf-8"));
String msg;
while ((msg = reader.readLine()) != null) { // 没有数据会阻塞
if (msg.length() == 0) {
break;
}
System.out.println(msg);
}
System.out.println("收到数据,来自:" + request.toString());
} catch (Exception e) {
// TODO: handle exception
} finally {
try {
request.close();
} catch (Exception e2) {
// TODO: handle exception
}
} }
}
}

sc.accept()会使服务端一直阻塞,直到连接被创建

InputStream也是阻塞的

客户端

public class BIOClient {

    public static void main(String[] args) throws IOException {
Socket s = new Socket("localhost", 9093);
OutputStream out = s.getOutputStream(); Scanner scanner = new Scanner(System.in);
System.out.println("请输入");
String msg = scanner.nextLine();
out.write(msg.getBytes());
scanner.close();
s.close();
} }

OutputStream也是阻塞的,写完成之后才会返回

当同时启动两个客户端的时候

服务器只建立了一个连接,并等待客户端的输入

我们在被等待的客户端输入123

服务器收到123,并建立了一个新的连接

这不满足我们的需求,下面引入多线程,升级服务器端

二、多线程引入

线程池

public class BIOServer {

    private static ExecutorService threadPool = Executors.newCachedThreadPool();

    public static void main(String[] args) throws Exception {
ServerSocket sc = new ServerSocket(9093);
System.out.println("服务器启动成功!");
while (!sc.isClosed()) {
Socket request = sc.accept(); // 阻塞
System.out.println("收到新连接:" + request.toString());
threadPool.execute(() -> {
try {
InputStream is = request.getInputStream(); // net+i/o
BufferedReader reader = new BufferedReader(new InputStreamReader(is, "utf-8"));
String msg;
while ((msg = reader.readLine()) != null) {
if (msg.length() == 0) {
break;
}
System.out.println(msg);
}
System.out.println("收到数据,来自:" + request.toString());
} catch (Exception e) {
// TODO: handle exception
} finally {
try {
request.close();
} catch (Exception e2) {
// TODO: handle exception
}
}
});
}
}

启动三个客户端,分别输入123、456、789

不必再等待其他客户端

三、服务端与浏览器交互

上面简单的C/S程序不能满足现实场景,下面进行服务端与浏览器的交互。

我们在浏览器,输入服务端的地址127.0.0.1:9093

我们发现服务端收到上面的信息,但是浏览器中没有访问到任何内容

原因:浏览器与服务端的交互使用的是HTTP协议

必须满足一定的格式:

1.请求数据包

上图没有第三部分和第四部分

2.响应数据包

响应状态码

因此,我们按照HTTP协议的格式,在服务端编写响应浏览器的内容,浏览器就可以收到我们的响应信息

public class BIOServer {

    private static ExecutorService threadPool = Executors.newCachedThreadPool();

    public static void main(String[] args) throws IOException {
ServerSocket serverSocket = new ServerSocket(9096);
System.out.println("tomcat 服务器启动成功");
while (!serverSocket.isClosed()) {
Socket request = serverSocket.accept();
System.out.println("收到新连接:" + request.toString());
threadPool.execute(() -> {
try {
InputStream inputStream = request.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8"));
String msg;
while ((msg = reader.readLine()) != null) {
if (msg.length() == 0) {
break;
}
System.out.println(msg);
}
System.out.println("收到数据,来自:" + request.toString());
// 响应结果
OutputStream outputStream = request.getOutputStream();
outputStream.write("HTTP/1.1 200 ok\r\n".getBytes());
outputStream.write("Content-Length: 40\r\n\r\n".getBytes());
outputStream.write("<button type=\\\"button\\\">Click Me!</button>".getBytes());
} catch (Exception e) {
// TODO: handle exception
} finally {
try {
request.close();
} catch (Exception e) {
// TODO: handle exception
}
}
}); }
serverSocket.close();
}
}

浏览器访问服务器

概念总结:

BIO:blocking IO,资源不可用时,IO请求一直阻塞,直到收到反馈结果(有数据或超时)。

NIO:non-blocking IO,资源不可用时,IO请求离开返回,返回数据标识资源不可用。

同步IO:synchronous IO,应用阻塞在发送或接收数据的状态,直到数据成功传输或返回失败。

异步IO:asynchronous IO,应用发送或接收数据后立刻返回,实际处理时异步执行的。

阻塞带来的问题:阻塞导致在处理网络I/O时,一个线程只能处理一个网络连接。

Java高并发网络编程(二)BIO的更多相关文章

  1. Java高并发网络编程(四)Netty

    在网络应用开发的过程中,直接使用JDK提供的NIO的API,比较繁琐,而且想要进行性能提升,还需要结合多线程技术. 由于网络编程本身的复杂性,以及JDK API开发的使用难度较高,所以在开源社区中,涌 ...

  2. Java高并发网络编程(一)

    一.OSI网络七层模型 因特网是一个极为复杂的网络,分层有助于我们对网络的理解 .分层也是一种标准,为了使不同厂商的计算机能够互相通信,以便在更大范围内建立计算机网络,有必要建立一个国际范围的网络体系 ...

  3. Java高并发网络编程(三)NIO

    从Java 1.4开始,Java提供了新的非阻塞IO操作API,用意是替代Java IO和Java Networking相关的API. NIO中有三个核心组件: Buffer缓冲区 Channel通道 ...

  4. Java高并发网络编程(五)Netty应用

    推送系统 一.系统设计 二.拆包和粘包 粘包.拆包表现形式 现在假设客户端向服务端连续发送了两个数据包,用packet1和packet2来表示,那么服务端收到的数据可以分为三种,现列举如下: 第一种情 ...

  5. 从菜鸟到大神:Java高并发核心编程(连载视频)

    任何事情是有套路的,学习是如此, Java的学习,更是如此. 本文,为大家揭示 Java学习的套路 背景 Java高并发.分布式的中间件非常多,网上也有很多组件的源码视频.原理视频,汗牛塞屋了. 作为 ...

  6. Linux下高并发网络编程

      Linux下高并发网络编程 1.修改用户进程可打开文件数限制 在Linux平台上,无论编写客户端程序还是服务端程序,在进行高并发TCP连接处理时, 最高的并发数量都要受到系统对用户单一进程同时可打 ...

  7. Linux高并发网络编程开发——10-Linux系统编程-第10天(网络编程基础-socket)

    在学习Linux高并发网络编程开发总结了笔记,并分享出来.有问题请及时联系博主:Alliswell_WP,转载请注明出处. 10-Linux系统编程-第10天(网络编程基础-socket) 在学习Li ...

  8. select poll epoll Linux高并发网络编程模型

    0 发展历程 同步阻塞迭代模型-->多进程并发模型-->多线程并发模型-->select-->poll-->epoll-->... 1 同步阻塞迭代模型 bind( ...

  9. java高并发实战(二)——线程(并行程序)基础

    转自:https://blog.csdn.net/gududedabai/article/details/80815666

随机推荐

  1. java 三元运算符

    /* 一元运算符:只需要一个数据就可以进行操作的运算符 如:取反! 自增++ 自减 -- 二元运算符:需要两个数据才可以进行操作的运算符 如:加法+ 赋值= 三元运算符: 需要三个数据才可以进行操作的 ...

  2. java面试题最容易犯错

    1. static 和 final 的用法 static 的作用从三个方面来谈,分别是静态变量.静态方法.静态类. 静态变量:声明为 static 的静态变量实质上就是全局变量,当声明一个对象时,并不 ...

  3. Motan框架初体验

    1.什么是Motan? Motan是一套基于java开发的RPC框架,除了常规的点对点调用外,motan还提供服务治理功能,包括服务节点的自动发现.摘除.高可用和负载均衡等.Motan具有良好的扩展性 ...

  4. UOJ197 线性规划

    传送门 由于这道题标程GG了所以必不可能AC嘛2333 单纯形法是一个很玄学的东西qwq 就是 非标准型 -> 标准型 -> 规范型 -> 松弛型 一个玄学操作——转轴操作(priv ...

  5. 阿里云公共DNS正式发布支持IPv6的版本

    在10月23日召开的GNTC 2019全球网络技术大会IPv6分论坛上,阿里云高级技术专家张先国宣布支持阿里公共DNS的IPv6版本正式发布,即阿里公共DNS在保持IPv4 稳定解析服务的基础上(An ...

  6. loadRunner之参数关联

    录制脚本,对用户名和密码进行参数化: Action() { web_url("WebTours", "URL=http://127.0.0.1:1080/WebTours ...

  7. Druid动态数据源配置

    上文已经讲了单个数据源的Druid的配置(http://www.cnblogs.com/nbfujx/p/7686634.html) Druid动态数据源配置 主要是继承AbstractRouting ...

  8. Android中通过进程注入技术修改广播接收器的优先级

    前言 这个周末又没有吊事,在家研究了如何通过进程的注入技术修改广播接收器的优先级,关于这个应用场景是很多的,而且也很重要,所以就很急的去fixed了. Android中的四大组件中有一个广播:Broa ...

  9. centos6编译安装php7

    https://www.cnblogs.com/wenwei-blog/p/6261637.html https://www.cnblogs.com/imzye/p/5109770.html cent ...

  10. 《ArcGIS Runtime SDK for .Net开发笔记》--介绍与环境搭建

    一. ArcGIS Runtime SDK for .NET介绍 ArcGIS Runtime SDK for .net是一款针对windows平台的开发包.能够在开发出在windows phone, ...