记录写thrift客户端socket连接池的问题解决
在网上找了写客户端Tsocket连接池的博客。可以参见https://blog.csdn.net/tianwei7518/article/details/44003287。
1.按照博客写好连接池之后。,用连接池管理类获取TSocket(获取的TSocket不为null),替换自己new的TSocket之后。启动报错。org.apache.thrift.transport.T Transport Exception:Socket_alreadyconnected.
2.把framedTransport.open();注释后启动报错org.apache.thrift.transport.T Transport Exception:java.net.SocketException:Software caused_connection abort:recv failed.
初步诊断应该是客户端和服务器端的协议不一致。
3.现打算把连接池中的transport.open();注释掉。然后启动。启动后报错java.lang.NullPointerExceptionat org.apache.thrift.transport.T Framed Transport.open(T Framed Transport.java:82)
连接池管理中获取的TSocket,在连接池工厂创建对象时, /** * 创建对象 */ @Override public TTransport makeObject() throws Exception { try { TTransport transport = new TSocket(this.serviceIP, this.servicePort, this.timeOut); transport.open(); return transport; } catch (Exception e) { logger.error("error ThriftPoolableObjectFactory()", e); throw new RuntimeException(e); } }
4.定位是获取Tsocket为null。debugger跟踪原来是校验时抛异常,如果TSocket没打开连接池就认为他不是有效的TSocket。把这个校验注释掉。就可以啦。
/** * 检验对象是否可以由pool安全返回 */ @Override public boolean validateObject(TTransport tTransport) { try { if (tTransport instanceof TSocket) { TSocket thriftSocket = (TSocket) tTransport; if (thriftSocket.isOpen()) { return true; } else { return false; } } else { return false; } } catch (Exception e) { return false; } } }
总结:分析原因,我的客户端程序中用的是TFramedTransport,连接池中的是TTransport。我的服务器端用的是TNonblockingServerTransport。所以open需要在我的代码中重写。