android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼
好幾年沒(méi)寫(xiě)博客了,很多知識(shí)不記是真的會(huì)忘記,以后還是保持寫(xiě)博客的習(xí)慣吧。堅(jiān)持不一定成功,但放棄一定很舒服?。ㄩ_(kāi)玩笑(#^.^#))
回歸正題,今天我要記錄的是拼音搜索功能,我記得16年的時(shí)候做過(guò)這個(gè)功能?,F(xiàn)在已經(jīng)忘記很多細(xì)節(jié)了,所以這次好好寫(xiě)一寫(xiě)!
第一步:準(zhǔn)備
- 第三方包——中文轉(zhuǎn)拼音,pinyin4j-2.5.0.jar,官網(wǎng)http://pinyin4j.sourceforge.net/
- Filterable.java 過(guò)濾接口
第二步:分析功能并實(shí)現(xiàn)
- 很明顯,這是兩個(gè)功能,一個(gè)是中文轉(zhuǎn)拼音,一個(gè)是查詢過(guò)濾
- 下面我們先實(shí)現(xiàn)主要功能,查詢過(guò)濾
- 創(chuàng)建布局文件 activity_main.xml,實(shí)現(xiàn)如下:
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" android:padding="16dp"> <EditText android:id="@+id/etSearchName" android:layout_width="match_parent" android:layout_height="50dp" android:hint="請(qǐng)輸入拼音" android:padding="5dp" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" /> <androidx.recyclerview.widget.RecyclerView android:id="@+id/recyclerView" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginTop="10dp" app:layout_constraintTop_toBottomOf="@+id/etSearchName" /> </androidx.constraintlayout.widget.ConstraintLayout>
就一個(gè)搜索框和列表控件
接著創(chuàng)建列表適配器 SearchAdapter.java,實(shí)現(xiàn)了過(guò)濾器類(lèi)Filterable
public class SearchAdapter extends RecyclerView.Adapter<SearchAdapter.MyViewHolder> implements Filterable { private Context context; private List<String> list; //保存原有的數(shù)據(jù) private List<String> originalList; private OnItemListener onItemListener; void setOnItemListener(OnItemListener onItemListener){ this.onItemListener = onItemListener; } private SearchFilter filter; SearchAdapter(Context context,List<String> list){ this.context =context; this.list = list; originalList = list; } @NonNull @Override public SearchAdapter.MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { View view = LayoutInflater.from(context).inflate(R.layout.item_rv_search,parent,false); return new MyViewHolder(view); } @Override public void onBindViewHolder(@NonNull SearchAdapter.MyViewHolder holder, final int position) { holder.tvName.setText(list.get(position)); holder.itemView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if (onItemListener!=null){ onItemListener.onItem(list.get(position)); } } }); } @Override public int getItemCount() { return list==null?0:list.size(); } @Override public Filter getFilter() { if (filter==null){ filter = new SearchFilter(); } return filter; } class MyViewHolder extends RecyclerView.ViewHolder{ private TextView tvName; MyViewHolder(@NonNull View itemView) { super(itemView); tvName = itemView.findViewById(R.id.tvName); } } interface OnItemListener{ void onItem(String name); } class SearchFilter extends Filter{ @Override protected FilterResults performFiltering(CharSequence constraint) { //輸入框傳來(lái)的數(shù)據(jù) constraint //用于保存過(guò)濾的結(jié)果 FilterResults filterResults = new FilterResults(); if (constraint==null || constraint.length()==0){ filterResults.values = originalList; filterResults.count = originalList.size(); }else { List<String> fList = new ArrayList<>(); String cons = constraint.toString().trim().toLowerCase(); for (String s : originalList) { //從首位開(kāi)始匹配 if (s.startsWith(cons)){ fList.add(s); } } filterResults.values = fList; filterResults.count = fList.size(); } return filterResults; } @Override protected void publishResults(CharSequence constraint, FilterResults results) { list = (List<String>) results.values; notifyDataSetChanged(); } } }
這段代碼,重點(diǎn)是SearchFilter類(lèi)的兩個(gè)方法
- performFiltering(CharSequence constraint) 接收輸入數(shù)據(jù),在這個(gè)方法實(shí)現(xiàn)處理邏輯
- publishResults(CharSequence constraint, FilterResults results) 接收過(guò)濾后的結(jié)果
注意點(diǎn):originalList 這個(gè)集合是用來(lái)保存原有數(shù)據(jù)的。因?yàn)閘ist會(huì)隨著搜索結(jié)果而變化,我們每次的過(guò)濾都是需要用到原有的數(shù)據(jù)。
適配器寫(xiě)好了,下面看看怎么使用它
private RecyclerView recyclerView; private SearchAdapter adapter; private List<String> list; private EditText etSearchName; public static final String[] str = new String[]{ "陳天麗","黃正","徐明" ,"李自成","林子祥","周星星" ,"周潤(rùn)發(fā)","林星辰","林青霞" ,"李賽鳳","劉德華","胡歌" ,"霍建華","林心如","趙薇" ,"趙四","趙本山","郭德綱" }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); etSearchName = findViewById(R.id.etSearchName); recyclerView = findViewById(R.id.recyclerView); recyclerView.setLayoutManager(new LinearLayoutManager(this)); list = new ArrayList<>(); list.addAll(Arrays.asList(str)); adapter = new SearchAdapter(this,list); recyclerView.setAdapter(adapter); adapter.setOnItemListener(new SearchAdapter.OnItemListener() { @Override public void onItem(String name) { etSearchName.setText(name); } }); etSearchName.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { adapter.getFilter().filter(etSearchName.getText().toString().trim()); } @Override public void afterTextChanged(Editable s) { } }); }
一頓RecyclerView的常規(guī)操作
這里需要注意的是這段代碼 adapter.getFilter().filter(etSearchName.getText().toString().trim()); 這里就調(diào)用了我們寫(xiě)好的過(guò)濾器。
到此,我們的搜索過(guò)濾功能已經(jīng)實(shí)現(xiàn)了。
看看運(yùn)行效果:
接下來(lái)說(shuō)說(shuō)中文轉(zhuǎn)拼音
首先需要修改數(shù)據(jù)源類(lèi)型
- 之前的數(shù)據(jù)源只是String集合,現(xiàn)在加了拼音,那就不能只用String了。建個(gè)實(shí)體UserName,字段有pinyin和name;
- 通過(guò)pinyin4j.jar包把中文轉(zhuǎn)成拼音,然后保存到UserName集合,代碼如下:
HanyuPinyinOutputFormat format = new HanyuPinyinOutputFormat(); format.setToneType(HanyuPinyinToneType.WITHOUT_TONE);//不顯示音標(biāo) format.setVCharType(HanyuPinyinVCharType.WITH_V);//“ü”輸出V format.setCaseType(HanyuPinyinCaseType.LOWERCASE);//拼音輸出小寫(xiě) userNameList = new ArrayList<>(); for (String name : list) { StringBuffer stringBuffer = new StringBuffer(); for (int j=0;j<name.length();j++){ char c = name.charAt(j); String[] cStrHY = new String[0]; try { cStrHY = PinyinHelper.toHanyuPinyinStringArray(c,format); } catch (BadHanyuPinyinOutputFormatCombination badHanyuPinyinOutputFormatCombination) { badHanyuPinyinOutputFormatCombination.printStackTrace(); } stringBuffer.append(cStrHY[0]); } UserName userName = new UserName(); userName.setPinyin(stringBuffer.toString()); userName.setName(name); userNameList.add(userName); }
PinyinHelper類(lèi)有很多轉(zhuǎn)換的方法,我選擇了toHanyuPinyinStringArray,將單個(gè)字符轉(zhuǎn)成拼音
值得注意的是,HanyuPinyinOutputFormat類(lèi),可以用你輸出不同的拼音格式
setToneType 設(shè)置音標(biāo)的顯示方式:
HanyuPinyinToneType.WITH_TONE_MARK:在拼音字母上顯示音標(biāo),如“zhòng”
HanyuPinyinToneType.WITH_TONE_NUMBER:在拼音字符串后面通過(guò)數(shù)字顯示,如“zhong4”
HanyuPinyinToneType.WITHOUT_TONE:不顯示音標(biāo)
setCaseType 設(shè)置拼音大小寫(xiě):
HanyuPinyinCaseType.LOWERCASE:返回的拼音為小寫(xiě)字母
HanyuPinyinCaseType.UPPERCASE:返回的拼音為大寫(xiě)字母
setVCharType 設(shè)置拼音字母“ü”的顯示方式
漢語(yǔ)拼音中的“ü”不能簡(jiǎn)單的通過(guò)英文來(lái)表示,所以需要單獨(dú)定義“ü”的顯示格式
HanyuPinyinVCharType.WITH_U_UNICODE:默認(rèn)的顯示方式,輸出“ü”
HanyuPinyinVCharType.WITH_V:輸出“v”
HanyuPinyinVCharType.WITH_U_AND_COLON:輸出“u:”
所以過(guò)濾的判斷需要改一下,代碼如下:
List<UserName> fList = new ArrayList<>(); String cons = constraint.toString().trim().toLowerCase(); for (UserName userName : originalList) { //從首位開(kāi)始匹配 if (userName.getPinyin().startsWith(cons)){ fList.add(userName); } } filterResults.values = fList; filterResults.count = fList.size();
看看運(yùn)行效果:
總結(jié):
這個(gè)拼音搜索功能還有待改進(jìn)
1、拼音搜索的準(zhǔn)確性,比如王重陽(yáng)(wangzhongyang,wangchongyang)其實(shí)應(yīng)該有兩種讀音,但是我現(xiàn)在項(xiàng)目只做了一種。
后續(xù)有時(shí)間再補(bǔ)吧,項(xiàng)目地址:https://github.com/tongtian00/CustonSearch
到此這篇關(guān)于android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼的文章就介紹到這了,更多相關(guān)android搜索中文內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Android 搜索結(jié)果匹配關(guān)鍵字且高亮顯示功能
- Android搜索結(jié)果顯示高亮實(shí)例(有數(shù)據(jù)滑動(dòng)底部自動(dòng)刷新)
- Android基于RecyclerView實(shí)現(xiàn)高亮搜索列表
- Android Studio實(shí)現(xiàn)搜索欄
- Android自定義流式布局實(shí)現(xiàn)淘寶搜索記錄
- Android本地實(shí)現(xiàn)搜索歷史記錄
- Android實(shí)現(xiàn)搜索歷史功能
- Android輸入框?qū)崟r(shí)模糊搜索效果的示例代碼
- Android Studio3.6.+ 插件搜索不到終極解決方案(圖文詳解)
- Android實(shí)現(xiàn)高亮搜索功能的示例
相關(guān)文章
Android平臺(tái)預(yù)置GMS包后關(guān)機(jī)鬧鐘失效問(wèn)題及解決方法
這篇文章主要介紹了Android平臺(tái)預(yù)置GMS包后,關(guān)機(jī)鬧鐘失效,本文給大家分享問(wèn)題原因及解決方法,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09關(guān)于WebView 重定向行為導(dǎo)致的多次加載的問(wèn)題
這篇文章主要介紹了關(guān)于WebView 重定向行為導(dǎo)致的多次加載的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android 顯示刷新頻率的實(shí)現(xiàn)代碼
這篇文章主要介紹了Android 顯示刷新頻率的實(shí)現(xiàn)代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08Android編程創(chuàng)建與解析xml的常用方法詳解
這篇文章主要介紹了Android編程創(chuàng)建與解析xml的常用方法,結(jié)合具體實(shí)例形式較為詳細(xì)的分析了Android操作xml文件的步驟、實(shí)現(xiàn)技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-05-05Android?Framework原理Binder驅(qū)動(dòng)源碼解析
這篇文章主要為大家介紹了Android?Framework原理Binder驅(qū)動(dòng)源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01移動(dòng)端android上line-height不居中的問(wèn)題的解決
現(xiàn)在越來(lái)越多的移動(dòng)界面使用rem適配,最近發(fā)現(xiàn)了移動(dòng)端android上line-height不居中的問(wèn)題,今日就來(lái)介紹一下解決的方法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2018-03-03Android實(shí)現(xiàn)毛玻璃效果的對(duì)話框
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)毛玻璃效果的對(duì)話框,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-04-04Android自定義View實(shí)現(xiàn)兩種二維碼的掃描效果
這篇文章主要為大家詳細(xì)介紹了Android如何自定義View實(shí)現(xiàn)兩種二維碼的掃描效果,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-01-01