`

Socket实战之六 使用NIO包实现Socket通信

    博客分类:
  • J2SE
阅读更多
本文地址:http://blog.csdn.net/kongxx/article/details/7288896
前面几篇文章介绍了使用java.io和java.net类库实现的Socket通信,下面介绍一下使用java.nio类库实现的Socket。
java.nio包是Java在1.4之后增加的,用来提高I/O操作的效率。在nio包中主要包括以下几个类或接口:

* Buffer:缓冲区,用来临时存放输入或输出数据。

* Charset:用来把Unicode字符编码和其它字符编码互转。

* Channel:数据传输通道,用来把Buffer中的数据写入到数据源,或者把数据源中的数据读入到Buffer。

* Selector:用来支持异步I/O操作,也叫非阻塞I/O操作。

nio包中主要通过下面两个方面来提高I/O操作效率:

* 通过Buffer和Channel来提高I/O操作的速度。

* 通过Selector来支持非阻塞I/O操作。


下面来看一下程序中是怎么通过这些类库实现Socket功能。

首先介绍一下几个辅助类
辅助类SerializableUtil,这个类用来把java对象序列化成字节数组,或者把字节数组反序列化成java对象。
[java] view plaincopyprint?
package com.googlecode.garbagecan.test.socket; 
 
import java.io.ByteArrayInputStream; 
import java.io.ByteArrayOutputStream; 
import java.io.IOException; 
import java.io.ObjectInputStream; 
import java.io.ObjectOutputStream; 
 
public class SerializableUtil { 
     
    public static byte[] toBytes(Object object) { 
        ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
        ObjectOutputStream oos = null; 
        try { 
            oos = new ObjectOutputStream(baos); 
            oos.writeObject(object); 
            byte[] bytes = baos.toByteArray(); 
            return bytes; 
        } catch(IOException ex) { 
            throw new RuntimeException(ex.getMessage(), ex); 
        } finally { 
            try { 
                oos.close(); 
            } catch (Exception e) {} 
        } 
    } 
     
    public static Object toObject(byte[] bytes) { 
        ByteArrayInputStream bais = new ByteArrayInputStream(bytes); 
        ObjectInputStream ois = null; 
        try { 
            ois = new ObjectInputStream(bais); 
            Object object = ois.readObject(); 
            return object; 
        } catch(IOException ex) { 
            throw new RuntimeException(ex.getMessage(), ex); 
        } catch(ClassNotFoundException ex) { 
            throw new RuntimeException(ex.getMessage(), ex); 
        } finally { 
            try { 
                ois.close(); 
            } catch (Exception e) {} 
        } 
    } 

辅助类MyRequestObject和MyResponseObject,这两个类是普通的java对象,实现了Serializable接口。MyRequestObject类是Client发出的请求,MyResponseObject是Server端作出的响应。
[java] view plaincopyprint?
package com.googlecode.garbagecan.test.socket.nio; 
 
import java.io.Serializable; 
 
public class MyRequestObject implements Serializable { 
 
    private static final long serialVersionUID = 1L; 
 
    private String name; 
     
    private String value; 
 
    private byte[] bytes; 
     
    public MyRequestObject(String name, String value) { 
        this.name = name; 
        this.value = value; 
        this.bytes = new byte[1024]; 
    } 
     
    public String getName() { 
        return name; 
    } 
 
    public void setName(String name) { 
        this.name = name; 
    } 
 
    public String getValue() { 
        return value; 
    } 
 
    public void setValue(String value) { 
        this.value = value; 
    } 
     
    @Override 
    public String toString() { 
        StringBuffer sb = new StringBuffer(); 
        sb.append("Request [name: " + name  + ", value: " + value + ", bytes: " + bytes.length+ "]"); 
        return sb.toString(); 
    } 

 
package com.googlecode.garbagecan.test.socket.nio; 
 
import java.io.Serializable; 
 
public class MyResponseObject implements Serializable { 
 
    private static final long serialVersionUID = 1L; 
 
    private String name; 
     
    private String value; 
 
    private byte[] bytes; 
     
    public MyResponseObject(String name, String value) { 
        this.name = name; 
        this.value = value; 
        this.bytes = new byte[1024]; 
    } 
     
    public String getName() { 
        return name; 
    } 
 
    public void setName(String name) { 
        this.name = name; 
    } 
 
    public String getValue() { 
        return value; 
    } 
 
    public void setValue(String value) { 
        this.value = value; 
    } 
     
    @Override 
    public String toString() { 
        StringBuffer sb = new StringBuffer(); 
        sb.append("Response [name: " + name  + ", value: " + value + ", bytes: " + bytes.length+ "]"); 
        return sb.toString(); 
    } 


下面主要看一下Server端的代码,其中有一些英文注释对理解代码很有帮助,注释主要是来源jdk的文档和例子,这里就没有再翻译
[java] view plaincopyprint?
package com.googlecode.garbagecan.test.socket.nio; 
 
import java.io.ByteArrayOutputStream; 
import java.io.IOException; 
import java.net.InetSocketAddress; 
import java.nio.ByteBuffer; 
import java.nio.channels.ClosedChannelException; 
import java.nio.channels.SelectionKey; 
import java.nio.channels.Selector; 
import java.nio.channels.ServerSocketChannel; 
import java.nio.channels.SocketChannel; 
import java.util.Iterator; 
import java.util.logging.Level; 
import java.util.logging.Logger; 
 
import com.googlecode.garbagecan.test.socket.SerializableUtil; 
 
public class MyServer3 { 
 
    private final static Logger logger = Logger.getLogger(MyServer3.class.getName()); 
     
    public static void main(String[] args) { 
        Selector selector = null; 
        ServerSocketChannel serverSocketChannel = null; 
         
        try { 
            // Selector for incoming time requests 
            selector = Selector.open(); 
 
            // Create a new server socket and set to non blocking mode 
            serverSocketChannel = ServerSocketChannel.open(); 
            serverSocketChannel.configureBlocking(false); 
             
            // Bind the server socket to the local host and port 
            serverSocketChannel.socket().setReuseAddress(true); 
            serverSocketChannel.socket().bind(new InetSocketAddress(10000)); 
             
            // Register accepts on the server socket with the selector. This 
            // step tells the selector that the socket wants to be put on the 
            // ready list when accept operations occur, so allowing multiplexed 
            // non-blocking I/O to take place. 
            serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT); 
     
            // Here's where everything happens. The select method will 
            // return when any operations registered above have occurred, the 
            // thread has been interrupted, etc. 
            while (selector.select() > 0) { 
                // Someone is ready for I/O, get the ready keys 
                Iterator<SelectionKey> it = selector.selectedKeys().iterator(); 
     
                // Walk through the ready keys collection and process date requests. 
                while (it.hasNext()) { 
                    SelectionKey readyKey = it.next(); 
                    it.remove(); 
                     
                    // The key indexes into the selector so you 
                    // can retrieve the socket that's ready for I/O 
                    execute((ServerSocketChannel) readyKey.channel()); 
                } 
            } 
        } catch (ClosedChannelException ex) { 
            logger.log(Level.SEVERE, null, ex); 
        } catch (IOException ex) { 
            logger.log(Level.SEVERE, null, ex); 
        } finally { 
            try { 
                selector.close(); 
            } catch(Exception ex) {} 
            try { 
                serverSocketChannel.close(); 
            } catch(Exception ex) {} 
        } 
    } 
 
    private static void execute(ServerSocketChannel serverSocketChannel) throws IOException { 
        SocketChannel socketChannel = null; 
        try { 
            socketChannel = serverSocketChannel.accept(); 
            MyRequestObject myRequestObject = receiveData(socketChannel); 
            logger.log(Level.INFO, myRequestObject.toString()); 
             
            MyResponseObject myResponseObject = new MyResponseObject( 
                    "response for " + myRequestObject.getName(),  
                    "response for " + myRequestObject.getValue()); 
            sendData(socketChannel, myResponseObject); 
            logger.log(Level.INFO, myResponseObject.toString()); 
        } finally { 
            try { 
                socketChannel.close(); 
            } catch(Exception ex) {} 
        } 
    } 
     
    private static MyRequestObject receiveData(SocketChannel socketChannel) throws IOException { 
        MyRequestObject myRequestObject = null; 
        ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
        ByteBuffer buffer = ByteBuffer.allocate(1024); 
         
        try { 
            byte[] bytes; 
            int size = 0; 
            while ((size = socketChannel.read(buffer)) >= 0) { 
                buffer.flip(); 
                bytes = new byte[size]; 
                buffer.get(bytes); 
                baos.write(bytes); 
                buffer.clear(); 
            } 
            bytes = baos.toByteArray(); 
            Object obj = SerializableUtil.toObject(bytes); 
            myRequestObject = (MyRequestObject)obj; 
        } finally { 
            try { 
                baos.close(); 
            } catch(Exception ex) {} 
        } 
        return myRequestObject; 
    } 
 
    private static void sendData(SocketChannel socketChannel, MyResponseObject myResponseObject) throws IOException { 
        byte[] bytes = SerializableUtil.toBytes(myResponseObject); 
        ByteBuffer buffer = ByteBuffer.wrap(bytes); 
        socketChannel.write(buffer); 
    } 

下面是Client的代码,代码比较简单就是启动了100个线程来访问Server
[java] view plaincopyprint?
package com.googlecode.garbagecan.test.socket.nio; 
 
import java.io.ByteArrayOutputStream; 
import java.io.IOException; 
import java.net.InetSocketAddress; 
import java.net.SocketAddress; 
import java.nio.ByteBuffer; 
import java.nio.channels.SocketChannel; 
import java.util.logging.Level; 
import java.util.logging.Logger; 
 
import com.googlecode.garbagecan.test.socket.SerializableUtil; 
 
public class MyClient3 { 
 
    private final static Logger logger = Logger.getLogger(MyClient3.class.getName()); 
     
    public static void main(String[] args) throws Exception { 
        for (int i = 0; i < 100; i++) { 
            final int idx = i; 
            new Thread(new MyRunnable(idx)).start(); 
        } 
    } 
     
    private static final class MyRunnable implements Runnable { 
         
        private final int idx; 
 
        private MyRunnable(int idx) { 
            this.idx = idx; 
        } 
 
        public void run() { 
            SocketChannel socketChannel = null; 
            try { 
                socketChannel = SocketChannel.open(); 
                SocketAddress socketAddress = new InetSocketAddress("localhost", 10000); 
                socketChannel.connect(socketAddress); 
 
                MyRequestObject myRequestObject = new MyRequestObject("request_" + idx, "request_" + idx); 
                logger.log(Level.INFO, myRequestObject.toString()); 
                sendData(socketChannel, myRequestObject); 
                 
                MyResponseObject myResponseObject = receiveData(socketChannel); 
                logger.log(Level.INFO, myResponseObject.toString()); 
            } catch (Exception ex) { 
                logger.log(Level.SEVERE, null, ex); 
            } finally { 
                try { 
                    socketChannel.close(); 
                } catch(Exception ex) {} 
            } 
        } 
 
        private void sendData(SocketChannel socketChannel, MyRequestObject myRequestObject) throws IOException { 
            byte[] bytes = SerializableUtil.toBytes(myRequestObject); 
            ByteBuffer buffer = ByteBuffer.wrap(bytes); 
            socketChannel.write(buffer); 
            socketChannel.socket().shutdownOutput(); 
        } 
 
        private MyResponseObject receiveData(SocketChannel socketChannel) throws IOException { 
            MyResponseObject myResponseObject = null; 
            ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
             
            try { 
                ByteBuffer buffer = ByteBuffer.allocateDirect(1024); 
                byte[] bytes; 
                int count = 0; 
                while ((count = socketChannel.read(buffer)) >= 0) { 
                    buffer.flip(); 
                    bytes = new byte[count]; 
                    buffer.get(bytes); 
                    baos.write(bytes); 
                    buffer.clear(); 
                } 
                bytes = baos.toByteArray(); 
                Object obj = SerializableUtil.toObject(bytes); 
                myResponseObject = (MyResponseObject) obj; 
                socketChannel.socket().shutdownInput(); 
            } finally { 
                try { 
                    baos.close(); 
                } catch(Exception ex) {} 
            } 
            return myResponseObject; 
        } 
    } 


最后测试上面的代码,首先运行Server类,然后运行Client类,就可以分别在Server端和Client端控制台看到发送或接收到的MyRequestObject或MyResponseObject对象了。

关于NIO和IO的比较,下面的两篇文章对理解很有帮助,可以参考一下。
http://tutorials.jenkov.com/java-nio/nio-vs-io.html

https://blogs.oracle.com/slc/entry/javanio_vs_javaio
分享到:
评论

相关推荐

    java nio 实现socket

    java nio 实现socketjava nio 实现socketjava nio 实现socketjava nio 实现socketjava nio 实现socket

    使用NIO实现非阻塞socket通信

    Java编写的简易聊天工具,使用NIO实现非阻塞socket通信,使用Java原生sdk实现,可以运行。

    Socket 之 BIO、NIO、Netty 简单实现

    《Socket 之 BIO、NIO、Netty 简单实现》博客附件。 博客地址:https://blog.csdn.net/Supreme_Sir/article/details/112725728

    Java使用NIO包实现Socket通信的实例代码

    本篇文章主要介绍了Java使用NIO包实现Socket通信的实例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

    Java Socket学习---nio实现阻塞多线程通信

    NULL 博文链接:https://1358440610-qq-com.iteye.com/blog/2115715

    socket通信NIO代理模式demo实例

    socket通信nio模式有很多实现方式,但是在性能上、资源上一般很少考虑,这里封装了一个性能极强的程序。

    Nio非阻塞socket通信demo

    本人写的Nio非阻塞socket通信demo,内有注释。

    android-socket-nio-master.zip

    socket使用nio替代传统实现,效率更高。

    基于NIO的socket举例

    基于NIO的socket举例 基于NIO的socket举例 基于NIO的socket举例 基于NIO的socket举例 基于NIO的socket举例基于NIO的socket举例 基于NIO的socket举例

    Socket与NIO的Demo.rar

    用于博文https://blog.csdn.net/lyz_zyx/article/details/104062815《Android网络编程(十四) 之 Socket与NIO》中演示Socket与NIO使用的Demo

    默蓝网络通信测试工具(NIOSocket工具)支持TCP/IP和HTTP通信-网络通信开发人员必备

    网络通信工具,服务端和客户端连接测试工具,可单条发送,循环发送,模拟多客户端发送,本工具可以作为网络通信工具或压力测试工具, Java NIO Socket编程,需JAVA运行环境

    java socketNIO 实现多客户端聊天室 代码

    利用socketNIO实现的多客户端聊天室,非阻塞式IO,java代码编写,使用方法:先启动服务端代码再启动客户端代码,可启动多个客户端代码。若使用多个电脑启动客户端,需在客户端代码中更改一下ip地址。

    java NIO socket聊天室

    使用NIO socket不需要多线程来处理多个连接的请求,效率非常高 可以作为NIO socket入门的例子,Reactor模式,重点理解key.attach, jar文件里包含了源代码 1,运行server.bat启动服务器,可以打开编辑,修改端口号 ...

    《NIO与Socket编程技术指南》高洪岩.zip

    非常详细地讲解了NIO中的缓冲区、通道、选择器、编码,以及使用Socket技术实现TCP/IP和UDP编程,细化到了演示全部SocketOption的特性,这对理解基于NIO和Socket技术为基础所开发的NIO框架是非常有好处的,本书以案例...

    java NIO socket聊天

    java NIO 高性能 socket通讯,服务端采用单线程,降低了cpu的压力,普通io socket通讯,server需要每个连接运行个线程,容易出现问题,效率也低

    java nio socket 例子

    本例包含服务器端和客户端,多线程,每线程多次发送,Eclipse工程,启动服务器使用 nu.javafaq.server.NioServer,启动客户端使用 nu.javafaq.client.NioClient。另本例取自javafaq.nv上的程序修改而成

    Java NIO Socket基本

    NULL 博文链接:https://b-l-east.iteye.com/blog/1254693

    java nio 通信服务器、客户端完整例子

    用java编写的nio通信的例子,nio是io编程的新版本,比io较流行。同时本例子是适用socket通信的。可以在此基础上,添加您的个人应用。本例子适用于:java通信的学习者,android平台通信的学习者。

    HP-Socket下载

    HP-Socket 对通信层实现完全封装,应用程序不必关注通信层的任何细节;HP-Socket 提供基于事件通知模型的 API 接口,能非常简单高效地整合到新旧应用程序中。  为了让使用者能方便快速地学习和使用 HP-Socket,...

    采用NIO实现一个Socket服务器

    NULL 博文链接:https://wjy320.iteye.com/blog/2002237

Global site tag (gtag.js) - Google Analytics