Активность Android зависает при поиске из базы данных

У меня есть textWatcher для моего editText. Идея заключается в следующем:

У меня есть два представления списка, и они оба обновляются после каждого вставленного в поле ввода editText символа. Каким-то образом я не мог реализовать это, используя стандартный ArrayAdapter с методом фильтра. Поэтому я должен был сделать свою собственную реализацию поиска. В начале это казалось хорошим, но как только я добавил в свою базу данных более менее достаточно данных, интерфейс начал зависать.

Это вызов

inputSearch.addTextChangedListener(createAndReturnTextWatcher());

И это метод, который создает и возвращает textWatcher

private TextWatcher createAndReturnTextWatcher() {
    final List<String> list = dbExtractor.getDataForSearchAdapters();
    TextWatcher watcher1;

    watcher1 = new TextWatcher() {

        @Override
        public void onTextChanged(CharSequence cs, int arg1, int arg2, int arg3) {

            if (cs.length() == 0) {

                lv.setAdapter(null);
                lv2.setAdapter(null);

                dc.clickedOnce = true;
                dc.decrement();
                dc.checkDimCounter();
            } else {


                setSecondListData(list, String.valueOf(cs));
                setFirstListData(list, String.valueOf(cs));

                if (dc.clickedOnce) {
                    dc.increment();
                    dc.checkDimCounter();
                    dc.clickedOnce = false;
                }

            }

        }

        @Override
        public void beforeTextChanged(CharSequence arg0, int arg1, int arg2,
                                      int arg3) {
        }

        @Override
        public void afterTextChanged(Editable arg0) {

        }
    };

    return watcher1;
}

И это методы, которые выполняют фильтрацию

private void setSecondListData(List<String> inputData, String cs) {
    List<String> result = turkishSearcher.performFiltering(inputData, cs);
    ArrayAdapter turkAdapt = new ArrayAdapter(this, R.layout.dictionary_list_item, R.id.product_name,
            result);
    lv2.setAdapter(turkAdapt);
}

private void setFirstListData(List<String> inputData, String cs) {
    List<String> result = normSearcher.performFiltering(inputData, cs);
    ArrayAdapter turkAdapt = new ArrayAdapter(this, R.layout.dictionary_list_item, R.id.product_name,
            result);
    lv.setAdapter(turkAdapt);
}

И это класс, который фильтрует данные

public class TurkishSearcher extends Searcher{

@Override
int returnPosition() {
    return 1;
}

@Override
String reverse(String couples) {
    java.lang.String[] arr = couples.split(" — ");
    return arr[1]+ " — " + arr[0];
}

@Override
String replaceTurkish(String words) {

    if (checkWithRegExp(words)) {
        return words.toLowerCase().replaceAll("ç", "c").replaceAll("ğ", "g").replaceAll("ı", "i").
                replaceAll("ö", "o").replaceAll("ş", "s").replaceAll("ü", "u");
    } else return words;
}

public static boolean checkWithRegExp(String word){
    Pattern p = Pattern.compile("[öçğışü]", Pattern.CASE_INSENSITIVE);
    Matcher m = p.matcher(word);
    return m.find();
}

}

Я четко понимаю, что то, как я храню данные, неверно, и алгоритм поиска также не очень хорош. Но мой вопрос:

Можно ли избежать зависаний, используя потоки для реализации поиска. Могу ли я поместить оба поисковика в свои потоки, чтобы избежать зависаний интерфейса? Если это возможно, скажите, пожалуйста, лучшее место в коде, где я могу использовать потоки.

Заранее спасибо!

2 ответа

Решение

Вы никогда не должны блокировать основной поток с помощью операций с базой данных. Вместо этого используйте:

     new AsyncTask<Void, Void, Void>() {
        @Override
        protected Void doInBackground(Void... unusedParams) {
            // TODO: do your database stuff
            return null;
        }

        @Override
        protected void onPostExecute(Void aVoid) {
            // TODO: refresh UI elements, because thread finished
            super.onPostExecute(aVoid);
        }
    }.execute();

Лучший способ сделать длинные задачи в фоновом потоке. Вы можете использовать asynctask для этого.

Другие вопросы по тегам