获取文本的方法必须从UI线程调用(Android Studio)

3
我是一个有用的助手,可以为您翻译文本。

我正在尝试创建一个应用程序的登录功能。但是我遇到了一个问题。

这是我的代码:

package com.forgetmenot.loginregister;


import java.util.ArrayList;
import java.util.List;

import org.apache.http.NameValuePair;
import org.apache.http.message.BasicNameValuePair;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import android.app.Activity;
import android.content.Intent;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

public class MainActivity extends Activity {
    EditText uname, password;
    Button submit;
    // Creating JSON Parser object
    JSONParser jParser = new JSONParser();
    private static final String TAG = "Login";

    JSONObject json;
    private static String url_login = "http://localhost:8080/ForgetMeNotApplication/Login";
   //JSONArray incoming_msg = null;
   @Override
   protected void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);
       findViewsById();
       submit.setOnClickListener(new View.OnClickListener() {

           @Override
           public void onClick(View arg0) {
               // execute method invokes doInBackground() where we open a Http URL connection using the given Servlet URL
               //and get output response from InputStream and return it.
                new Login().execute();

           }
       });
    }
    private void findViewsById() {

        uname = (EditText) findViewById(R.id.txtUser);
        password = (EditText) findViewById(R.id.txtPass);
        submit = (Button) findViewById(R.id.login);
    }
    private class Login extends AsyncTask<String, String, String>{

        @Override
        protected String doInBackground(String... args) {
            // Getting username and password from user input

            String username = uname.getText().toString();
            String pass = password.getText().toString();

            List<NameValuePair> params = new ArrayList<NameValuePair>();
            params.add(new BasicNameValuePair("u",username));
            params.add(new BasicNameValuePair("p",pass));
            json = jParser.makeHttpRequest(url_login, "GET", params);
            String s=null;

            try {
                s= json.getString("info");
                Log.d("Msg", json.getString("info"));
                if(s.equals("success")){
                    Intent login = new Intent(getApplicationContext(), home.class);
                    login.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
                    startActivity(login);
                    finish();
                }
            } catch (JSONException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            return null;
        }

    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();
        if (id == R.id.action_settings) {
            return true;
        }
        return super.onOptionsItemSelected(item);
    }
}

Android studio提示在以下代码行中,getText()方法必须从UI线程调用:uname.getText().toString();password.getText().toString(); 有什么可能的解决方案吗?
5个回答

14
尝试通过execute(param1, param2, ..., paramN)方法将您的值传递给Login AsyncTask
submit.setOnClickListener(new View.OnClickListener() {

           @Override
           public void onClick(View arg0) {

               String username = uname.getText().toString();
               String pass = password.getText().toString();
               new Login().execute(username, pass);

           }
       });
    }
    private void findViewsById() {

        uname = (EditText) findViewById(R.id.txtUser);
        password = (EditText) findViewById(R.id.txtPass);
        submit = (Button) findViewById(R.id.login);
    }
    private class Login extends AsyncTask<String, String, String>{

        @Override
        protected String doInBackground(String... args) {
            // Getting username and password from user input

            String username = args[0];
            String pass = args[1];

2

usernamepass设置为login类变量,并重写onPreExcecute()方法,执行以下操作:

    @Override
    protected void onPreExecute() {

        username = uname.getText().toString();
        pass = password.getText().toString();

    }

1

您正在从后台线程访问UI线程:

String username = uname.getText().toString();
String pass = password.getText().toString();

你需要做的就是将用户名/密码字符串传递给后台任务构造函数,或直接传递给执行方法。如果它们是必需的(像你的一样),我更喜欢在构造函数中定义它们。
像这样定义你的LoginTask:
String uname;
String password;
public Login(String username, String password({
    this.uname = username;
    this.password = password;
}

然后在doInBackground()中使用成员变量。

List<NameValuePair> params = new ArrayList<NameValuePair>();
params.add(new BasicNameValuePair("u",this.username));
params.add(new BasicNameValuePair("p",this.pass));
json = jParser.makeHttpRequest(url_login, "GET", params);

编辑 - 那么您的新 Login().execute() 调用将更像这样

new Login(uname.getText().toString(), password.getText().toString()).execute();

1
除非有我不知道的改变,这应该不是个问题。UI元素无法从后台更新,但访问它们的getter从未成为问题。 无论如何,您可以通过向您的AsyncTask添加构造函数来解决此问题,该构造函数将获取两个字符串,然后在创建任务时发送它们。
private class Login extends AsyncTask<String, String, String>{

    // member variables of the task class
    String uName, pwd
    public Login(String userName, String password) {
        uName = userName;
        pwd = password;
    }

    @Override
    protected String doInBackground(String... args) {...}

并在您的onClick()中传递它们。
 @Override
 public void onClick(View arg0) {
     // execute method invokes doInBackground() where we open a Http URL connection using the given Servlet URL
     //and get output response from InputStream and return it.

     // pass them here
     new Login(uname.getText().toString(), password.getText().toString()).execute();
 }

在我的API 19设备上,我发现getText()似乎可以工作,但其他方法,比如TextureViewgetBitmap()会崩溃。 - VinceFior
我认为getText可以在工作线程中访问,这只是IDE提示,而不是运行时异常。 - Ninja

0

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接