`

3,在EchoServer中混合用阻塞模式与非阻塞模式

 
阅读更多

      在本例中,将接收客户连接的操作单独由一个线程完成,把接收数据和发送数据的操作由另一个线程完成,这可以提高服务器的并发性能。

      负责接收客户连线的线程按照阻塞模式工作,如果收到客户连接,就向selector注册读就绪和写就绪事件,否则进入阻塞状态,直到接收到了客户的连接。负责接收数据和发送数据的线程按照非阻塞模式工作,只有在读就绪和写就绪的事件发生时,才执行相应的接收数据和发送数据操作。

 

package com.test.socket.nio.thread2;
import java.io.*;
import java.nio.*;
import java.nio.channels.*;
import java.nio.charset.*;
import java.net.*;
import java.util.*;

public class EchoServer{
  private Selector selector = null;
  private ServerSocketChannel serverSocketChannel = null;
  private int port = 8000;
  private Charset charset=Charset.forName("GBK");

  public EchoServer()throws IOException{
    selector = Selector.open();
    serverSocketChannel= ServerSocketChannel.open();
    serverSocketChannel.socket().setReuseAddress(true);
    serverSocketChannel.socket().bind(new InetSocketAddress(port));
    System.out.println("服务器启动");
  }

  public void accept(){
      for(;;){
        try{
            SocketChannel socketChannel = serverSocketChannel.accept();
            System.out.println("接收到客户连接,来自:" +
                               socketChannel.socket().getInetAddress() +
                               ":" + socketChannel.socket().getPort());
            socketChannel.configureBlocking(false);

            ByteBuffer buffer = ByteBuffer.allocate(1024);
            synchronized(gate){
                selector.wakeup();
                socketChannel.register(selector,
                                       SelectionKey.OP_READ |
                                       SelectionKey.OP_WRITE, buffer);
            }
        }catch(IOException e){e.printStackTrace();}
      }
  }
  private Object gate=new Object();
  public void service() throws IOException{
    for(;;){
      synchronized(gate){}
      int n = selector.select();

      if(n==0)continue;
      Set readyKeys = selector.selectedKeys();
      Iterator it = readyKeys.iterator();
      while (it.hasNext()){
        SelectionKey key=null;
        try{
            key = (SelectionKey) it.next();
            it.remove();
            if (key.isReadable()) {
                receive(key);
            }
            if (key.isWritable()) {
                send(key);
            }
        }catch(IOException e){
           e.printStackTrace();
           try{
               if(key!=null){
                   key.cancel();
                   key.channel().close();
               }
           }catch(Exception ex){e.printStackTrace();}
        }
      }//#while
    }//#while
  }

  public void send(SelectionKey key)throws IOException{
    ByteBuffer buffer=(ByteBuffer)key.attachment();
    SocketChannel socketChannel=(SocketChannel)key.channel();
    buffer.flip();  //把极限设为位置
    String data=decode(buffer);
    if(data.indexOf("\n")==-1)return;
    String outputData=data.substring(0,data.indexOf("\n")+1);
    System.out.print(outputData);
    ByteBuffer outputBuffer=encode("echo:"+outputData);
    while(outputBuffer.hasRemaining())
      socketChannel.write(outputBuffer);

    ByteBuffer temp=encode(outputData);
    buffer.position(temp.limit());
    buffer.compact();

    if(outputData.equals("bye\r\n")){
      key.cancel();
      socketChannel.close();
      System.out.println("关闭与客户的连接");
    }
  }

  public void receive(SelectionKey key)throws IOException{
    ByteBuffer buffer=(ByteBuffer)key.attachment();

    SocketChannel socketChannel=(SocketChannel)key.channel();
    ByteBuffer readBuff= ByteBuffer.allocate(32);
    socketChannel.read(readBuff);
    readBuff.flip();

    buffer.limit(buffer.capacity());
    buffer.put(readBuff);
  }

  public String decode(ByteBuffer buffer){  //解码
    CharBuffer charBuffer= charset.decode(buffer);
    return charBuffer.toString();
  }
  public ByteBuffer encode(String str){  //编码
    return charset.encode(str);
  }

  public static void main(String args[])throws Exception{
    final EchoServer server = new EchoServer();
    Thread accept=new Thread(){
        public void run(){
            server.accept();
        }
    };
    accept.start();
    server.service();
  }
}


 
分享到:
评论

相关推荐

    EchoServer回射服务器

    在实现EchoServer时,通常会使用非阻塞I/O和套接字选项如SO_REUSEADDR来优化服务器的启动和连接处理。非阻塞I/O确保服务器在等待数据时不会被阻塞,而SO_REUSEADDR选项则允许服务器快速重启,即使前一个实例可能还...

    udp_echoserver DP83848网络服务函数库

    3. 数据接收:使用阻塞或非阻塞模式等待接收UDP数据包,通常会有一个循环来持续接收数据。 4. 数据处理:接收到数据后,服务器可能会对数据进行某种处理,例如检查内容或计算校验和。 5. 数据响应:将处理后的数据回...

    关于lwip echo server 例程1

    在 lwip echo server 例程1 中,main.c 文件中的主循环程序是整个示例程序的核心部分。该循环程序负责接收和处理网络数据包,包括接收 TCP 数据包、处理 TCP 连接请求和传送数据包等。 TcpFastTmrFlag 和 ...

    echoServer(按行读取版本)

    在我们的`echoServer`中,实现`readline`的关键在于正确处理不同大小的行,同时确保不丢失任何数据。由于`recv`函数可能不会一次性读取到整行数据,因此需要配合`recv_peek`来判断何时能完整地读取一行。 `writen`...

    以netty4.1源码中的EchoServer为例对netty的源码进行分析.docx

    - 在 `NioServerSocketChannel` 的实例化过程中,通过 `java.nio.channels.ServerSocketChannel` 接收新的连接请求,并设置为非阻塞模式。 - `NioServerSocketChannelConfig` 被用来配置创建的 `...

    echoServer(thread版)源代码

    本示例"echoServer(thread版)源代码"关注的是如何将一个并发的回显服务器转化为多线程模式,以提高服务处理能力并应对高并发场景。回显服务器是一种简单的服务器,它接收客户端发送的数据并原封不动地返回给客户端,...

    evpp based echo server

    在"evpp based echo server"项目中,可能使用valgrind来检查代码中是否存在内存泄漏或者内存使用不当的问题,以确保服务器在长期运行时的稳定性。 至于压缩包中的"event_server"文件,很可能是项目的主程序或者...

    java网络编程(非阻塞与阻塞编程)

    在深入探讨Java网络编程中的非阻塞与阻塞编程之前,我们先来了解这两个概念的基本含义。阻塞编程,通常指的是在程序执行过程中,当某一部分代码遇到I/O操作时,如读写文件或网络通信,整个程序会暂停运行,等待I/O...

    echoServer定长包与添加报头版本

    "echoServer定长包与添加报头版本"是一个这样的实践案例,它涉及到如何发送和接收具有固定长度前缀的报文,以及如何在报文头部添加额外的信息来增强协议的效率和可扩展性。 首先,我们要理解报文的基本结构。在...

    EchoServer

    在这个场景中,我们有两个关键文件:EchoServer.cpp和EchoServer.h。这两个文件分别代表了EchoServer的源代码实现和头文件,其中包含了类定义和函数声明。 EchoServer的工作原理非常简单:它监听特定的端口,一旦...

    (基于KC705 Microblaze 的lwip echo server 实现例程)RWddr3Ether_v5.rar

    标题中的"(基于KC705 Microblaze 的lwip echo server 实现例程)RWddr3Ether_v5.rar"指示了这是一个使用Xilinx KC705开发板,基于Microblaze处理器,并实现了LwIP(Lightweight IP)echo服务器的工程示例。...

    EchoClient EchoServer 客户 服务器程序

    而在EchoServer程序中,重点在于如何正确地监听和接受连接,以及处理接收到的数据并回送。 在实际的代码实现中,你可以使用各种编程语言,如C、C++、Java或Python等。例如,Python中的socket库提供了简单的API来...

    udp echo server client

    在本项目"udp echo server client"中,我们将探讨如何使用C语言实现一个简单的UDP回显服务器(echo server)和客户端(client)。这个项目可能来自于《Unix Network Programming》(通常简称UNP)一书中的示例。 ...

    echo server client

    标题中的"echo server client"指的是一个基于网络通信的回显服务和客户端程序。回显服务是一种常见的网络测试工具,它接收客户端发送的数据,并原封不动地返回给客户端,主要用于检查网络连接、数据传输等基本功能。...

    IOCP echo server client 服务器+客户端

    2. 监听连接:服务器开启监听,当有新的连接请求时,将新连接的套接字与IOCP关联,并设置为非阻塞模式。 3. 接收数据:客户端发送数据后,服务器会从IOCP中取出完成的接收操作,获取数据并存储。 4. 回显数据:...

    libevent echo server

    libevent example. echo server. use c++11 gcc 4.8.5 thread_pool singleton , io thread , servant thread layer

    TCP_ECHO.rar_ECHO JAVA_echo server_java tcp server

    描述中提到的"echo server"和"一些基础的概念"提示我们将学习如何创建一个简单的Java Echo服务器,以及在这个过程中会涉及到的TCP基础知识。 首先,要创建一个Java TCP Echo服务器,我们需要使用Java的`java.net....

    FPGA MPSoC_XCZU4EV基于lwip的echo server实验(VITIS实现).zip

    FPGA MPSoC XCZU2CG、XCZU2EG和XCZU4EV驱动程序。 基于VITIS实现。 项目代码可顺利编译运行~

    tcpecho.rar_ tcpecho_EchoServer_echo_echo server_tcp echo

    在本案例中,"tcpecho.rar" 是一个压缩包,包含了一个名为 "tcp echo" 的程序或脚本,用于实现TCP协议的Echo服务。Echo服务是一种简单的网络协议,它的工作原理是服务器接收客户端发送的数据并原封不动地返回给...

    echoserver多进程版

    我们的echo服务器最大的缺点就是无法支持多客户连接,即使客户端能够连接到服务器上,服务器也不为该客户做服务,(直接没什么反应),虽然链接是有的(也就是说,客户端是已经连接到服务器上的了,但是服务器就是不搭理你......

Global site tag (gtag.js) - Google Analytics