1

editText 用の textWatcher があります。アイデアは次のとおりです。

2 つのリスト ビューがあり、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();
}

}

データの保持方法が間違っていて、検索アルゴリズムも良くないことをはっきりと理解しています。しかし、私の質問は次のとおりです。

検索の実装にスレッドを使用してフリーズを回避することは可能ですか? インターフェイスのフリーズを何とか回避するために、両方のサーチャーをそれぞれのスレッドに配置できますか? 可能であれば、スレッドを使用できるコード内の最適な場所を教えてください。

前もって感謝します!

4

2 に答える 2

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();
于 2016-09-13T15:00:21.693 に答える
2

バックグラウンド スレッドで長いタスクを実行するための最良の方法。これを行うには、asynctask を使用できます。

于 2016-09-13T14:49:09.170 に答える