0

我有一个 AsyncTask (AddressUpdaterTask),它从 AutoCompleteTextView 获取文本,对其进行解析,然后显示地址的下拉列表,就像您在谷歌地图中键入地址时一样。每次用户键入一个字符时,都会执行一个新的 AddressUpdaterTask。我的问题是这效率不高,因为当用户开始键入时,如果他键入的速度足够快,则会执行一个新的 AsyncTask 而之前的 AsyncTask 可能尚未完成执行。

所以我这样做了:

AddressUpdaterTask addressUpdaterTask = new AddressUpdaterTask();

其中 addressUpdaterTask 是全局的,当用户键入一个字符时,我这样做:

addressUpdaterTask.cancel(true);
addressUpdaterTask.execute();

所以我可以操作 AddressUpdaterTask 的一个实例。但是这样它不起作用:(我怎样才能实现我想要的?

编辑:这是我的代码:

public class RegisterDoctor extends Activity {

        .
        .
        .

    private AutoCompleteTextView addressField;
    private ArrayAdapter<String> adapter;
    private Filter filter;
    private List<SimpleAddress> simpleAddresses = new ArrayList<SimpleAddress>();
    private SimpleAddress currentAddress = null;
    AddressUpdaterTask addressUpdaterTask = new AddressUpdaterTask();



    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.register_doctor);

        .
        .
        .


        //AutoCompleteTextView Creation
        addressField = (AutoCompleteTextView) findViewById(R.id.addressEditText);
        addressField.setThreshold(ADDRESS_TRESHOLD);
        addressField.setHint("Οδός Αριθμός, Περιοχή");
        filter = new Filter() {
            @Override
            protected void publishResults(CharSequence constraint, FilterResults results) {
                adapter.notifyDataSetChanged();
            }

            @Override
            protected FilterResults performFiltering(CharSequence constraint) {
                Log.i("FILTER_ACTION", "Filter:" + constraint);
                if (constraint != null && constraint.length() > ADDRESS_TRESHOLD) {
                    Log.i("FILTER_ACTION", "doing a search ..");
                    addressUpdaterTask.cancel(true);
                    addressUpdaterTask.execute();
                }
                return null;
            }
        };
        adapter = new ArrayAdapter<String>(this, android.R.layout.simple_dropdown_item_1line) {
            public android.widget.Filter getFilter() {
                return filter;
            }
        };
        addressField.setAdapter(adapter);
        adapter.setNotifyOnChange(false);
        addressField.setOnItemClickListener(itemClicked);

        .
        .
        .

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


    private AdapterView.OnItemClickListener itemClicked = new AdapterView.OnItemClickListener() {
        @Override
        public void onItemClick(AdapterView<?> parent, View arg1, int pos, long id) {
            currentAddress = simpleAddresses.get(pos);
            Log.i("ITEM_CLICKED", simpleAddresses.get(pos).getFormatted_address());
        }
    };

        .
        .
        .

    public class AddressUpdaterTask extends AsyncTask<Void, Void, Void> {

        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            //adapter.clear();
        }


        @Override
        protected Void doInBackground(Void... voids) {
            try {
                simpleAddresses = new JsonHelper().getAddresses(addressField.getText().toString());
            } catch (NullPointerException e) {
            }
            return null;
        }
@Override
        protected void onPostExecute(Void aVoid) {
            int size = simpleAddresses.size();

            if (size > 0) {
                adapter.clear();
                for (int i = 0; i < size; i++) {
                    adapter.add(simpleAddresses.get(i).getFormatted_address());
                    Log.i("ADDRESS_ADDED", simpleAddresses.get(i).getFormatted_address());
                }
                adapter.notifyDataSetChanged();
                addressField.showDropDown();
            }
            super.onPostExecute(aVoid);
        }
    }
        .
        .
        .
}

编辑2:这是从用户输入=> 6个字符(我设置的阈值)开始的logcat,并且可以执行AsyncTask:

10-11 18:56:59.795    1465-1494/com.user.project I/FILTER_ACTION: Filter:panepi
10-11 18:57:02.695    1465-1494/com.user.project I/FILTER_ACTION: Filter:panepis
10-11 18:57:02.695    1465-1494/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:06.544    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepist
10-11 18:57:06.544    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:06.555    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:09.004    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepisti
10-11 18:57:09.004    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:09.014    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:09.904    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepistim
10-11 18:57:09.904    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:09.914    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:10.194    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepistimi
10-11 18:57:10.194    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:10.194    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:11.865    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepistimio
10-11 18:57:11.865    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:11.954    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:12.196    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepistimiou
10-11 18:57:12.196    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:12.295    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
10-11 18:57:14.795      404-966/com.android.inputmethod.latin E/ActivityThread: Failed to find provider info for com.android.inputmethod.latin.dictionarypack
10-11 18:57:14.805      404-966/com.android.inputmethod.latin E/BinaryDictionaryGetter: Could not find a dictionary pack
10-11 18:57:14.815    1465-1496/com.user.project I/FILTER_ACTION: Filter:panepistimiou
10-11 18:57:14.815    1465-1496/com.user.project I/FILTER_ACTION: doing a search ..
10-11 18:57:14.815    1465-1496/com.user.project W/Filter: An exception occured during performFiltering()!
        java.lang.IllegalStateException: Cannot execute task: the task has already been executed (a task can be executed only once)
        at android.os.AsyncTask.executeOnExecutor(AsyncTask.java:578)
        at android.os.AsyncTask.execute(AsyncTask.java:534)
        at com.user.project.RegisterDoctor$2.performFiltering(RegisterDoctor.java:104)
        at android.widget.Filter$RequestHandler.handleMessage(Filter.java:234)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:137)
        at android.os.HandlerThread.run(HandlerThread.java:60)
4

1 回答 1

0

在实现自动完成时,您通常会引入一些延迟,以便在没有返回结果的情况下多次调用服务。

尝试添加一个变量(可能是静态的)来跟踪用户何时输入他们的第一个输入,并且仅在经过一定时间段(500-750ms 相当常见)后才执行搜索。这是一个更好的用户体验,但不能解决您最初的问题。

对于您的“双重执行”问题,只需设置一个条件,在请求开始时设置为 true,在请求完成时设置为 false,并且在设置变量时不允许另一个请求开始。

像这样的东西:

bool requestInProgress = false;

...

@Override
protected void onPreExecute() {
    super.onPreExecute();
    requestInProgress = true;
}

protected void onPostExecute(Void aVoid) {
    int size = simpleAddresses.size();

    if (size > 0) {
        adapter.clear();
        for (int i = 0; i < size; i++) {
            adapter.add(simpleAddresses.get(i).getFormatted_address());
            Log.i("ADDRESS_ADDED", simpleAddresses.get(i).getFormatted_address());
        }
        adapter.notifyDataSetChanged();
        addressField.showDropDown();
    }
    super.onPostExecute(aVoid);
    requestInProgress = false;
}

以上...

@Override
protected FilterResults performFiltering(CharSequence constraint) {
    Log.i("FILTER_ACTION", "Filter:" + constraint);
    if (constraint != null && constraint.length() > ADDRESS_TRESHOLD 
           && !requestInProgress) {
        Log.i("FILTER_ACTION", "doing a search ..");
        addressUpdaterTask.cancel(true);
        addressUpdaterTask.execute();
    }
    return null;
}
于 2013-10-11T18:50:44.040 回答