java socket编程三:异步服务器;

开发异步服务器,需要使用Java的NIO才可以,,下面给出代码。

package socket;



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 ToUpperTCPNonBlockServer {
//服务器IP
public static final String SERVER_IP = "127.0.0.1";

//服务器端口号
public static final int SERVER_PORT = 9999;

//请求终结字符串
public static final char REQUEST_END_CHAR = '#';

public void startServer(String serverIP, int serverPort) throws IOException {
//使用NIO需要用到ServerSocketChannel
//其中包含一个ServerSocket对象
ServerSocketChannel serverChannel = ServerSocketChannel.open();

//创建地址对象
InetSocketAddress localAddr = new InetSocketAddress(serverIP, serverPort);

//服务器绑定地址
serverChannel.bind(localAddr);

//设置为非阻塞
serverChannel.configureBlocking(false);

//注册到selector,会调用ServerSocket的accept
//我们用selector监听accept能否返回
//当调用accept可以返回时,会得到通知
//注意,是可以返回,还需要调用accept
Selector selector = Selector.open();
serverChannel.register(selector, SelectionKey.OP_ACCEPT);

while (true) {

//调用select,阻塞在这里,直到有注册的channel满足条件
selector.select();

//如果走到这里,有符合条件的channel
//可以通过selector.selectedKeys().iterator()拿到符合条件的迭代器
Iterator<SelectionKey> keys = selector.selectedKeys().iterator();

//处理满足条件的keys
while (keys.hasNext()) {
//取出一个key并移除
SelectionKey key = keys.next();
keys.remove();
try {
if (key.isAcceptable()) {
//有accept可以返回
//取得可以操作的channel
ServerSocketChannel server = (ServerSocketChannel) key.channel();

//调用accept完成三次握手,返回与客户端可以通信的channel
SocketChannel channel = server.accept();

//将该channel置非阻塞
channel.configureBlocking(false);

//注册进selector,当可读或可写时将得到通知,select返回
channel.register(selector, SelectionKey.OP_READ);
} else if (key.isReadable()) {
//有channel可读,取出可读的channel
SocketChannel channel = (SocketChannel) key.channel();

//创建读取缓冲区,一次读取1024字节
ByteBuffer buffer = ByteBuffer.allocate(1024);
channel.read(buffer);

//锁住缓冲区,缓冲区使用的大小将固定
buffer.flip();

//附加上buffer,供写出使用
key.attach(buffer);
key.interestOps(SelectionKey.OP_WRITE);
} else if (key.isWritable()) {
//有channel可写,取出可写的channel
SocketChannel channel = (SocketChannel) key.channel();

//取出可读时设置的缓冲区
ByteBuffer buffer = (ByteBuffer) key.attachment();

//将缓冲区指针移动到缓冲区开始位置
buffer.rewind();

//读取为String
String recv = new String(buffer.array());

//清空缓冲区
buffer.clear();
buffer.flip();

//写回数据
byte[] sendBytes = recv.toUpperCase().getBytes();
channel.write(ByteBuffer.wrap(sendBytes));
}
} catch (IOException e) {
//当客户端Socket关闭时,会走到这里,清理资源
key.cancel();
try {
key.channel().close();
} catch (IOException e1) {
e1.printStackTrace();
}
}
}
}
}

public static void main(String[] args) {
ToUpperTCPNonBlockServer server = new ToUpperTCPNonBlockServer();
try {
server.startServer(SERVER_IP, SERVER_PORT);
} catch (IOException e) {
e.printStackTrace();
}
}

}


客户端代码;】

package socket;


import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;




public class ToUpperTCPClient {

//客户端使用的TCP Socket
private Socket clientSocket;

public String toUpperRemote(String serverIp, int serverPort, String str) {
StringBuilder recvStrBuilder = new StringBuilder();
try {
//创建连接服务器的Socket
clientSocket = new Socket(serverIp, serverPort);

//写出请求字符串
OutputStream out = clientSocket.getOutputStream();
out.write(str.getBytes());

//读取服务器响应
InputStream in = clientSocket.getInputStream();
for (int c = in.read(); c != '#'; c = in.read()) {
recvStrBuilder.append((char)c);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (clientSocket != null) {
clientSocket.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}

return recvStrBuilder.toString();
}

public static void main(String[] args) {
ToUpperTCPClient client = new ToUpperTCPClient();
String recvStr = client.toUpperRemote(ToUpperTCPNonBlockServer.SERVER_IP, ToUpperTCPNonBlockServer.SERVER_PORT, 
"dffdsfdfdfdfdfAAA" + ToUpperTCPNonBlockServer.REQUEST_END_CHAR);
System.out.println("收到:" + recvStr);
}
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值