[首页]
[文章]
[教程]
首页
Web开发
Windows开发
编程语言
数据库技术
移动平台
系统服务
微信
设计
布布扣
其他
数据分析
首页
>
编程语言
> 详细
java nio socket
时间:
2015-04-05 23:34:01
阅读:
341
评论:
0
收藏:
0
[点我收藏+]
jdk供的无阻塞I/O(NIO)有效解决了多线程服务器存在的线程开销问题,但在使用上略显得复杂一些。在NIO中使用多线程,主要目的已不是为了应对每个客户端请求而分配独立的服务线程,而是通过多线程充分使用用多个CPU的处理能力和处理中的等待时间,达到提高服务能力的目的。
这段时间在研究NIO,写篇博客来记住学过的东西。还是从最简单的Hello World开始,
client多线程请求server端,server接收client的名字,并返回Hello! +名字的字符格式给client。当然实际应用并不这么简单,实际可能是访问文件或者数据库获取信息返回给client。非阻塞的NIO有何神秘之处?代码:
1)server端代码
Java代码
/**
*
* @author Jeff
*
*/
public
class
HelloWorldServer {
static
int
BLOCK =
1024
;
static
String name =
""
;
protected
Selector selector;
protected
ByteBuffer clientBuffer = ByteBuffer.allocate(BLOCK);
protected
CharsetDecoder decoder;
static
CharsetEncoder encoder = Charset.forName(
"GB2312"
).newEncoder();
public
HelloWorldServer(
int
port)
throws
IOException {
selector =
this
.getSelector(port);
Charset charset = Charset.forName(
"GB2312"
);
decoder = charset.newDecoder();
}
// 获取Selector
protected
Selector getSelector(
int
port)
throws
IOException {
ServerSocketChannel server = ServerSocketChannel.open();
Selector sel = Selector.open();
server.socket().bind(
new
InetSocketAddress(port));
server.configureBlocking(
false
);
server.register(sel, SelectionKey.OP_ACCEPT);
return
sel;
}
// 监听端口
public
void
listen() {
try
{
for
(;;) {
selector.select();
Iterator iter = selector.selectedKeys().iterator();
while
(iter.hasNext()) {
SelectionKey key = (SelectionKey) iter.next();
iter.remove();
process(key);
}
}
}
catch
(IOException e) {
e.printStackTrace();
}
}
// 处理事件
protected
void
process(SelectionKey key)
throws
IOException {
if
(key.isAcceptable()) {
// 接收请求
ServerSocketChannel server = (ServerSocketChannel) key.channel();
SocketChannel channel = server.accept();
//设置非阻塞模式
channel.configureBlocking(
false
);
channel.register(selector, SelectionKey.OP_READ);
}
else
if
(key.isReadable()) {
// 读信息
SocketChannel channel = (SocketChannel) key.channel();
int
count = channel.read(clientBuffer);
if
(count >
0
) {
clientBuffer.flip();
CharBuffer charBuffer = decoder.decode(clientBuffer);
name = charBuffer.toString();
// System.out.println(name);
SelectionKey sKey = channel.register(selector,
SelectionKey.OP_WRITE);
sKey.attach(name);
}
else
{
channel.close();
}
clientBuffer.clear();
}
else
if
(key.isWritable()) {
// 写事件
SocketChannel channel = (SocketChannel) key.channel();
String name = (String) key.attachment();
ByteBuffer block = encoder.encode(CharBuffer
.wrap(
"Hello !"
+ name));
channel.write(block);
//channel.close();
}
}
public
static
void
main(String[] args) {
int
port =
8888
;
try
{
HelloWorldServer server =
new
HelloWorldServer(port);
System.out.println(
"listening on "
+ port);
server.listen();
}
catch
(IOException e) {
e.printStackTrace();
}
}
}
server主要是读取client发过来的信息,并返回一条信息
2)client端代码
Java代码
/**
*
* @author Jeff
*
*/
public
class
HelloWorldClient {
static
int
SIZE =
10
;
static
InetSocketAddress ip =
new
InetSocketAddress(
"localhost"
,
8888
);
static
CharsetEncoder encoder = Charset.forName(
"GB2312"
).newEncoder();
static
class
Message
implements
Runnable {
protected
String name;
String msg =
""
;
public
Message(String index) {
this
.name = index;
}
public
void
run() {
try
{
long
start = System.currentTimeMillis();
//打开Socket通道
SocketChannel client = SocketChannel.open();
//设置为非阻塞模式
client.configureBlocking(
false
);
//打开选择器
Selector selector = Selector.open();
//注册连接服务端socket动作
client.register(selector, SelectionKey.OP_CONNECT);
//连接
client.connect(ip);
//分配内存
ByteBuffer buffer = ByteBuffer.allocate(
8
*
1024
);
int
total =
0
;
_FOR:
for
(;;) {
selector.select();
Iterator iter = selector.selectedKeys().iterator();
while
(iter.hasNext()) {
SelectionKey key = (SelectionKey) iter.next();
iter.remove();
if
(key.isConnectable()) {
SocketChannel channel = (SocketChannel) key
.channel();
if
(channel.isConnectionPending())
channel.finishConnect();
channel
.write(encoder
.encode(CharBuffer.wrap(name)));
channel.register(selector, SelectionKey.OP_READ);
}
else
if
(key.isReadable()) {
SocketChannel channel = (SocketChannel) key
.channel();
int
count = channel.read(buffer);
if
(count >
0
) {
total += count;
buffer.flip();
while
(buffer.remaining() >
0
) {
byte
b = buffer.get();
msg += (
char
) b;
}
buffer.clear();
}
else
{
client.close();
break
_FOR;
}
}
}
}
double
last = (System.currentTimeMillis() - start) *
1.0
/
1000
;
System.out.println(msg +
"used time :"
+ last +
"s."
);
msg =
""
;
}
catch
(IOException e) {
e.printStackTrace();
}
}
}
public
static
void
main(String[] args)
throws
IOException {
String names[] =
new
String[SIZE];
for
(
int
index =
0
; index < SIZE; index++) {
names[index] =
"jeff["
+ index +
"]"
;
new
Thread(
new
Message(names[index])).start();
}
}
}
java nio socket
原文:http://blog.csdn.net/zhangping871/article/details/44892779
踩
(
0
)
赞
(
0
)
举报
评论
一句话评论(
0
)
登录后才能评论!
分享档案
更多>
2021年09月23日 (328)
2021年09月24日 (313)
2021年09月17日 (191)
2021年09月15日 (369)
2021年09月16日 (411)
2021年09月13日 (439)
2021年09月11日 (398)
2021年09月12日 (393)
2021年09月10日 (160)
2021年09月08日 (222)
最新文章
更多>
2021/09/28 scripts
2022-05-27
vue自定义全局指令v-emoji限制input输入表情和特殊字符
2022-05-27
9.26学习总结
2022-05-27
vim操作
2022-05-27
深入理解计算机基础 第三章
2022-05-27
C++ string 作为形参与引用传递(转)
2022-05-27
python 加解密
2022-05-27
JavaScript-对象数组里根据id获取name,对象可能有children属性
2022-05-27
SQL语句——保持现有内容在后面增加内容
2022-05-27
virsh命令文档
2022-05-27
教程昨日排行
更多>
1.
list.reverse()
2.
Django Admin 管理工具
3.
AppML 案例模型
4.
HTML 标签列表(功能排序)
5.
HTML 颜色名
6.
HTML 语言代码
7.
jQuery 事件
8.
jEasyUI 创建分割按钮
9.
jEasyUI 创建复杂布局
10.
jEasyUI 创建简单窗口
友情链接
汇智网
PHP教程
插件网
关于我们
-
联系我们
-
留言反馈
- 联系我们:wmxa8@hotmail.com
© 2014
bubuko.com
版权所有
打开技术之扣,分享程序人生!