从网络资源搜索建议到快速搜索框

时间:2010-11-24 08:04:17

标签: android json networking quick-search

我正在应用程序中构建搜索,需要有一种方法可以将我从服务器获取的建议作为JSON数组放入快速搜索框下方显示的建议列表中。

有一种简单的方法可以让快速搜索框读取这些资源吗?

目前我正在尝试使用ContentProvider,但接口方法清楚地表明应该查询数据库以获取建议。我想如果您正在搜索存储在应用程序中的数据,那么使用ContentProvider是正确的方法。但是,我不太确定,如果您需要查询网络资源,这是正确的方法。

将我从网络上获得的建议保存到本地数据库是没有意义的,因为建议及其命中率会不时变化。

有人有这个问题吗?或者可以指出我在类似问题的方向?我在堆栈上找不到提到网络建议的问题。

3 个答案:

答案 0 :(得分:12)

在developer.android.com上找到解决方案:

如果您有来自网络位置的建议,则可以在从服务器获得结果时动态构建游标。

这包含在ContentProvider的query()方法中:

String[] columns = {
   BaseColumns._ID, 
   SearchManager.SUGGEST_COLUMN_TEXT_1, 
   SearchManager.SUGGEST_COLUMN_INTENT_DATA
};

MatrixCursor cursor = new MatrixCursor(columns);

for (int i = 0; i < arr.length(); i++)
{
  String[] tmp = {Integer.toString(i), arr.getString(i), arr.getString(i)};
  cursor.addRow(tmp);
}
return cursor;

光标在快速搜索框中用于填写建议列表。

答案 1 :(得分:11)

以下是SearchView的示例,其中包含来自网络服务的建议(我使用了Retrofit):

@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_search_activity, menu);

        final SearchView searchView = (SearchView) MenuItemCompat.getActionView(menu.findItem(R.id.search));

        final CursorAdapter suggestionAdapter = new SimpleCursorAdapter(this,
                android.R.layout.simple_list_item_1,
                null,
                new String[]{SearchManager.SUGGEST_COLUMN_TEXT_1},
                new int[]{android.R.id.text1},
                0);
        final List<String> suggestions = new ArrayList<>();

        searchView.setSuggestionsAdapter(suggestionAdapter);
        searchView.setOnSuggestionListener(new SearchView.OnSuggestionListener() {
            @Override
            public boolean onSuggestionSelect(int position) {
                return false;
            }

            @Override
            public boolean onSuggestionClick(int position) {
                searchView.setQuery(suggestions.get(position), false);
                searchView.clearFocus();
                doSearch(suggestions.get(position));
                return true;
            }
        });

        searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {

            @Override
            public boolean onQueryTextSubmit(String query) {
                return false;
            }

            @Override
            public boolean onQueryTextChange(String newText) {

                MyApp.autocompleteService.search(newText, new Callback<Autocomplete>() {
                    @Override
                    public void success(Autocomplete autocomplete, Response response) {
                        suggestions.clear();
                        suggestions.addAll(autocomplete.suggestions);

                        String[] columns = {
                                BaseColumns._ID,
                                SearchManager.SUGGEST_COLUMN_TEXT_1,
                                SearchManager.SUGGEST_COLUMN_INTENT_DATA
                        };

                        MatrixCursor cursor = new MatrixCursor(columns);

                        for (int i = 0; i < autocomplete.suggestions.size(); i++) {
                            String[] tmp = {Integer.toString(i), autocomplete.suggestions.get(i), autocomplete.suggestions.get(i)};
                            cursor.addRow(tmp);
                        }
                        suggestionAdapter.swapCursor(cursor);
                    }

                    @Override
                    public void failure(RetrofitError error) {
                        Toast.makeText(SearchFoodActivity.this, error.getMessage(), Toast.LENGTH_SHORT).show();
                        Log.w("autocompleteService", error.getMessage());
                    }
                });
                return false;
            }
        });

        return true;
}

答案 2 :(得分:0)

不知道人们是否还需要这个。以防万一,对于未来的搜索者。我的SearchView建议有问题,但没有显示出来。搜遍后,我找到了解决方案。我使用Volley作为我的内容提供者类(从web api获取建议),这似乎不适合Android的内容提供者框架。我还发现我的内容提供程序在UI线程中运行。因此,如果我在其中同步使用Volley的Future,它会减慢(阻止)UI,直到请求返回(超时)。此外,我在网上发现了Volley的Future请求应该在其他线程中运行的信息,例如在异步任务中运行,以使其运行良好。因此,它没有解决我的问题,因为如果我必须使用它(在异步任务中),我会首先使用Volley的正常(异步)请求(这就是我当时使用的)。 我做的是这个:

  1. 在我的ContentProvider子类中,我定义了一个侦听器接口:

    public interface ResultListener {

      void onProviderResult(Cursor mCursor);
    
      void onProviderError(String errorMsg);
    

    }

  2. 在我的活动中(实现了LoaderCallbacks),我也在接口上面实现了。

  3. 在我的单例应用程序类中,我定义了一个静态变量,它用作瞬态数据及其getter / setter方法:

    private static HashMap transientData = new HashMap();

    public static Object getTransientData(String objectName){

    return transientData.get(objectName);
    

    }

    public static void setTransientData(String objectName,Object object){

    transientData.put(objectName, object);
    

    }

  4. 现在的逻辑: 在活动中,在调用 getSupportLoaderManager()。initLoader(...)之前, 我打电话给 MyApplication.setTransientData(&#34;请求者&#34;,这个)
  5. 在我的内容提供商的课程中,在凌空请求的onResponse回调中,我这样做了:

    public void onResponse(JSONArray response){

      ...
    
      ResultListener requestor =   (ResultListener)TheApplication.getTransientData("requestor");
    
      if (requestor!=null) requestor.onProviderResult(mCursor);
    

    }

    这样当凌空请求返回时,它将触发请求者的回调方法,向其传递填充了响应数据的光标,然后请求者(活动)通过调用以下方式通知游标适配器:       adapter.swapCursor(C);       adapter.notifyDataSetChanged();

    希望这有助于某人。 祝福。