public class JDK_OIOServer{
public void start(int port) throws IOException {
// 初始化ServerSocket 并且绑定端口
final ServerSocket serverSocket = new ServerSocket(port);
for(;;){
//获取远程客户端的链接
final Socket clientSocket = serverSocket.accept();
//启用线程
new Thread(new Runnable(){
@override
public void run(){
try{
//获得写输出流
OutputStream out= client.getOutputStream();
//写入字符并设置字符集
out.wirte("Hello",getBytes(Charset.forName("UTF-8")));
//冲刷数据
out.flush();
}
catch (IOException e){
e.printStackTrace();
}
finally{
//关闭客户端链接
try{
clientSocket.close();
}catch(IOException e){
e.printStackTrace();
}
}
}
}).start();
}
}
}
public class JDK_NIOServer{
public void start(int port){
//得到serverChannel对象
ServerSocketChannel serverChannel = ServerSocketChannel.open();
//配置为非阻塞状态
serverChannel.configureBlocking(false);
//得到选择器
Selector selector = Selector.open();
//得到serverChannel中的serverSocket,网络链接真实发生的地方就是serverSocket
ServerSocket serverSocket = serverChannel.channel();
//初始化地址和端口,这里使用了本地回环地址
InetSocketAddress address = new InetSocketAddress(port);
//serverSocket绑定地址
serverSocket.bind(address);
//最后将serverChannel注册进选择器,并且监听OP_ACCEPT事件,也就是网络链接发生事件
serverChannel.register(selector,SelectionKey.OP_ACCEPT);
for(;;){
try{
//阻塞自己,监听事件发生,
//select()方法也有一个参数,可以接受时间,这样当前线程不会被完全阻塞
//加时间参数的方法适用于不需要及时响应的服务器环境
selector.select();
}catch(IOException e){
e.printStackTrace();
break;
}
//获取到选择器中发生时间的key的集合
Set<SelectionKey> keys = selector.selectedKeys();
//获取key集合的迭代器
Iterator<SelectionKey> iterators = keys.iterator();
while(iterators.hasNext()){
//获取key
SelectKey key = iterators.next();
try{
//如果是链接进入事件
if(key.isAcceptable()){
//这里这么做是因为可能有多个ServerSocketChannel同时工作在一个选择器
//key中包含了很多信息
ServerSocketChannel sChanel =(ServerSocketChannel) key.channel();
//调用ServerSocketChannel的accept方法获取当前客户端链接Channel
SocketChannel cChanel = sChannel.accept();
//配置为非阻塞
cChannel.configureBlocking(false);
//将channel注册进选择器,并且分配一个1024字节的缓冲区
cChannel.register(selector,SelectionKey.OP_READ | SelectionKey.OP_WRITE,BufferByte.allocate(1024));
}
//写事件发生
if(key.isWriteable()){
//获得到当前的channel,注意,读事件写事件的channel都是客户端链接Channel,
SocketChannel cChannel = (SocketChannel)key.channel();
//获得到链接的字符缓冲
ByteBuffer buffer = (ByteBuffer) key.attachment();
while(buffer.hasRemaining()){
if(cChannel.write(buffer)==0){
break;
}
}
//关闭channel
cChannel.close();
}
}catch(IOException e ){
key.cancel();
try{
key.channel.close();
}catch(IOException e){
e.printStackTrace();
}
}finally{
//最终移除掉这个key,防止重复操作
iterators.remove();
}
}
}
}
}
public class Netty_OIOServer{
public void start(int port){
//创建内容为Hi的字节缓冲流
final ByteBuf buf = Unpooled.unreleasableBuffer(Unpooled.copiedBuffer("Hi!\r\n",Charset.forName("UTF-8")));
//创建一个EventLoopGroup组
EventLoopGroup evtLoopGroup = new OioEventLoopGroup();
try{
//bootStrap是netty服务器管理的核心组件
ServerBootStrap bootStrap = new ServerBootStrap();
bootStrap.group(evtLoopGroup)//添加组
//使用什么ServerSocketChannel作为服务器接受链接事件的通道
.channel(OioServerSocketChannel.class)
//绑定地址
.localAddress(new InetSocketAddress(port))
//添加回调函数,使用ChannelInitialize初始化
//其实就是当有链接进入时,回用什么进行操作
.childHandler(new ChannelInitialize<SocketChannel>(){
//链接进入时回创建一个SocketChannel
//此时调用ChannelInitialize的initChannel进行初始化
@override
public void initChannel(SocketChannel ch) throws Exception {
//获得SocketChannel,并且对SocketChannel的pipeline添加回调函数
//继承自ChannelInboundHandler,或者ChannelOutboundHandler
ch.pipeline.addLast(new ChannelInboundHandler{
@override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
ctx.writeAndFlush(buf.duplicate())
.addListener(ChannelFutureListen.CLOSE);
}
})
}
})
}
}
}
public class Netty_OIOServer{
public void start(int port){
//创建内容为Hi的字节缓冲流
final ByteBuf buf = Unpooled.unreleasableBuffer(Unpooled.copiedBuffer("Hi!\r\n",Charset.forName("UTF-8")));
//创建一个EventLoopGroup组
EventLoopGroup evtLoopGroup = new NioEventLoopGroup();
try{
//bootStrap是netty服务器管理的核心组件
ServerBootStrap bootStrap = new ServerBootStrap();
bootStrap.group(evtLoopGroup)//添加组
//使用什么ServerSocketChannel作为服务器接受链接事件的通道
.channel(NioServerSocketChannel.class)
//绑定地址
.localAddress(new InetSocketAddress(port))
//添加回调函数,使用ChannelInitialize初始化
//其实就是当有链接进入时,回用什么进行操作
.childHandler(new ChannelInitialize<SocketChannel>(){
//链接进入时回创建一个SocketChannel
//此时调用ChannelInitialize的initChannel进行初始化
@override
public void initChannel(SocketChannel ch) throws Exception {
//获得SocketChannel,并且对SocketChannel的pipeline添加回调函数
//继承自ChannelInboundHandler,或者ChannelOutboundHandler
ch.pipeline.addLast(new ChannelInboundHandler{
@override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
ctx.writeAndFlush(buf.duplicate())
.addListener(ChannelFutureListen.CLOSE);
}
})
}
})
}
}
}
这里没有详细的将Netty的工作流程说的很详细,但是大家可以做一个小对比