Java套接字提供流的过早结束

时间:2011-02-02 16:21:44

标签: java sockets xmpp

我正在设置连接到XMPP服务器的彗星服务器。这是它如何下降:

客户端连接彗星服务器,其中包括打开套接字连接:

try {
        radio = new Socket("server", 5222);
        out = new PrintWriter(radio.getOutputStream(), true);
        in = new BufferedReader(new InputStreamReader(radio.getInputStream()));
    } catch (UnknownHostException e) {
        System.out.println("Unknown host: "+e);
        error = e.toString();
    } catch(IOException e) {
        System.out.println("IO error: "+e);
        error = e.toString();
    }

接下来,启动一个线程,它等待来自套接字的数据:

public void run() {
        System.out.println("Thread started.");
        String data;
        String error;
        Client remote;
        Client client;
        while(!done) {
            data = this.output();
            remote = bayeux.getClient(remoteId);
            client = bayeux.getClient(clientId);
            if(data!=null) {
                Map<String, Object> packet = new HashMap<String, Object>();
                packet.put("xml", data);
                remote.deliver(client, "/radio/from", packet, null);
            }
            error = this.error();
            if(error!=null) {
                Map<String, Object> packet = new HashMap<String, Object>();
                packet.put("details", error);
                remote.deliver(client, "/radio/error", packet, null);
            }
            /* try { 
                Thread.sleep(500); 
            }  
            catch (InterruptedException e ) {
                System.out.println("Interrupted!"); } */
        }

        try {
            in.close();
            out.close();
            radio.close();
        } catch(IOException e) {
            System.out.println("Error disconnecting: "+e);
            error = e.toString();
        }
        System.out.println("Thread stopped.");
    }

    public String output() {
        try {
            String data = in.readLine();
            System.out.println("From: "+data);
            if(data==null) {
                System.out.println("End of stream!");
                try {
                    Thread.currentThread().sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                //error = "End of stream.";
                //this.disconnect();
            }
            return data;
        } catch (IOException e) {
            error = e.toString();
            System.out.println("IO error! "+e);
        }
        return null;
    }

从客户端收到的任何输入都会转发到XMPP服务器:

    public void input(String xml) {
        System.out.println("To: "+xml);
        out.println(xml);
    }

所以这就是问题所在。客户端打开连接并将正确的XML发送到XMPP服务器以启动流。 {I}应该挂起in.readLine();,直到收到服务器的响应。收到后,in.readLine();开始一遍又一遍地返回null。这不应该发生;它应该挂起,直到它收到数据。服务器似乎不太可能关闭我,它还没有发送</stream:stream>来表示XMPP流的结束。关于什么可能是问题的任何想法?

感谢您的帮助!

1 个答案:

答案 0 :(得分:0)

请记住,XMPP连接可以并且将在一次读取中为您提供不完整的节或多个节。如果您的COMET连接期望您传递的内容是格式良好的XML,那么您将遇到问题。同样,XMPP不是新行终止的,所以我不确定为什么你期望readLine()非常有用。

接下来,您是否在同一线程上的两个不同套接字上进行同步I / O操作?听起来像死锁的秘诀。如果你坚持沿着这条路走下去(而不仅仅是使用BOSH),我强烈建议你使用NIO,而不是睡觉。