1

在阅读了几篇关于 ASYNCTASK 的帖子并在开发者页面上阅读之后,我想出了以下代码,并将其分配给一个按钮:

private class TalkToServerTask extends AsyncTask<String, Void, String> {

    @Override
    protected String doInBackground(String... params) {
        String response = "";
        try {
            InetAddress serverAddr = InetAddress.getByName(params[0]);
            Socket s = new Socket(serverAddr, Integer.valueOf(params[1]));

            PrintWriter out = new PrintWriter(new BufferedWriter(
                    new OutputStreamWriter(s.getOutputStream())), true);

            // WHERE YOU ISSUE THE COMMANDS

            out.println(params[2]);
            // BufferedReader input = new BufferedReader(
            // new InputStreamReader(s.getInputStream()));

            DataInputStream dataInputStream = null;
            dataInputStream = new DataInputStream(s.getInputStream());
            st = dataInputStream.readLine().toString();
            // String st = s.readLine();
            // st = input.readLine();
            // read line(s)

            s.close();
            return st;

        } catch (UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return response;
    }

    @Override
    protected void onPostExecute(String result) {
        serverresponse = result;
    }

}

想法是,当单击按钮时,ASyncTask 将单词“getDomains”发送到在我的服务器上运行的控制台应用程序,服务器对此进行操作并发送回一个字符串,其中包含在电子邮件服务器上创建的域列表。

我已经验证服务器正在接收命令“getdomains”,然后它又以管道分隔的域字符串进行回复。然而,问题是,我设置了一个 Toast 来弹出套接字事务的结果,而 toast 什么也没显示。如果我再次点击按钮,Toast 会显示域列表。对我来说,似乎套接字首先返回一个空的

这是按钮代码:

case R.id.btnDomains:
            SharedPreferences sp = PreferenceManager
                    .getDefaultSharedPreferences(this);
            IpAddress = sp.getString("ipaddress", "0.0.0.0");
            Serverport = sp.getString("tcpport", "12345");
            buttonpressed = "domains";
            // TalkToServerTask task = new TalkToServerTask();
            new TalkToServerTask().execute(IpAddress, Serverport, "getDomains");

            Intent buttonActivity = new Intent(MainActivity.this, Rules.class);
            buttonActivity.putExtra(MainActivity.DOMAINLIST, serverresponse);

            Toast.makeText(getApplicationContext(), serverresponse,
                    Toast.LENGTH_SHORT).show();

变量“serverresponse”最初显示为空,但随后显示服务器列表。

4

3 回答 3

3

正如 Thepoosh 提到的,AsyncTask 正在一个单独的线程上工作。

因此,当您第一次按下按钮时,线程正在执行并且尚未得到结果。

您应该做的是在 onPostExecute 方法中显示数据。此外,您应该将上下文传递给您的 AsyncTask。

public TalkToServerTask(Context context)  {
    this.context = context;
}

@Override
protected void onPostExecute(String result) {
    Intent buttonActivity = new Intent(context, Rules.class);
    buttonActivity.putExtra(MainActivity.DOMAINLIST, result);

    Toast.makeText(context.getApplicationContext(), result, Toast.LENGTH_SHORT).show();
}
于 2012-05-28T07:29:48.447 回答
0

你只需要稍微改变你的代码,

你需要移动这 3 行,

Intent buttonActivity = new Intent(MainActivity.this, Rules.class); 
buttonActivity.putExtra(MainActivity.DOMAINLIST, serverresponse); 
// and also the line to startActivity() as well.
Toast.makeText(getApplicationContext(), serverresponse, 
                    Toast.LENGTH_SHORT).show(); 

从你Switch..case到你写过onPostExecute的地方AsyncTaskserverresponse = result

那是因为 AsyncTask 在不同的线程中运行,而您在 Swtich 中执行...情况下,下一行代码不会在 AsyncTask 完成之前执行,所以所有依赖代码都应该写在任务的 onPostExecute .

于 2012-05-28T07:28:12.720 回答
0

AsynTask在与 UI 线程不同的线程上运行。所以有2个线程并行运行。当您尝试在主 UI 线程上显示列表时,AsynTask仍在准备获取列表,或者可能在其自己的线程中创建套接字。因此列表数据仍然是空的。

postexecute方法Asynctask在主 UI 线程上运行,因此在其中更新 UI 是安全且正确的。

于 2012-05-28T07:39:33.723 回答