socket异步通信

所谓异步输入输出机制,是指在进行输入输出处理时,不必等到输入输出处理完毕才返回。所以异步的同义语是非阻塞(None Blocking)。在服务器端,ServerSocketChannel通过静态函数open()返回一个实例serverChl。然后该通道调用serverChl.socket().bind()绑定到服务器某端口,并调用register(Selector sel, SelectionKey.OP_ACCEPT)注册OP_ACCEPT事件到一个选择器中(ServerSocketChannel只可以注册OP_ACCEPT事件)。当有客户请求连接时,选择器就会通知该通道有客户连接请求,就可以进行相应的输入输出控制了;在客户端,clientChl实例注册自己感兴趣的事件后(可以是OP_CONNECT,OP_READ,OP_WRITE的组合),调用clientChl.connect(InetSocketAddress )连接服务器然后进行相应处理。注意,这里的连接是异步的,即会立即返回而继续执行后面的代码。
  
  选择器和选择键介绍:
  选择器(Selector)的作用是:将通道感兴趣的事件放入队列中,而不是马上提交给应用程序,等已注册的通道自己来请求处理这些事件。换句话说,就是选择器将会随时报告已经准备好了的通道,而且是按照先进先出的顺序。那么,选择器是通过什么来报告的呢?选择键(SelectionKey)。选择键的作用就是表明哪个通道已经做好了准备,准备干什么。你也许马上会想到,那一定是已注册的通道感兴趣的事件。不错,例如对于服务器端serverChl来说,可以调用key.isAcceptable()来通知serverChl有客户端连接请求。相应的函数还有:SelectionKey.isReadable(),SelectionKey.isWritable()。一般的,在一个循环中轮询感兴趣的事件(具体可参照下面的代码)。如果选择器中尚无通道已注册事件发生,调用Selector.select()将阻塞,直到有事件发生为止。另外,可以调用selectNow()或者select(long timeout)。前者立即返回,没有事件时返回0值;后者等待timeout时间后返回。一个选择器最多可以同时被63个通道一起注册使用。

public class NBlockingServer {
  int port = 8000;
  int BUFFERSIZE = 1024;
  Selector selector = null;
  ServerSocketChannel serverChannel = null;
  HashMap clientChannelMap = null;//用来存放每一个客户连接对应的套接字和通道
  
  public NBlockingServer( int port ) {
    this.clientChannelMap = new HashMap();
  this.port = port;
  }
  
  public void initialize() throws IOException {
   //初始化,分别实例化一个选择器,一个服务器端可选择通道
   this.selector = Selector.open();
   this.serverChannel = ServerSocketChannel.open();
   this.serverChannel.configureBlocking(false);
   InetAddress localhost = InetAddress.getLocalHost();
   InetSocketAddress isa = new InetSocketAddress(localhost, this.port );
   this.serverChannel.socket().bind(isa);//将该套接字绑定到服务器某一可用端口
  }
  //结束时释放资源
  public void finalize() throws IOException {
    this.serverChannel.close();
    this.selector.close();
  }
  //将读入字节缓冲的信息解码
  public String decode( ByteBuffer byteBuffer ) throws
  CharacterCodingException {
    Charset charset = Charset.forName( "ISO-8859-1" );
    CharsetDecoder decoder = charset.newDecoder();
    CharBuffer charBuffer = decoder.decode( byteBuffer );
    String result = charBuffer.toString();
    return result;
  }
  //监听端口,当通道准备好时进行相应操作
  public void portListening() throws IOException, InterruptedException {
   //服务器端通道注册OP_ACCEPT事件
   SelectionKey acceptKey =this.serverChannel.register( this.selector,
                     SelectionKey.OP_ACCEPT );
    //当有已注册的事件发生时,select()返回值将大于0
    while (acceptKey.selector().select() > 0 ) {
      System.out.println("event happened");
      //取得所有已经准备好的所有选择键
      Set readyKeys = this.selector.selectedKeys();
      //使用迭代器对选择键进行轮询
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值