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 通信 是将这个阻 ...
随机推荐
- MySql数据库--持续记录ing
1 基本,引擎,数据类型,运算1.1 基本操作启动:net start mysql停止:net stop mysql连接: mysql –uroot -h127.0.0.1 -proot断开连接:qu ...
- uva1380 A Scheduling Problem
按紫书来注意这道题的题目给了很大的方便,就相当于验证k是不是答案,不是的话就是k+1 #include<iostream> #include<string> #include& ...
- wkhtmltopdf导出html到pdf
1.使用背景 最近公司需要把html页面的内容生成pdf并下载,试过很多方法都没有满意的效果,后来找到wkhtmltopdf这款软件,终于解决了这个问题. wkhtmltopdf是exe文件, ...
- C++ 类中的static成员的初始化和特点
C++ 类中的static成员的初始化和特点 #include <iostream> using namespace std; class Test { public: Test() : ...
- C++静态全局变量和全局变量的区别
静态全局变量 非静态全局变量 存储方式 静态存储 静态存储 作用域 定义该变量的源文件内 所有源文件 解释: 共同点:全局变量(外部变量)的说明之前再冠以static 就构 成了静态的全局变量.全 ...
- es6(三set和map数据结构)
es6中提供了一个新的数据结构Set,他有点类似数组,但和数组不同的是,在里面你如果写入重复的值的话,他不会显示重复值. const s =new Set(); [2,3,4,5,6,6,6,7,8, ...
- jquery-closest
1.closest() 本例演示如何通过 closest() 完成事件委托.当被最接近的列表元素或其子后代元素被点击时,会切换黄色背景: $( document ).bind("click& ...
- 退役选手ZlycerQan的强大的的高精度
#include <cstdio> #include <iostream> #include <vector> #include <iomanip> # ...
- poj-1163 动态规划
这道题目并不能直接使用递归,因为 7(1) 7(1) 7(1) 7(1) 7(2) 7(1) 7(1) 7(3) 7(3) ...
- 常见的User-Agent
User_Agent = ["Mozilla/5.0 (iPod; U; CPU iPhone OS 4_3_2 like Mac OS X; zh-cn) AppleWebKit/533. ...