代码语言
.
CSharp
.
JS
Java
Asp.Net
C
MSSQL
PHP
Css
PLSQL
Python
Shell
EBS
ASP
Perl
ObjC
VB.Net
VBS
MYSQL
GO
Delphi
AS
DB2
Domino
Rails
ActionScript
Scala
代码分类
文件
系统
字符串
数据库
网络相关
图形/GUI
多媒体
算法
游戏
Jquery
Extjs
Android
HTML5
菜单
网页交互
WinForm
控件
企业应用
安全与加密
脚本/批处理
开放平台
其它
【
Java
】
基于NIO的多线程服务器端
作者:
Burning_BL
/ 发布于
2013/7/1
/
490
在连接数很大的情况下,使用Selector能减少并发线程数,减少上下文切换的消耗,非阻塞IO使得具备流程的读写。
package main; import java.io.IOException; import java.net.InetSocketAddress; import java.net.ServerSocket; import java.nio.ByteBuffer; 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; import java.util.Iterator; import java.util.Set; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class Server { private final int PORT = 1234; private ExecutorService pool; private ServerSocketChannel ssc; private Selector selector; private static Charset charset = Charset.forName("utf-8"); private int n; public static void main(String[] args) throws IOException{ Server server = new Server(); server.doService(); } public Server() throws IOException{ pool = Executors.newFixedThreadPool(5); ssc = ServerSocketChannel.open(); ssc.configureBlocking(false); ServerSocket ss = ssc.socket(); ss.bind(new InetSocketAddress(PORT)); selector = Selector.open(); ssc.register(selector,SelectionKey.OP_ACCEPT); System.out.println("Server started..."); } public void doService(){ while(true){ try{ n = selector.select(); }catch (IOException e) { throw new RuntimeException("Selector.select()异常!"); } if(n==0) continue; Set<SelectionKey> keys = selector.selectedKeys(); Iterator<SelectionKey> iter = keys.iterator(); while(iter.hasNext()){ SelectionKey key = iter.next(); iter.remove(); if(key.isAcceptable()){ SocketChannel sc = null; try{ sc = ((ServerSocketChannel)key.channel()).accept(); sc.configureBlocking(false); System.out.println("客户端:"+sc.socket().getInetAddress().getHostAddress()+" 已连接"); SelectionKey k = sc.register(selector, SelectionKey.OP_READ); ByteBuffer buf = ByteBuffer.allocate(1024); k.attach(buf); }catch (Exception e) { try{ sc.close(); }catch (Exception ex) { } } } else if(key.isReadable()){ key.interestOps(key.interestOps()&(~SelectionKey.OP_READ)); pool.execute(new Worker(key)); } } } } public static class Worker implements Runnable{ private SelectionKey key; public Worker(SelectionKey key){ this.key = key; } @Override public void run() { SocketChannel sc = (SocketChannel)key.channel(); ByteBuffer buf = (ByteBuffer)key.attachment(); buf.clear(); int len = 0; try{ while((len=sc.read(buf))>0){//非阻塞,立刻读取缓冲区可用字节 buf.flip(); System.out.println("客户端:"+charset.decode(buf).toString()); buf.clear(); } if(len==-1){ System.out.println("客户端断开。。。"); sc.close(); } //没有可用字节,继续监听OP_READ key.interestOps(key.interestOps()|SelectionKey.OP_READ); key.selector().wakeup(); }catch (Exception e) { try { sc.close(); } catch (IOException e1) { } } } } }
试试其它关键字
多线程服务器
同语言下
.
List 切割成几份 工具类
.
一行一行读取txt的内容
.
Java PDF转换成图片并输出给前台展示
.
java 多线程框架
.
double类型如果小数点后为零则显示整数否则保留两位小
.
将图片转换为Base64字符串公共类抽取
.
sqlParser 处理SQL(增删改查) 替换schema 用于多租户
.
JAVA 月份中的第几周处理 1-7属于第一周 依次类推 29-
.
java计算两个经纬度之间的距离
.
输入时间参数计算年龄
可能有用的
.
C#实现的html内容截取
.
List 切割成几份 工具类
.
SQL查询 多列合并成一行用逗号隔开
.
一行一行读取txt的内容
.
C#动态修改文件夹名称(FSO实现,不移动文件)
.
c# 移动文件或文件夹
.
c#图片添加水印
.
Java PDF转换成图片并输出给前台展示
.
网站后台修改图片尺寸代码
.
处理大图片在缩略图时的展示
Burning_BL
贡献的其它代码
(
1
)
.
基于NIO的多线程服务器端
Copyright © 2004 - 2024 dezai.cn. All Rights Reserved
站长博客
粤ICP备13059550号-3