社区
Java SE
帖子详情
java中非阻塞的socket
glacierful
2012-06-04 02:40:15
java中Socket类默认是阻塞的。如果想使用非阻塞的socket该如何做呢?
...全文
215
8
打赏
收藏
java中非阻塞的socket
java中Socket类默认是阻塞的。如果想使用非阻塞的socket该如何做呢?
复制链接
扫一扫
分享
转发到动态
举报
写回复
配置赞助广告
用AI写文章
8 条
回复
切换为时间正序
请发表友善的回复…
发表回复
打赏红包
龙四
2012-06-06
打赏
举报
回复
用jetty或者mina,不要自己去写NIO,太恶心太多东西要考虑了
qunhao
2012-06-06
打赏
举报
回复
楼主去看下Java NIO就知道了。
beiouwolf
2012-06-05
打赏
举报
回复
我觉得LZ应该是想问NIO吧...
可以学学MINA2
Seanake
2012-06-05
打赏
举报
回复
使用NIO或java 7 AIO实现非阻塞Socket通信
这边我只学个皮毛给个《疯狂java讲义》例子楼主可以参考相关资料
import java.util.*;
import java.net.*;
import java.io.*;
import java.nio.*;
import java.nio.channels.*;
import java.nio.charset.*;
public class NClient
{
// 定义检测SocketChannel的Selector对象
private Selector selector = null;
static final int PORT = 30000;
// 定义处理编码和解码的字符集
private Charset charset = Charset.forName("UTF-8");
// 客户端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();
// 将键盘输入的内容输出到SocketChannel中
sc.write(charset.encode(line));
}
}
// 定义读取服务器数据的线程
private class ClientThread extends Thread
{
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();
ByteBuffer buff = ByteBuffer.allocate(1024);
String content = "";
while(sc.read(buff) > 0)
{
sc.read(buff);
buff.flip();
content += charset.decode(buff);
}
// 打印输出读取的内容
System.out.println("聊天信息:" + content);
// 为下一次读取作准备
sk.interestOps(SelectionKey.OP_READ);
}
}
}
}
catch (IOException ex)
{
ex.printStackTrace();
}
}
}
public static void main(String[] args)
throws IOException
{
new NClient().init();
}
}
import java.net.*;
import java.io.*;
import java.nio.*;
import java.nio.channels.*;
import java.nio.charset.*;
public class NServer
{
// 用于检测所有Channel状态的Selector
private Selector selector = null;
static final int PORT = 30000;
// 定义实现编码、解码的字符集对象
private Charset charset = Charset.forName("UTF-8");
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上的已选择Key集中删除正在处理的SelectionKey
selector.selectedKeys().remove(sk); //①
// 如果sk对应的Channel包含客户端的连接请求
if (sk.isAcceptable()) //②
{
// 调用accept方法接受连接,产生服务器端的SocketChannel
SocketChannel sc = server.accept();
// 设置采用非阻塞模式
sc.configureBlocking(false);
// 将该SocketChannel也注册到selector
sc.register(selector, SelectionKey.OP_READ);
// 将sk对应的Channel设置成准备接受其他请求
sk.interestOps(SelectionKey.OP_ACCEPT);
}
// 如果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 += charset.decode(buff);
}
// 打印从该sk对应的Channel里读取到的数据
System.out.println("读取的数据:" + content);
// 将sk对应的Channel设置成准备下一次读取
sk.interestOps(SelectionKey.OP_READ);
}
// 如果捕捉到该sk对应的Channel出现了异常,即表明该Channel
// 对应的Client出现了问题,所以从Selector中取消sk的注册
catch (IOException ex)
{
// 从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 targetChannel = key.channel();
// 如果该channel是SocketChannel对象
if (targetChannel instanceof SocketChannel)
{
// 将读到的内容写入该Channel中
SocketChannel dest = (SocketChannel)targetChannel;
dest.write(charset.encode(content));
}
}
}
}
}
}
}
public static void main(String[] args)
throws IOException
{
new NServer().init();
}
}
24K純帥
2012-06-04
打赏
举报
回复
用SocketChannel
http://jasonshieh.iteye.com/blog/927376
myisfei
2012-06-04
打赏
举报
回复
可用多线程做到类似的的情况
李狗蛋52635
2012-06-04
打赏
举报
回复
用ServerSocketChannel,设置为非阻塞方式,网上有相关的例子。
socket
java
源码-demo-
socket
s-io-nio-nio2:“
Java
套接字I/O:阻塞,
非阻塞
和异步”文章和源代码
socket
java
原始码
Java
套接字I / O:阻塞,
非阻塞
和异步 介绍 在描述I / O时,术语“
非阻塞
”和“异步”通常可以互换使用,但是它们之间存在显着差异。 本文描述了
Java
中
非阻塞
和异步套接字I / O操作之间的理论和实践差异。 套接字是通过TCP和UDP协议执行双向通信的端点。
Java
套接字API是用于操作系统相应功能的适配器。 符合POSIX的操作系统(Unix,Linux,Mac OS X,BSD,Solaris,AIX等)
中
的套接字通信由Berkeley套接字执行。 Windows
中
的套接字通信由Winsock执行, Winsock也是基于Berkeley套接字的,具有符合Windows编程模型的附加功能。 POSIX定义 本文使用了POSIX规范
中
的简化定义。 阻塞的线程-在可以继续执行之前正在等待某种条件的线程。 阻塞-套接字的属性,使套接字对其进行调用,以等待所请求的操作在返回之前执行。
非阻塞
-套接字的一种属性,当检测到所请求的动作没有未知的延迟就无法完成时,套接字的调用将立即返回。 同步I / O操作-导致请求线程被阻塞直到该I / O操作完成的I
用
Java
实现
非阻塞
通信 和阻塞通讯
用Server
Socket
和
Socket
来编写服务器程序和客户程序,是
Java
网络编程的最基本的方式。这些服务器程序或客户程序在运行过程
中
常常会阻塞。例如当一个线程执行Server
Socket
的accept()方法时,假如没有客户连接,该线程就会一直等到有了客户连接才从accept()方法返回。再例如当线程执行
Socket
的read()方法时,如果输入流
中
没有数据,该线程就会一直等到读入了足够的数
JQuery入门与实战
jQuery 是一个
Java
Script 库。jQuery 极大地简化了
Java
Script 编程。jQuery 超级容学。jQuery侧重点就是选择器和基本的DOM操作,还有一些动画操作,但是js
中
非DOM操作的部分基本没有涉及。了解JQuery魅力,利用JQ在项目或者工作
中
来实现网页的效果特效,从而让网页更加大气。
android
中
非阻塞
socket
通信
1、什么是同步与异步,阻塞与
非阻塞
首先我们要明白搞明白:同步就等于阻塞?异步就等于
非阻塞
?这是不对的,同步不等于阻 塞,而异步也不等于
非阻塞
。 1)那什么是同步编程? 什么是同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回。根据这个定义,android
中
绝大多数函数都是同步调用。但是一般而言,我们在谈论同步、异步的时候,特指那些需要其他部件协作或者需要一定
android
中
非堵塞
socket
通信
1、什么是同步与异步。堵塞与非堵塞 首先我们要明确搞明确:同步就等于堵塞?异步就等于非堵塞?这是不正确的,同步不等于阻 塞。而异步也不等于非堵塞。 1)那什么是同步编程? 什么是同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回。依据这个定义,android
中
绝大多数函数都是同步调用。可是一般而言,我们在谈论...
Java SE
62,615
社区成员
307,328
社区内容
发帖
与我相关
我的任务
Java SE
Java 2 Standard Edition
复制链接
扫一扫
分享
社区描述
Java 2 Standard Edition
社区管理员
加入社区
获取链接或二维码
近7日
近30日
至今
加载中
查看更多榜单
社区公告
暂无公告
试试用AI创作助手写篇文章吧
+ 用AI写文章