java nio实现非阻塞Socket通信实例
服务器
- package com.java.xiong.Net17;
- import java.io.IOException;
- import java.net.InetSocketAddress;
- import java.nio.ByteBuffer;
- import java.nio.channels.Channel;
- import java.nio.channels.SelectionKey;
- import java.nio.channels.Selector;
- import java.nio.channels.ServerSocketChannel;
- import java.nio.channels.SocketChannel;
- import java.nio.charset.Charset;
- public class NServer {
- // 用于检测所有的Channel状态的selector
- private Selector selector = null;
- static final int PORT = 30000;
- // 定义实现编码、解码的字符串集对象
- private Charset charse = Charset.forName("GBK");
- public void init() throws IOException {
- selector = Selector.open();
- // 通过open方法来打开一个未绑定的ServerSocketChannel是咧
- ServerSocketChannel server = ServerSocketChannel.open();
- InetSocketAddress isa = new InetSocketAddress("127.0.0.1", PORT);
- // 将该ServerSocketChannel绑定到指定的IP地址
- server.bind(isa);
- // 设置serverSocket已非阻塞方式工作
- server.configureBlocking(false);
- // 将server注册到指定的selector对象
- server.register(selector, SelectionKey.OP_ACCEPT);
- while (selector.select() > 0) {
- // 一次处理selector上的每个选择的SelectionKey
- for (SelectionKey sk : selector.selectedKeys()) {
- // 从selector上已选择的Kye集中删除正在处理的SelectionKey
- selector.selectedKeys().remove(sk);
- // 如果sk对应的Channel包含客户端的连接请求
- if (sk.isAcceptable()) {
- // 调用accept方法接收连接,产生服务器段的SocketChennal
- SocketChannel sc = server.accept();
- // 设置采用非阻塞模式
- sc.configureBlocking(false);
- // 将该SocketChannel注册到selector
- sc.register(selector, SelectionKey.OP_READ);
- }
- // 如果sk对应的Channel有数据需要读取
- if (sk.isReadable()) {
- // 获取该SelectionKey对银行的Channel,该Channel中有刻度的数据
- SocketChannel sc = (SocketChannel) sk.channel();
- // 定义备注执行读取数据源的ByteBuffer
- ByteBuffer buff = ByteBuffer.allocate(1024);
- String content = "";
- // 开始读取数据
- try {
- while (sc.read(buff) > 0) {
- buff.flip();
- content += charse.decode(buff);
- }
- System.out.println("读取的数据:" + content);
- // 将sk对应的Channel设置成准备下一次读取
- sk.interestOps(SelectionKey.OP_READ);
- }
- // 如果捕获到该sk对银行的Channel出现了异常,表明
- // Channel对应的Client出现了问题,所以从Selector中取消
- catch (IOException io) {
- // 从Selector中删除指定的SelectionKey
- sk.cancel();
- if (sk.channel() != null) {
- sk.channel().close();
- }
- }
- // 如果content的长度大于0,则连天信息不为空
- if (content.length() > 0) {
- // 遍历selector里注册的所有SelectionKey
- for (SelectionKey key : selector.keys()) {
- // 获取该key对应的Channel
- Channel targerChannel = key.channel();
- // 如果该Channel是SocketChannel对象
- if (targerChannel instanceof SocketChannel) {
- // 将读取到的内容写入该Channel中
- SocketChannel dest = (SocketChannel) targerChannel;
- dest.write(charse.encode(content));
- }
- }
- }
- }
- }
- }
- }
- public static void main(String [] args) throws IOException{
- new NServer().init();
- }
- }
客户端
- package com.java.xiong.Net17;
- import java.io.IOException;
- import java.net.InetSocketAddress;
- import java.nio.ByteBuffer;
- import java.nio.channels.SelectionKey;
- import java.nio.channels.Selector;
- import java.nio.channels.SocketChannel;
- import java.nio.charset.Charset;
- import java.util.Scanner;
- public class NClient {
- //定义检测Sockethannel的Selector对象
- private Selector selector=null;
- static final int PORT=30000;
- //定义处理编码的字符集
- private Charset charset=Charset.forName("GBK");
- //客户端SocketChannel
- private SocketChannel sc=null;
- public void init() throws IOException{
- selector=Selector.open();
- InetSocketAddress isa=new InetSocketAddress("127.0.0.1", PORT);
- //调用open的静态方法创建连接指定的主机的SocketChannel
- sc=SocketChannel.open(isa);
- //设置该sc已非阻塞的方式工作
- sc.configureBlocking(false);
- //将SocketChannel对象注册到指定的Selector
- sc.register(selector, SelectionKey.OP_READ);
- //启动读取服务器数据端的线程
- new ClientThread().start();
- //创建键盘输入流
- Scanner scan=new Scanner(System.in);
- while(scan.hasNextLine()){
- //读取键盘的输入
- String line=scan.nextLine();
- //将键盘的内容输出到SocketChanenel中
- sc.write(charset.encode(line));
- }
- }
- //定义读取服务器端的数据的线程
- private class ClientThread extends Thread{
- @Override
- public void run() {
- try{
- while(selector.select()>0){
- //遍历每个有可能的IO操作的Channel对银行的SelectionKey
- for(SelectionKey sk:selector.selectedKeys()){
- //删除正在处理的SelectionKey
- selector.selectedKeys().remove(sk);
- //如果该SelectionKey对应的Channel中有可读的数据
- if(sk.isReadable()){
- //使用NIO读取Channel中的数据
- SocketChannel sc=(SocketChannel)sk.channel();
- String content="";
- ByteBuffer bff=ByteBuffer.allocate(1024);
- while(sc.read(bff)>0){
- sc.read(bff);
- bff.flip();
- content+=charset.decode(bff);
- }
- //打印读取的内容
- System.out.println("聊天信息:"+content);
- sk.interestOps(SelectionKey.OP_READ);
- }
- }
- }
- }catch(IOException io){
- io.printStackTrace();
- }
- }
- }
- public static void main(String [] args) throws IOException{
- new NClient().init();
- }
- }
java nio实现非阻塞Socket通信实例的更多相关文章
- JAVA基础知识之网络编程——-基于NIO的非阻塞Socket通信
阻塞IO与非阻塞IO 通常情况下的Socket都是阻塞式的, 程序的输入输出都会让当前线程进入阻塞状态, 因此服务器需要为每一个客户端都创建一个线程. 从JAVA1.4开始引入了NIO API, NI ...
- Java NIO 同步非阻塞
同步非阻塞IO (NIO) NIO是基于事件驱动思想的,实现上通常采用Reactor(http://en.wikipedia.org/wiki/Reactor_pattern)模式,从程序角度而言,当 ...
- JAVA NIO使用非阻塞模式实现高并发服务器
参考:http://blog.csdn.net/zmx729618/article/details/51860699 https://zhuanlan.zhihu.com/p/23488863 ht ...
- 【转载】高性能IO设计 & Java NIO & 同步/异步 阻塞/非阻塞 Reactor/Proactor
开始准备看Java NIO的,这篇文章:http://xly1981.iteye.com/blog/1735862 里面提到了这篇文章 http://xmuzyq.iteye.com/blog/783 ...
- java的高并发IO原理,阻塞BIO同步非阻塞NIO,异步非阻塞AIO
原文地址: IO读写的基础原理 大家知道,用户程序进行IO的读写,依赖于底层的IO读写,基本上会用到底层的read&write两大系统调用.在不同的操作系统中,IO读写的系统调用的名称可能不完 ...
- Flex通信-与Java实现Socket通信实例
Flex通信-与Java实现Socket通信实例 转自:http://blessht.iteye.com/blog/1136888 博客分类: Flex 环境准备 [服务器端] JDK1.6,“ja ...
- Java Socket 通信实例 - 转载
基于Tcp协议的简单Socket通信实例(JAVA) 好久没写博客了,前段时间忙于做项目,耽误了些时间,今天开始继续写起~ 今天来讲下关于Socket通信的简单应用,关于什么是Socket以及一些 ...
- java并发之非阻塞算法介绍
在并发上下文中,非阻塞算法是一种允许线程在阻塞其他线程的情况下访问共享状态的算法.在绝大多数项目中,在算法中如果一个线程的挂起没有导致其它的线程挂起,我们就说这个算法是非阻塞的. 为了更好的理解阻塞算 ...
- 4.NIO的非阻塞式网络通信
/*阻塞 和 非阻塞 是对于 网络通信而言的*/ /*原先IO通信在进行一些读写操作 或者 等待 客户机连接 这种,是阻塞的,必须要等到有数据被处理,当前线程才被释放*/ /*NIO 通信 是将这个阻 ...
随机推荐
- css3 calc()属性介绍以及自适应布局使用方法
前端知识 Calc()介绍 calc的英文是calculate的缩写,中文为计算的意思,是css3的一个新增的功能,用来只当元素的长度.比如说:你可以用calc()给元素margin.padding. ...
- iview table 已选项的数据 this.$refs.tables.$refs.tablesMain.getSelection()
iview table 已选项的数据 this.$refs.tables.$refs.tablesMain.getSelection() 由于我这里table组件是套了两层组件 所以是进入了两个层次拿 ...
- 【整理】 vue-cli 打包后显示favicon.ico小图标
vue-cli 打包后显示favicon.ico小图标 https://www.cnblogs.com/mmzuo-798/p/9285013.html
- Web前端基础怎么学? JavaScript、html、css知识架构图
以前开发者只要掌握 HTML.CSS.JavaScript 三驾马车就能胜任一份前端的工作了.而现在除了普通的编码以外,还要考虑如何性能优化,如何跨端.跨平台实现功能,尤其是 AI.5G 技术的来临, ...
- 全志T8智能汽车方案芯片参数介绍
T8处理器代表了Allwinner在智能汽车市场上的最新成就.T8适用于需要三维图形.高级视频处理.精密相机.多种连接选项和高水平系统集成的应用程序.它将把先进的消费电子体验带入未来的汽车,实现高性能 ...
- Spring框架 (log4j :WARN No appenders could be found for logger log4j:WARN Please initialize the log4j system properly.)问题解决
Spring框架需要的jar包 1.Spring压缩包中的四个核心JAR包 beans .context.core 和expression 下载地址: https://pan.baidu.com/s/ ...
- linux批量检测服务器能否ping通和硬盘容量状态并抛出报警的一个脚本-附详细解释
有一些linux基础,最近刚开始学shell,参考了阿良老师的一个监测服务器硬盘状态的脚本,自己进行了一些扩展,今天比较晚了,后边会把注释放上来,感觉脚本还很不完善,希望大家一起探讨一下,共同学习 2 ...
- 测试linux服务器带宽
测试准备 1. 计划考量参数 TCP上传数据带宽 TCP下载数据带宽 UDP上传带宽 UDP下载带宽 多并发支持 稳定性 Tcp通讯网络延迟(小包:32.中包1k.大包1M) UDP通讯网络延迟(小包 ...
- 搭建Mysql主从复制
mysql 主从复制流程图 Server version: 10.0.24-MariaDB-7 Ubuntu 16.04 Master 记录二进制文件 导出数据并记录二进制位置 导入数据,设置二进制位 ...
- day17-python之文件操作
1.内置函数 #!/usr/bin/env python # -*- coding:utf-8 -*- # print(abs(-1)) # print(abs(1)) # # print(all([ ...