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 通信 是将这个阻 ...
随机推荐
- 云服务器linux使用之开发环境搭建(一)
Host key verification failed. @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @ WARNING: ...
- scanf_s读取键盘输入字符串失败
#include<stdio.h> int main() { ]; ]; printf("Input string:\n"); scanf_s("%s&quo ...
- 【thinkPHP5实现文件上传】
上传文件 ThinkPHP5.0对文件上传的支持更加简单. 内置的上传只是上传到本地服务器,上传到远程或者第三方平台的话需要自己扩展. 假设表单代码如下: <form action=" ...
- phpStrom+xdebug调试php
1>xdebug下载 1.1>xdebug官网可以根据phpinfo()源代码来提供对应版本的xdebug,地址:https://xdebug.org/wizard.php 如下截图 1. ...
- 【java面试宝典】一年工作经验者适用
时光转瞬即逝,一年时间已经过去,自己准备跳槽了,请了几天假,面试了几家公司之后,拿到了一份offer,有了一点儿小小的心得,记录下来分享给搞Java的朋友,面试主要是几个部分内容. 很多公司会要求笔试 ...
- hdu1394(Minimum Inversion Number)线段树
明知道是线段树,却写不出来,搞了半天,戳,没办法,最后还是得去看题解(有待于提高啊啊),想做道题还是难啊. 还是先贴题吧 HDU-1394 Minimum Inversion Number Time ...
- python 连接docker
发现15年写过的一小段,先mark一下,后续玩儿起来 import docker c = docker.Client(base_url='unix://var/run/docker.sock',ver ...
- UVALive - 6267 Beer Pressure
题意: 给定n个酒吧, 然后有k个学生投票今晚去哪个酒吧, 然后会有a个(a<=k)学生先投票了, 先投的票会影响后面的人投票的概率, 求每个酒吧今晚去的概率. 分析: 我们可以从最初的状态开始 ...
- Ext.js双击事件
/** * 联系人列表panel */ Ext.define('Op.OpBill.OpBillCustLinkGridPanel', { extend: 'Ext.grid.Panel', id: ...
- 使用hadoop mapreduce分析mongodb数据
使用hadoop mapreduce分析mongodb数据 (现在很多互联网爬虫将数据存入mongdb中,所以研究了一下,写此文档) 版权声明:本文为yunshuxueyuan原创文章.如需转载请标明 ...