zoukankan      html  css  js  c++  java
  • java NIO 学习笔记

        项目组是做IM产品的,服务端当然用的是NIO技术做通信底层。可是一直都是对NIO有些理论的了解,没有实践。近期有空了。就实践了下NIO。

        NIO,新IO,也称之为非堵塞IO。

    非堵塞是它跟传统IO的最重要的差别之中的一个。传统IO用Socket进行通信,NIO则用channel进行消息交互。channel必须注冊到selector上。把它感兴趣的事件告诉selector。这是个观察者模式的实现。能够这样描写叙述channel和selector的关系,channel是火车轨道,selector是火车调度室。

    多个channel能够注冊到同一个selector上。在一个线程了对这些channel进行调度。然而传统IO须要为每个socket创建一个线程。

    这也是NIO的优势之中的一个,不须要太多的线程。

        NIO版本号的服务端代码:

    package cn.nio;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    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.util.Iterator;
    
    public class NioServer {
    	private Selector	serverSelector;
    
    	public NioServer initServer(int port) throws IOException {
    		ServerSocketChannel serverChannel = ServerSocketChannel.open();
    		// 非堵塞模式
    		serverChannel.configureBlocking(false);
    		// 绑定端口号
    		serverChannel.socket().bind(new InetSocketAddress(port));
    		this.serverSelector = Selector.open();
    		serverChannel.register(serverSelector, SelectionKey.OP_ACCEPT);
    		return this;
    	}
    
    	public void startListening() throws IOException {
    		System.out.println("服务端启动成功!");
    		while (true) {
    			// 当注冊的事件到达时。方法返回。否则,该方法会一直堵塞
    			serverSelector.select();
    			Iterator<SelectionKey> iterator = this.serverSelector.selectedKeys().iterator();
    			while (iterator.hasNext()) {
    				SelectionKey key = (SelectionKey) iterator.next();
    				// 必须删除。否则下次遍历时还会遍历旧的key
    				iterator.remove();	
    				if (key.isAcceptable()) {
    					accept(key);
    				} else if (key.isReadable()) {
    					read(key);
    				}
    			}
    		}
    	}
    	
    	// 接受client的连接请求
    	private void accept(SelectionKey key) throws IOException {
    		ServerSocketChannel server = (ServerSocketChannel) key.channel();
    		// 获得和client连接的通道
    		SocketChannel channel = server.accept();
    		// 设置成非堵塞
    		channel.configureBlocking(false);
    		channel.write(ByteBuffer.wrap(new String("client连接成功
    ").getBytes()));
    		// 注冊
    		channel.register(this.serverSelector, SelectionKey.OP_READ);
    	}
    
    	private void read(SelectionKey key) {
    		try {
    			SocketChannel channel = (SocketChannel) key.channel();
    			// 创建读取的缓冲区
    			ByteBuffer buffer = ByteBuffer.allocate(1024);
    			channel.read(buffer);
    			String msg = new String(buffer.array()).trim();
    			System.out.println("服务端:" + msg);
    		} catch (Exception e) {
    			e.printStackTrace();
    		}
    	}
    
    	public static void main(String[] args) throws IOException {
    		new NioServer().initServer(8080).startListening();
    	}
    }
        NIO版本号的client代码:

    package cn.nio;
    
    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.SocketChannel;
    import java.util.Iterator;
    
    public class NioClient {
    
    	private Selector	selector;
    
    	public NioClient initClient(String ip, int port) throws IOException {
    		SocketChannel channel = SocketChannel.open();
    		// 非堵塞
    		channel.configureBlocking(false);
    		this.selector = Selector.open();
    
    		// connect方法的凝视:此方法返回 false。而且必须在以后通过调用 finishConnect 方法来完毕该连接操作。
    		boolean result = channel.connect(new InetSocketAddress(ip, port));
    		System.out.println(result); // 返回false
    		// 注冊
    		channel.register(selector, SelectionKey.OP_CONNECT);
    		return this;
    	}
    
    	public void startListening() throws IOException {
    		while (true) {
    			selector.select();
    			Iterator<SelectionKey> iterator = this.selector.selectedKeys().iterator();
    			while (iterator.hasNext()) {
    				SelectionKey key = (SelectionKey) iterator.next();
    				// 必须删除,否则下次遍历时还会遍历旧的key
    				iterator.remove();
    				if (key.isConnectable()) {
    					connect(key);
    				} else if (key.isReadable()) {
    					read(key);
    				}
    			}
    		}
    	}
    
    	private void connect(SelectionKey key) throws IOException, ClosedChannelException {
    		SocketChannel channel = (SocketChannel) key.channel();
    		// 假设正在连接,则完毕连接
    		if (channel.isConnectionPending()) {
    			channel.finishConnect();
    		}
    		channel.configureBlocking(false);
    		channel.write(ByteBuffer.wrap(new String("client连接成功").getBytes()));
    		// 注冊
    		channel.register(this.selector, SelectionKey.OP_READ);
    	}
    
    	private void read(SelectionKey key) throws IOException {
    		// server可读取消息:得到事件发生的Socket通道
    		SocketChannel channel = (SocketChannel) key.channel();
    		// 创建读取的缓冲区
    		ByteBuffer buffer = ByteBuffer.allocate(1024);
    		channel.read(buffer);
    		String msg = new String(buffer.array()).trim();
    		System.out.println("client收到信息:" + msg);
    	}
    
    	/**
    	 * 启动client測试
    	 * 
    	 * @throws IOException
    	 */
    	public static void main(String[] args) throws IOException {
    		new NioClient().initClient("localhost", 8080).startListening();
    	}
    
    }


        考虑到其它语言不一定有NIO技术。我们的client都是C++的。

    那么一个NIO服务端能否够和一个非NIO的client进行通信呢?做了一个传统IO的client。做測试。可行!

    package cn.nio;
    
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.PrintWriter;
    import java.net.Socket;
    
    public class NormalClient {
    
    	Socket	clientSocket	= null;
    
    	public NormalClient initClient(String ip, int port) {
    		try {
    			clientSocket = new Socket(ip, port);
    		} catch (IOException e) {
    			e.printStackTrace();
    		}
    		return this;
    	}
    
    	public void read() throws IOException {
    		PrintWriter pw = new PrintWriter((clientSocket.getOutputStream()));
    		BufferedReader br = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));
    		String line = null;
    		while (true) {
    			pw.println("客户端发来的消息");
    			pw.flush();
    			line = br.readLine();
    			System.out.println(line);
    		}
    	}
    
    	public static void main(String[] args) throws IOException {
    		new NormalClient().initClient("127.0.0.1", 8000).read();
    	}
    }
    
        

        写这个client的时候。client一直无法收到服务端的消息。请教资深同事,问题在readline()方法上。假设调用的是readline(),那么服务端给client发消息时,须要传换行符。




  • 相关阅读:
    C#调用C++ ---参数传递
    Retained Mode Versus Immediate Mode
    mind map in latex
    vk example
    基本环境
    [转]ld 和 ld.gold 和 ld.bfd
    [转] c++11 int&& 右值引用
    [转] c++11列表初始化
    [转] c++ const, volatile, mutable用法
    [转] c++11 模板元编程
  • 原文地址:https://www.cnblogs.com/lcchuguo/p/5068036.html
Copyright © 2011-2022 走看看