NIO——简易Client端
2019-07-03 本文已影响0人
大黑跟小白的日常
步骤说明
1、确定要连接的目标服务器的地址跟端口,开启一个新的线程,单独连接服务器;
2、初始化线程参数,打开selector;
3、打开SocketChannel,设置为非阻塞;
4、用SocketChannel尝试连接服务端,如果连接上,则将SocketChannel注册到选择器,赋予读取操作位——OP_READ,并通过Buffer向通道中写出请求数据;如果没有连接上,则将SocketChannel注册到选择器,并赋予连接操作位——OP_CONNECT;
5、定时轮询,获取选择键集合;
6、定义单独的方法处理具体选择键;
7、判断选择键是否有效,用SocketChannel来接收key中的channel值;
8、如果选择键处于可连接状态(isConnectable),则判断是否已经完成连接,如果已经连接则不再予以处理;如果没有完成连接则将SocketChannel注册到选择器上,并且赋予读取操作位——OP_READ,并通过Buffer向SocketChannel写出请求;
9、如果选择键处于可读取状态(isReadable),则创建Buffer,从通道SocketChannel中读取数据;
代码如下
public class NioTimeClient {
public static void main(String[] args) {
int port = 10002;
if (args != null && args.length > 0) {
port = Integer.valueOf(args[0]);
}
new Thread(new TimeClientHandle("127.0.0.1", port), "TimeClient-001").start();
}
}
class TimeClientHandle implements Runnable {
private String host;
private int port;
private Selector selector;
private SocketChannel socketChannel;
private volatile boolean stop;
public TimeClientHandle(String host, int port) {
this.host = host == null ? "127.0.0.1" : host;
this.port = port;
try {
this.selector = Selector.open();
this.socketChannel = SocketChannel.open();
this.socketChannel.configureBlocking(false);
} catch (IOException e) {
e.printStackTrace();
System.exit(1);
}
}
@Override
public void run() {
try {
doConnect();
} catch (Exception e) {
e.printStackTrace();
}
while (!stop) {
try {
selector.select(1000);
Set<SelectionKey> selectedKeys = selector.selectedKeys();
Iterator<SelectionKey> it = selectedKeys.iterator();
SelectionKey key = null;
while (it.hasNext()) {
key = it.next();
it.remove();
try {
handleInput(key);
} catch (Exception e) {
if (key != null) {
key.cancel();
if (key.channel() != null) {
key.channel().close();
}
}
}
}
} catch (IOException e) {
e.printStackTrace();
System.exit(1);
}
}
}
private void handleInput(SelectionKey key) throws IOException {
if (key.isValid()) {
SocketChannel sc = (SocketChannel) key.channel();
if (key.isConnectable()) {
if (sc.finishConnect()) {
sc.register(selector, SelectionKey.OP_READ);
doWrite(sc);
} else {
System.exit(1);
}
}
if (key.isReadable()) {
ByteBuffer readBuffer = ByteBuffer.allocate(1024);
int readBytes = sc.read(readBuffer);
if (readBytes > 0) {
readBuffer.flip();
byte[] bytes = new byte[readBuffer.remaining()];
readBuffer.get(bytes);
String body = new String(bytes, "UTF-8");
System.out.println("Now is : " + body);
this.stop = true;
} else if (readBytes < 0) {
key.cancel();
sc.close();
} else
;
}
}
}
private void doConnect() throws IOException {
if (socketChannel.connect(new InetSocketAddress(host, port))) {
socketChannel.register(selector, SelectionKey.OP_READ);
doWrite(socketChannel);
} else {
socketChannel.register(selector, SelectionKey.OP_CONNECT);
}
}
private void doWrite(SocketChannel sc) throws IOException {
byte[] req = "QUERY TIME ORDER".getBytes();
ByteBuffer writeBuffer = ByteBuffer.allocate(req.length);
writeBuffer.put(req);
writeBuffer.flip();
sc.write(writeBuffer);
if (!writeBuffer.hasRemaining()) {
System.out.println("Send order 2 server succeed.");
}
}