在线程中引用对象时使用java NPE?
我想使用socket对象。这似乎是一种使用它的“标准实践”,或者是一个线程中与互联网相关的任何东西
我想通过按下按钮在这个socket上发送一些数据,所以我自然需要在onClick事件处理程序中引用这个socket对象。socket将在mainActivity的onCreate方法中启动的线程中构造。以下是代码:
package com.mypack.app.demo;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
import 安卓.app.Activity;
import 安卓.os.Bundle;
import 安卓.view.View;
import 安卓.widget.EditText;
public class MainActivity extends Activity {
private Socket socket;
private static final int SERVERPORT = 12345;
private static final String SERVER_IP = "111.11.111.11";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
new Thread(new ClientThread()).start();
}
public void onClick(View view) {
try {
EditText et = (EditText) findViewById(R.id.EditText01);
String str = et.getText().toString();
PrintWriter out = new PrintWriter(new BufferedWriter(
new OutputStreamWriter(socket.getOutputStream())),
true);
out.println(str);
out.flush();
out.close();
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
}
class ClientThread implements Runnable {
@Override
public void run() {
try {
InetAddress serverAddr = InetAddress.getByName(SERVER_IP);
socket = new Socket(serverAddr, SERVERPORT);
} catch (UnknownHostException e1) {
e1.printStackTrace();
} catch (IOException e1) {
e1.printStackTrace();
}
}
}
}
奇怪的是,你可能认为在调用onClick时,socket将被实例化。但我在日志中发现了以下错误:
07-09 19:31:42.911 2243-2243/com.myapp.app.mydemo D/OpenGLRenderer: Enabling debug mode 0 07-09 19:32:31.961 2243-2243/com.myapp.app.mydemo W/System.err: java.lang.NullPointerException at com.myapp.app.mydemo.MainActivity.onClick(MainActivity.java:62)
error msg所指的具体行是:
new OutputStreamWriter(socket.getOutputStream()))
这个错误非常持久,每当我点击屏幕上的按钮时就会发生。显然,由于某种原因,“socket”对象在我按下按钮之前无法存活足够长的时间
我该怎么解决这个问题?有人建议我使用某种同步块之类的东西。我对java有点陌生,所以我该怎么做呢
# 1 楼答案
您的代码有很多问题:
1)不能在onClick中向套接字创建OutputWriter,因为这样会在主线程上向套接字写入。这将导致NetworkOnMainThread错误
2)如果在创建套接字时发生任何异常,您将得到一个NPE,因为没有写入套接字变量
3)这里的套接字需要易失性,才能在两个线程上可靠地访问它。(尽管你根本不应该在主线程上访问它)
4)即使它是易变的,也存在一个争用条件,即可以在套接字构造函数完成之前单击按钮,这也会允许套接字变量为null