首页 新闻 会员 周边

多线程中applicationContext.getBean方法未返回也未报错

0
悬赏园豆:50 [已解决问题] 解决于 2017-07-08 22:14
public class TcpServer implements ApplicationContextAware{
    private OpenbankExecutor executor;
    private int port;
    private PacketHandler packetHandler;
    private ServerSocket serverSocket;
    private Socket socket;
    private ApplicationContext applicationContext;
    private final Logger log = LoggerFactory.getLogger(getClass());
    public void init() throws IOException {
        serverSocket = new ServerSocket(port);
        log.debug("TcpServer 成功启动");
        while(true) {
            socket = serverSocket.accept();
            System.out.println("收到socket请求");
            executor.execute(new Runnable() {

                @Override
                public void run() {
                    try {
                        Map context = packetHandler.handle(socket.getInputStream());
                        String tranCode = (String) context.get("tranCode");
                        Assert.hasText(tranCode);
                        Transaction transaction = (Transaction) applicationContext.getBean(tranCode);
                        Template template = (Template) applicationContext.getBean(transaction.getTemplate());
                        log.debug("transaction: "+transaction.getId());
                        log.debug("transaction: "+template.getId());
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            });
        }
    }

走到Transaction transaction = (Transaction) applicationContext.getBean(tranCode);就没有往下走了,没有报错。非多线程方式下这个方法是没有问题的,比如我将这一段放到executor.execute(new Runnable() {之前就没有问题,求解

注销111的主页 注销111 | 初学一级 | 园豆:46
提问于:2017-07-03 23:19
< >
分享
最佳答案
0

不太懂,多线程才有问题会不会是哪儿死锁了

收获园豆:50
苍枫露雨 | 小虾三级 |园豆:1027 | 2017-07-04 19:34

不清楚原因,我重新写了一个简单的多线程场景的测试并没有复现这个问题,等我搞清楚原因再来结贴

注销111 | 园豆:46 (初学一级) | 2017-07-04 19:36

前段时间在准备面试,所以这时候才来结贴,唉面试失败了

 搞懂了,是这样的:TcpServer这个类是我通过spring ioc配置的bean:

<bean id="tcpServer" class="com.openbank.portal.server.TcpServer" init-method="init">

</bean>

spring在加载这个类的时候,就使用init方法来初始化,结果我里面启了一个serversocket并且进入了serverSocket.accept()方法,结果就在这里阻塞了spring继续实例化剩下还没有来得及实例的bean。

我改了一下,serverSocket.accept()之前,我起了一个新的线程来做这件事情,就不会有这个问题了。

 

我debug了一下spring的源代码,卡在了一个synchronized方法,一直没有拿到锁。反正大概意思就是这样的。

注销111 | 园豆:46 (初学一级) | 2017-07-08 22:13

@捡肉王: 

哦,所以是因为serverSocket.accept()一直处于等待状态,导致锁没有释放吗?

苍枫露雨 | 园豆:1027 (小虾三级) | 2017-07-10 15:25

@苍枫露雨: 嗯哼 我是这样认为的

注销111 | 园豆:46 (初学一级) | 2017-07-10 15:31
清除回答草稿
   您需要登录以后才能回答,未注册用户请先注册