我正在尝试使用从Socket检索的InputStream创建一个新的ObjectInputStream。这是我的代码:
这是我的MessageGetterSender类的构造函数。该程序没有到达Checkpoint 4。
public MessageGetterSender(Socket socket) {
System.out.println("MessageGetterSender: Checkpoint 1");
this.socket = socket;
// Get input and output streams
try {
System.out.println("MessageGetterSender: Checkpoint 2");
InputStream is = socket.getInputStream();
System.out.println("MessageGetterSender: Checkpoint 3");
this.in = new ObjectInputStream(is);
System.out.println("MessageGetterSender: Checkpoint 4");
} catch (IOException ioe) {
System.out.println("Could not get ObjectInputStream on socket: " + socket.getLocalPort());
}
try {
this.out = new ObjectOutputStream(socket.getOutputStream());
} catch (IOException ioe) {
System.out.println("Could not get ObjectOutputStream on socket: " + socket.getLocalPort());
}
System.out.println("MessageGetterSender: Checkpoint 5");
}
我正在从连接到服务器的类中实例化一个新的MessageGetterSender对象以获取套接字。这是相关的代码。它是InstantMessageClass的构造函数,即实例化MessageGetterSender对象的类:
public InstantMessageClient(String username) {
try {
socket = new Socket("localhost", 5555);
} catch (IOException ioe) {
System.out.println("Error: Could not connect to socket on port: " + serverPort);
}
messageGetterSender = new MessageGetterSender(socket);
...
由于代码没有执行到Checkpoint 4但它确实到达了Checkpoint 3,我很确定ObjectInputStream的实例化是罪魁祸首。我无法弄清楚为什么。有任何想法吗?谢谢你的帮助。
答案 0 :(得分:43)
只是为了扩展FatGuy对其他发现此问题的Google员工的回答;这个“鸡和鸡蛋问题”的解决方案是让每一方首先打开输出流,刷新输出流,然后打开输入流。
ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
out.flush();
ObjectInputStream in = new ObjectInputStream(socket.getInputStream());
答案 1 :(得分:42)
构造ObjectInputStream
时,在构造函数中,类尝试读取连接另一端的关联ObjectOutputStream
已写入的标头。在读取该标题之前,它不会返回。因此,如果您看到构造函数“挂起”,那是因为套接字的另一端未使用ObjectOutputStream
,或者尚未刷新数据。
答案 2 :(得分:1)
您还可以延迟ObjectInputStream的初始化,直到基础流中的数据可用。
无论流初始化顺序如何,此方法都有效,如果通道的一端位于您无法更改的库代码中,则此方法尤其有用。