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 通信 是将这个阻 ...
随机推荐
- LinuxMint 编译 LittlevGL GUI
编译必须 安装arm-linux-gcc-4.4.3.tar.gz PC simulator You can try out the Littlev Graphics Library using on ...
- intellij idea关闭重复代码提醒
- 在windows下.NET CORE 与Consul简单使用
0)基本概念 consul常用于服务发现 [微服务] web_Api \ \ ->注册服务 raft选举le ...
- IFE春季班第一阶段任务(请仔细阅读)
第一阶段的主要目标是帮助大家 了解.认识.学习.掌握HTML及CSS.第一阶段任务从 3月14日 开始,持续到 4月3日.当然,您也可以在这个时间以后继续自行实践练习. 第一阶段任务一共有 12 个题 ...
- zzuli 1905 小火山的跳子游戏
Description 小火山和火山火山在一块玩跳子游戏.规则如下: 1:跳子的起始位置为0,棋盘大小从1到N 2:每次跳子跳k步. 例如当前位置为i, 那么下一步为i + k 3:跳 ...
- 【HIHOCODER 1044】题目1 : 状态压缩·一
描述 小Hi和小Ho在兑换到了喜欢的奖品之后,便继续起了他们的美国之行,思来想去,他们决定乘坐火车前往下一座城市--那座城市即将举行美食节! 但是不幸的是,小Hi和小Ho并没有能够买到很好的火车票-- ...
- HDU 1087 Super Jumping! Jumping! Jumping! (LIS的最大和)
题意: 给定n个数的序列, 找出最长上升子序列和. 分析: #include<cstdio> #include<iostream> #include<queue> ...
- 集合框架学习之List接口
Java语言的java.util包中提供了一些集合类,这些集合类又被称为容器.用来完善数组的不足之处.集合类与数组的不同之处是,数组的长度是固定的,集合的长度是可变的:数组用来存放基本类型的数据,集合 ...
- Triangular Pastures (二维01背包)
描述Like everyone, cows enjoy variety. Their current fancy is new shapes for pastures. The old rectang ...
- 【Ajax 2】封装Ajax的核心对象:XMLHttpRequest对象
导读:AJAX利用一个构建到所有现代浏览器内部的对象-XMLHttpRequest-来实现发送和接收HTTP请求与响应信息.那么,XMLHttpRequest对象是怎么创建和封装的呢? 一.简介 1. ...