我用nio编写了一个简单的异步tcp服务器。 服务器应该能够为每个客户端同时读写。 我用一个简单的数据包队列来实现。
public class TcpJobHandler {
private BlockingQueue<TcpJob> _packetQueue = new LinkedBlockingQueue<TcpJob>();
private Thread _jobThread;
private final ReentrantLock _lock = new ReentrantLock();
public TcpJobHandler(){
_jobThread = new Thread(new Runnable() {
@Override
public void run() {
jobLoop();
}
});
_jobThread.start();
}
private void jobLoop(){
while(true){
try {
_lock.lock();
TcpJob job = _packetQueue.take();
if(job == null){
continue;
}
job.execute();
} catch (Exception e) {
AppLogger.error("Failed to dequeue packet from job queue.", e);
}finally{
_lock.unlock();
}
}
}
public void insertJob(TcpJob job){
try{
_packetQueue.put(job);
}catch(InterruptedException e){
AppLogger.error("Failed to queue packet to the tcp job queue.", e);
}
}
}
这段代码的作用是检查新数据包。如果有新数据包,则将该数据包发送给客户端。 在类tcp作业中,只有要发送的数据包和将数据包写入客户端流的写入类。 如您所见,只有一个线程应该能够将数据包写入客户端流。
这就是为什么我不明白,为什么我会收到这个错误?如果我是对的,这个例外说,我尝试将数据发送到流中,但是已经存在一个将数据写入此流的线程。但为什么?
//编辑: 我得到了这个例外:
19:18:41.468 [ERROR] - [mufisync.server.data.tcp.handler.TcpJobHandler] : Failed to dequeue packet from job queue. Exception: java.nio.channels.WritePendingException
at sun.nio.ch.AsynchronousSocketChannelImpl.write(Unknown Source)
at sun.nio.ch.AsynchronousSocketChannelImpl.write(Unknown Source)
at mufisync.server.data.tcp.stream.OutputStreamAdapter.write(OutputStreamAdapter.java:35)
at mufisync.server.data.tcp.stream.OutputStreamAdapter.write(OutputStreamAdapter.java:26)
at mufisync.server.data.tcp.stream.BinaryWriter.write(BinaryWriter.java:21)
at mufisync.server.data.tcp.TcpJob.execute(TcpJob.java:29)
at mufisync.server.data.tcp.handler.TcpJobHandler.jobLoop(TcpJobHandler.java:40)
at mufisync.server.data.tcp.handler.TcpJobHandler.access$0(TcpJobHandler.java:32)
at mufisync.server.data.tcp.handler.TcpJobHandler$1.run(TcpJobHandler.java:25)
at java.lang.Thread.run(Unknown Source)
TcpJob看起来像这样:
public class TcpJob {
private BasePacket _packet;
private BinaryWriter _writer;
public TcpJob(BasePacket packet, BinaryWriter writer){
_packet = packet;
_writer = writer;
}
public void execute(){
try {
if(_packet == null){
AppLogger.warn("Tcp job packet is null");
return;
}
_writer.write(_packet.toByteArray());
} catch (IOException e) {
AppLogger.error("Failed to write packet into the stream.", e);
}
}
public BasePacket get_packet() {
return _packet;
}
}
BinaryStream只是耦合到AsynchronousSocketChannel,它从套接字通道调用write(byte [])方法。
答案 0 :(得分:1)
您正在使用异步NIO2。使用异步IO时,在最后一次写入完成之前,不能调用write()。来自Javadoc
* @throws WritePendingException
* If a write operation is already in progress on this channel
e.g。如果您已经使用过
public abstract Future<Integer> write(ByteBuffer src);
在Future.get()返回之前,你不能再写。
如果您使用
public abstract <A> void write(ByteBuffer src,
long timeout,
TimeUnit unit,
A attachment,
CompletionHandler<Integer,? super A> handler);
在调用CompletionHandler之前,您无法再次写入。
注意:您不能同时执行两次读取。
在你的情况下,你需要像
这样的东西ByteBuffer lastBuffer = null;
Future<Integer> future = null;
public void execute(){
try {
if(_packet == null){
AppLogger.warn("Tcp job packet is null");
return;
}
// need to wait until the last buffer was written completely.
while (future != null) {
future.get();
if (lastBuffer.remaining() > 0)
future = _writer.write(lasBuffer);
else
break;
}
// start another write.
future = _writer.write(lastBuffer = _packet.toByteArray());
} catch (IOException e) {
AppLogger.error("Failed to write packet into the stream.", e);
}
}