我正在将我从theneboston.com学到的客户端 - 服务器应用程序转换为Android应用程序,它是一个使用套接字(TCP连接)进行通信的即时消息应用程序,在我的Android应用程序中,一个设备是服务器而另一个设备是是客户。现在,在模拟器上运行服务器端应用程序时出现问题,应用程序冻结了大约40秒,然后弹出一条错误消息说:
Activity ServerIM(在应用程序Android Server Instant Messaging中) 没有回复
我认为问题在于创建ServerSocket。我真的需要这个,所以我很感激任何人都可以帮助我。
这是我的ServeIM
Android类:
package com.example.android.server.instant.messenger;
import java.io.EOFException;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ScrollView;
import android.widget.TextView;
import android.view.View.OnClickListener;
public class ServerIM extends Activity {
private ScrollView scrollView;
private EditText userText;
private Button connectButton;
private Button sendButton;
private TextView chatLog;
private ObjectOutputStream output;
private ObjectInputStream input;
private ServerSocket server;
private Socket connection;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_server_im);
scrollView = (ScrollView) findViewById(R.id.ScrollView);
userText = (EditText) findViewById(R.id.user_Text);
connectButton = (Button) findViewById(R.id.button_connect);
sendButton = (Button) findViewById(R.id.button_send);
chatLog = (TextView) findViewById(R.id.chat_Log);
ableToType(false);
connectButton.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
showMessage("1"); // this message doesn't appends to the
// TextView
startRunning();
}
});
}
public void startRunning() {
try {
(new Thread(new Runnable() {
public void run() {
try {
server = new ServerSocket(5678, 100);
} catch (IOException e) {
e.printStackTrace();
}
}
})).start();
while (true) {
try {
waitForConnection();
setupStreams();
whileChatting();
} catch (EOFException eofException) {
showMessage("\n Server ended the connection!");
} finally {
closeCrap();
}
}
} catch (IOException ioException) {
ioException.printStackTrace();
}
}
// wait for connection, then display connection information
private void waitForConnection() throws IOException {
showMessage("Waiting for someone to connect... \n");
connection = server.accept();
showMessage("Now connected to "
+ connection.getInetAddress().getHostName());
}
// get stream to send and recieve data
private void setupStreams() throws IOException {
output = new ObjectOutputStream(connection.getOutputStream());
output.flush();
input = new ObjectInputStream(connection.getInputStream());
showMessage("\n Streams are now setup! \n");
}
// during the chat conversation
private void whileChatting() throws IOException {
String message = " You are now connected! ";
sendMessage(message);
ableToType(true);
do {
// have a conversation
try {
message = (String) input.readObject();
showMessage("\n" + message);
} catch (ClassNotFoundException classNotFoundException) {
showMessage("\n idk wtf that user sent! ");
}
} while (!message.equals("CLIENT - END"));
}
// close streams ans sockets after you are done chatting
private void closeCrap() {
showMessage("\n Closing Connections... \n");
ableToType(false);
try {
output.close();
input.close();
connection.close();
} catch (IOException ioException) {
ioException.printStackTrace();
}
}
// send message to client
private void sendMessage(String message) {
try {
output.writeObject("SERVER - " + message);
output.flush();
showMessage("\n Server - " + message);
} catch (IOException ioException) {
chatLog.append("\n ERROR: I can't send this message");
}
}
// update chatWindow
private void showMessage(final String text) {
final Handler myHandler = new Handler();
(new Thread(new Runnable() {
public void run() {
myHandler.post(new Runnable() {
public void run() {
chatLog.append(text + "\n");
}
});
}
})).start();
}
// let the user type stuff into their box
private void ableToType(final boolean tof) {
final Handler myHandler = new Handler();
(new Thread(new Runnable() {
public void run() {
myHandler.post(new Runnable() {
public void run() {
userText.setEnabled(tof);
}
});
}
})).start();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_server_im, menu);
return true;
}
}
这是XML文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<ScrollView
android:id="@+id/ScrollView"
android:layout_width="fill_parent"
android:layout_height="150dp"
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true" >
<TextView
android:id="@+id/chat_Log"
android:layout_width="wrap_content"
android:layout_height="339dp"
android:text="" />
</ScrollView>
<EditText
android:inputType="text"
android:id="@+id/user_Text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_below="@+id/ScrollView"
android:layout_marginTop="39dp"
android:ems="10" />
<Button
android:id="@+id/button_send"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBottom="@+id/user_Text"
android:layout_toRightOf="@+id/user_Text"
android:minWidth="100dip"
android:text="Send" />
<Button
android:id="@+id/button_connect"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/button_send"
android:layout_alignRight="@+id/button_send"
android:layout_below="@+id/button_send"
android:text="Connect" />
</RelativeLayout>
和Manifest:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.server.instant.messenger"
android:versionCode="1"
android:versionName="1.0" >
<uses-permission android:name="android.permission.INTERNET" />
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="15" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".ServerIM"
android:label="@string/title_activity_server_im" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
答案 0 :(得分:1)
ServerSocket.accept()
函数正在UI线程中运行,并将阻塞直到它获得连接。
您在另一个主题中创建ServerSocket
时有正确的想法,但您必须在后台运行accept()
功能。
答案 1 :(得分:0)
事件处理程序中有while (true)
(ServerIM.java:68),这使得EDT挂起。