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 通信 是将这个阻 ...
随机推荐
- 爬虫学习之第一次获取网页内容及BeautifulSoup处理
from urllib.request import urlopen from urllib.request import HTTPError from bs4 import BeautifulSou ...
- java文件上传,自动判断文件类型
public enum FileType { /** * JEPG. */ JPEG("FFD8FF"), /** * PNG. */ PNG("89504E47&quo ...
- 高德地图api之location定位
关于定位,分为GPS定位和网络定位.本文将详细描述的浏览器定位,属于网络定位.这是一种通过使用高德JS-API来实现位置定位.城市定位的方法,包含了IP定位,检索等多种网络定位方式.如果您的手机支持G ...
- C# 如何正确删除List中的item
参考博客 https://blog.csdn.net/Le_Sam/article/details/75633737 https://www.cnblogs.com/hedianzhan/p/9130 ...
- 基于jquery的自定义显示消息数量
根据需求简单的实现一个小功能控件,暂时不支持扩展 $("xxxxxxx").iconCountPlugin(options, start, isOffset) {//三个参数,自定 ...
- TCP socket编程记录(C语言)
最近使用到TCP的socket编程,记录一下思路. TCP Socket 服务器编程思路 0.初始化套接字库 1.创建套接字----------2个(服务器本身需要一个套接字,客户端要返回一个套接字) ...
- Go:单元测试
测试用的文件名必须以 _test.go 结尾: 测试用的函数名必须以 Test 开头,一般来说:Test+被测试的函数名(第一个字母必须大写): func TestXx(t *testing.T) { ...
- Servlet+JSP教程之:第一个Web程序
我们知道当浏览器发送请求给服务器后,服务器会调用并执行对应的逻辑代码进行请求处理.逻辑代 码是由程序员自己编写然后放进服务器进行运行,其实就是Servlet程序. 第一个Web程序: 开发工具: My ...
- CSS3 pointer-events:none 让你摆脱事件的烦恼
以前没遇到这个属性,在一个偶然的博文下发现该属性真的好用,你是否遇到过写鼠标移入显示文本的效果时,鼠标在元素内的每一次移动都会造成要显示文本的闪烁或是突然的消失?只要在被控制的元素中加上这个属性完美解 ...
- Python浅拷贝copy()与深拷贝deepcopy()区别
其实呢,浅拷贝copy()与深拷贝deepcopy()之间的区分必须要涉及到python对于数据的存储方式. 首先直接上结论: —–我们寻常意义的复制就是深复制,即将被复制对象完全再复制一遍作为独立的 ...