当前位置:   article > 正文

android通过拼音搜索中文的功能_android 拼音搜索逻辑

android 拼音搜索逻辑

好几年没写博客了,很多知识不记是真的会忘记,以后还是保持写博客的习惯吧。坚持不一定成功,但放弃一定很舒服!(开玩笑(#^.^#))

回归正题,今天我要记录的是拼音搜索功能,我记得16年的时候做过这个功能。现在已经忘记很多细节了,所以这次好好写一写!

第一步:准备

第二步:分析功能并实现

  1. 很明显,这是两个功能,一个是中文转拼音,一个是查询过滤
  2. 下面我们先实现主要功能,查询过滤
  3. 创建布局文件 activity_main.xml,实现如下:
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:app="http://schemas.android.com/apk/res-auto"
  4. xmlns:tools="http://schemas.android.com/tools"
  5. android:layout_width="match_parent"
  6. android:layout_height="match_parent"
  7. tools:context=".MainActivity"
  8. android:padding="16dp">
  9. <EditText
  10. android:id="@+id/etSearchName"
  11. android:layout_width="match_parent"
  12. android:layout_height="50dp"
  13. android:hint="请输入拼音"
  14. android:padding="5dp"
  15. app:layout_constraintLeft_toLeftOf="parent"
  16. app:layout_constraintRight_toRightOf="parent"
  17. app:layout_constraintTop_toTopOf="parent" />
  18. <androidx.recyclerview.widget.RecyclerView
  19. android:id="@+id/recyclerView"
  20. android:layout_width="match_parent"
  21. android:layout_height="wrap_content"
  22. android:layout_marginTop="10dp"
  23. app:layout_constraintTop_toBottomOf="@+id/etSearchName"
  24. />
  25. </androidx.constraintlayout.widget.ConstraintLayout>

就一个搜索框和列表控件

接着创建列表适配器  SearchAdapter.java,实现了过滤器类Filterable

  1. public class SearchAdapter extends RecyclerView.Adapter<SearchAdapter.MyViewHolder> implements Filterable {
  2. private Context context;
  3. private List<String> list;
  4. //保存原有的数据
  5. private List<String> originalList;
  6. private OnItemListener onItemListener;
  7. void setOnItemListener(OnItemListener onItemListener){
  8. this.onItemListener = onItemListener;
  9. }
  10. private SearchFilter filter;
  11. SearchAdapter(Context context,List<String> list){
  12. this.context =context;
  13. this.list = list;
  14. originalList = list;
  15. }
  16. @NonNull
  17. @Override
  18. public SearchAdapter.MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
  19. View view = LayoutInflater.from(context).inflate(R.layout.item_rv_search,parent,false);
  20. return new MyViewHolder(view);
  21. }
  22. @Override
  23. public void onBindViewHolder(@NonNull SearchAdapter.MyViewHolder holder, final int position) {
  24. holder.tvName.setText(list.get(position));
  25. holder.itemView.setOnClickListener(new View.OnClickListener() {
  26. @Override
  27. public void onClick(View v) {
  28. if (onItemListener!=null){
  29. onItemListener.onItem(list.get(position));
  30. }
  31. }
  32. });
  33. }
  34. @Override
  35. public int getItemCount() {
  36. return list==null?0:list.size();
  37. }
  38. @Override
  39. public Filter getFilter() {
  40. if (filter==null){
  41. filter = new SearchFilter();
  42. }
  43. return filter;
  44. }
  45. class MyViewHolder extends RecyclerView.ViewHolder{
  46. private TextView tvName;
  47. MyViewHolder(@NonNull View itemView) {
  48. super(itemView);
  49. tvName = itemView.findViewById(R.id.tvName);
  50. }
  51. }
  52. interface OnItemListener{
  53. void onItem(String name);
  54. }
  55. class SearchFilter extends Filter{
  56. @Override
  57. protected FilterResults performFiltering(CharSequence constraint) {
  58. //输入框传来的数据 constraint
  59. //用于保存过滤的结果
  60. FilterResults filterResults = new FilterResults();
  61. if (constraint==null || constraint.length()==0){
  62. filterResults.values = originalList;
  63. filterResults.count = originalList.size();
  64. }else {
  65. List<String> fList = new ArrayList<>();
  66. String cons = constraint.toString().trim().toLowerCase();
  67. for (String s : originalList) {
  68. //从首位开始匹配
  69. if (s.startsWith(cons)){
  70. fList.add(s);
  71. }
  72. }
  73. filterResults.values = fList;
  74. filterResults.count = fList.size();
  75. }
  76. return filterResults;
  77. }
  78. @Override
  79. protected void publishResults(CharSequence constraint, FilterResults results) {
  80. list = (List<String>) results.values;
  81. notifyDataSetChanged();
  82. }
  83. }
  84. }

这段代码,重点是SearchFilter类的两个方法

  • performFiltering(CharSequence constraint)  接收输入数据,在这个方法实现处理逻辑
  • publishResults(CharSequence constraint, FilterResults results)  接收过滤后的结果

注意点:originalList 这个集合是用来保存原有数据的。因为list会随着搜索结果而变化,我们每次的过滤都是需要用到原有的数据。

适配器写好了,下面看看怎么使用它

  1. private RecyclerView recyclerView;
  2. private SearchAdapter adapter;
  3. private List<String> list;
  4. private EditText etSearchName;
  5. public static final String[] str = new String[]{
  6. "陈天丽","黄正","徐明"
  7. ,"李自成","林子祥","周星星"
  8. ,"周润发","林星辰","林青霞"
  9. ,"李赛凤","刘德华","胡歌"
  10. ,"霍建华","林心如","赵薇"
  11. ,"赵四","赵本山","郭德纲"
  12. };
  13. @Override
  14. protected void onCreate(Bundle savedInstanceState) {
  15. super.onCreate(savedInstanceState);
  16. setContentView(R.layout.activity_main);
  17. etSearchName = findViewById(R.id.etSearchName);
  18. recyclerView = findViewById(R.id.recyclerView);
  19. recyclerView.setLayoutManager(new LinearLayoutManager(this));
  20. list = new ArrayList<>();
  21. list.addAll(Arrays.asList(str));
  22. adapter = new SearchAdapter(this,list);
  23. recyclerView.setAdapter(adapter);
  24. adapter.setOnItemListener(new SearchAdapter.OnItemListener() {
  25. @Override
  26. public void onItem(String name) {
  27. etSearchName.setText(name);
  28. }
  29. });
  30. etSearchName.addTextChangedListener(new TextWatcher() {
  31. @Override
  32. public void beforeTextChanged(CharSequence s, int start, int count, int
  33. after) {
  34. }
  35. @Override
  36. public void onTextChanged(CharSequence s, int start, int before, int count) {
  37. adapter.getFilter().filter(etSearchName.getText().toString().trim());
  38. }
  39. @Override
  40. public void afterTextChanged(Editable s) {
  41. }
  42. });
  43. }

一顿RecyclerView的常规操作

这里需要注意的是这段代码  adapter.getFilter().filter(etSearchName.getText().toString().trim()); 这里就调用了我们写好的过滤器。

到此,我们的搜索过滤功能已经实现了。

看看运行效果:

接下来说说中文转拼音

首先需要修改数据源类型

  • 之前的数据源只是String集合,现在加了拼音,那就不能只用String了。建个实体UserName,字段有pinyin和name;
  • 通过pinyin4j.jar包把中文转成拼音,然后保存到UserName集合,代码如下:
    1. HanyuPinyinOutputFormat format = new HanyuPinyinOutputFormat();
    2. format.setToneType(HanyuPinyinToneType.WITHOUT_TONE);//不显示音标
    3. format.setVCharType(HanyuPinyinVCharType.WITH_V);//“ü”输出V
    4. format.setCaseType(HanyuPinyinCaseType.LOWERCASE);//拼音输出小写
    5. userNameList = new ArrayList<>();
    6. for (String name : list) {
    7. StringBuffer stringBuffer = new StringBuffer();
    8. for (int j=0;j<name.length();j++){
    9. char c = name.charAt(j);
    10. String[] cStrHY = new String[0];
    11. try {
    12. cStrHY = PinyinHelper.toHanyuPinyinStringArray(c,format);
    13. } catch (BadHanyuPinyinOutputFormatCombination
    14. badHanyuPinyinOutputFormatCombination) {
    15. badHanyuPinyinOutputFormatCombination.printStackTrace();
    16. }
    17. stringBuffer.append(cStrHY[0]);
    18. }
    19. UserName userName = new UserName();
    20. userName.setPinyin(stringBuffer.toString());
    21. userName.setName(name);
    22. userNameList.add(userName);
    23. }

    PinyinHelper类有很多转换的方法,我选择了toHanyuPinyinStringArray,将单个字符转成拼音

  • 值得注意的是,HanyuPinyinOutputFormat类,可以用你输出不同的拼音格式
    setToneType 设置音标的显示方式:

    HanyuPinyinToneType.WITH_TONE_MARK:在拼音字母上显示音标,如“zhòng”
    HanyuPinyinToneType.WITH_TONE_NUMBER:在拼音字符串后面通过数字显示,如“zhong4”
    HanyuPinyinToneType.WITHOUT_TONE:不显示音标

  • setCaseType 设置拼音大小写:

    HanyuPinyinCaseType.LOWERCASE:返回的拼音为小写字母
    HanyuPinyinCaseType.UPPERCASE:返回的拼音为大写字母

  • setVCharType 设置拼音字母“ü”的显示方式
    汉语拼音中的“ü”不能简单的通过英文来表示,所以需要单独定义“ü”的显示格式

    HanyuPinyinVCharType.WITH_U_UNICODE:默认的显示方式,输出“ü”
    HanyuPinyinVCharType.WITH_V:输出“v”
    HanyuPinyinVCharType.WITH_U_AND_COLON:输出“u:”

所以过滤的判断需要改一下,代码如下:

  1. List<UserName> fList = new ArrayList<>();
  2. String cons = constraint.toString().trim().toLowerCase();
  3. for (UserName userName : originalList) {
  4. //从首位开始匹配
  5. if (userName.getPinyin().startsWith(cons)){
  6. fList.add(userName);
  7. }
  8. }
  9. filterResults.values = fList;
  10. filterResults.count = fList.size();

看看运行效果:

总结:

这个拼音搜索功能还有待改进

1、拼音搜索的准确性,比如王重阳(wangzhongyang,wangchongyang)其实应该有两种读音,但是我现在项目只做了一种。

后续有时间再补吧,项目地址:GitHub - tongtian00/CustonSearch

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/AllinToyou/article/detail/596821
推荐阅读
相关标签
  

闽ICP备14008679号