现在流行NIO网络编程,比较火的框架有Netty和Mina,这个地方我实现传统Socket编程,每一个请求,都会为之创建一个线程来进行处理操作,在Socker数据传输中,用到了PrintWrite,需要注意println()和write()两个方法的区别,println()是带有回车符号的些数据,而write()没有回车符的些数据,需要手动加上回车符,不然消息发送不过去,因为回车符是消息结束的标志,没有回车符,就认为消息没读完。

概念简介

Java对BIO、NIO、AIO的支持:

  1、Java BIO : 同步并阻塞,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,如果这个连接不做任何事情会造成不必要的线程开销,当然可以通过线程池机制改善。

  2、Java NIO : 同步非阻塞,服务器实现模式为一个请求一个线程,即客户端发送的连接请求都会注册到多路复用器上,多路复用器轮询到连接有I/O请求时才启动一个线程进行处理。

  3、Java AIO(NIO.2) : 异步非阻塞,服务器实现模式为一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理

BIO、NIO、AIO适用场景分析:

  1、BIO方式适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4以前的唯一选择,但程序直观简单易理解。

  2、NIO方式适用于连接数目多且连接比较短(轻操作)的架构,比如聊天服务器,并发局限于应用中,编程比较复杂,JDK1.4开始支持。

  3、AIO方式使用于连接数目多且连接比较长(重操作)的架构,比如相册服务器,充分调用OS参与并发操作,编程比较复杂,JDK7开始支持。

案例

服务端

服务端中,定义了一个SocketHandler,用于处理消息和数据的,每进来一个消息,就给他创建一个线程,这种传统的方式,效率比较的低下

package yellowcong.socket;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * 创建日期:2017年10月6日 <br/>
 * 创建用户:yellowcong <br/>
 * 功能描述:
 */
public class Server {

    //端口
    public static final Integer PORT = 8080;

    //ip地址
    public static final String ADDRESS = "127.0.0.1";

    /**
     * 创建日期:2017年10月6日<br/>
     * 创建用户:yellowcong<br/>
     * 功能描述:
     * 
     * @param args
     * @throws Exception
     */
    public static void main(String[] args) throws Exception {

        // 创建Socker的链接
        ServerSocket server = new ServerSocket(PORT);

        System.out.println("Socket服务器端,启动");

        // 接受数据, 线程会阻塞到这个地方
        Socket socket = server.accept();

        System.out.println("连接上了");

        new Thread(new SocketHandler(socket)).start();
    }
}

消息处理

消息的读取用到了BufferedReader,消息的写用到了PrintWriter ,写消息的时候,建议使用 println()方法,如果非要用write方法,就需要在消息后面添加回车符,不然数据传输过去后,读取方法的时候,BufferedReader中readLine()需要读取到回车符,才可以结束。

package yellowcong.socket;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;

/**
 * 创建日期:2017年10月6日 <br/>
 * 创建用户:yellowcong <br/>
 * 功能描述:  
 */
public class SocketHandler implements Runnable{
    private Socket socket ;
    public SocketHandler(Socket socket) {
        super();
        this.socket = socket;
    }

    public void run() {
        //获取写的数据,自动刷新
        BufferedReader in = null;
        //写数据
        PrintWriter out = null;

        try {
            //获取写的数据,自动刷新
            in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            //写数据, 第二个参数,表示自动flush
            out = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()),true);

            String content = null;

            while(true){
                content = in.readLine();
                if(content == null){
                    break;
                }
                System.out.println("服务器收到:\t"+content);
                out.println("逗比连接上了");
            }
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }finally{
            try {
                if(in != null){
                    in.close();
                }
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            if(out != null){
                out.close();
            }

            socket = null;
        }
    }

}

客户端

package yellowcong.socket;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import java.net.UnknownHostException;

/**
 * 创建日期:2017年10月6日 <br/>
 * 创建用户:yellowcong <br/>
 * 功能描述:  
 */
public class Client {

    public static void main(String[] args) {
        //发送数据
        PrintWriter out = null;
        //读取数据
        BufferedReader in = null;
        Socket socket = null;
        try {
            socket = new Socket(Server.ADDRESS,Server.PORT);
            System.out.println("客户端启动");

            //设定自动flush
            out = new PrintWriter(socket.getOutputStream(),true);

            //write方法需要 加上\r\n,这样服务器端,才会收到终止命令,不然就会一直读
//          out.write("hello 服务期2\r\n");
            out.println("逗比服务器");

            in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            System.out.println("客户端收到:\t"+in.readLine());
        } catch (UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{

            //资源释放
            if(in != null){
                try {
                    in.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(out != null){
                out.close();
            }

            socket = null;
        }

    }
}

测试结果

测试的时候,先启动服务器端,然后再启动客户端

服务端信息

Java网络编程之BIO(Socket)-yellowcong_客户端

客户端信息

Java网络编程之BIO(Socket)-yellowcong_客户端_02