android通過拼音搜索中文的功能實現(xiàn)代碼
好幾年沒寫博客了,很多知識不記是真的會忘記,以后還是保持寫博客的習慣吧。堅持不一定成功,但放棄一定很舒服!(開玩笑(#^.^#))
回歸正題,今天我要記錄的是拼音搜索功能,我記得16年的時候做過這個功能?,F(xiàn)在已經(jīng)忘記很多細節(jié)了,所以這次好好寫一寫!
第一步:準備
- 第三方包——中文轉拼音,pinyin4j-2.5.0.jar,官網(wǎng)http://pinyin4j.sourceforge.net/
- Filterable.java 過濾接口
第二步:分析功能并實現(xiàn)
- 很明顯,這是兩個功能,一個是中文轉拼音,一個是查詢過濾
- 下面我們先實現(xiàn)主要功能,查詢過濾
- 創(chuàng)建布局文件 activity_main.xml,實現(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="請輸入拼音"
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>
就一個搜索框和列表控件
接著創(chuàng)建列表適配器 SearchAdapter.java,實現(xiàn)了過濾器類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) {
//輸入框傳來的數(shù)據(jù) constraint
//用于保存過濾的結果
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) {
//從首位開始匹配
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();
}
}
}
這段代碼,重點是SearchFilter類的兩個方法
- performFiltering(CharSequence constraint) 接收輸入數(shù)據(jù),在這個方法實現(xiàn)處理邏輯
- publishResults(CharSequence constraint, FilterResults results) 接收過濾后的結果
注意點:originalList 這個集合是用來保存原有數(shù)據(jù)的。因為list會隨著搜索結果而變化,我們每次的過濾都是需要用到原有的數(shù)據(jù)。
適配器寫好了,下面看看怎么使用它
private RecyclerView recyclerView;
private SearchAdapter adapter;
private List<String> list;
private EditText etSearchName;
public static final String[] str = new String[]{
"陳天麗","黃正","徐明"
,"李自成","林子祥","周星星"
,"周潤發(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()); 這里就調用了我們寫好的過濾器。
到此,我們的搜索過濾功能已經(jīng)實現(xiàn)了。
看看運行效果:

接下來說說中文轉拼音
首先需要修改數(shù)據(jù)源類型
- 之前的數(shù)據(jù)源只是String集合,現(xiàn)在加了拼音,那就不能只用String了。建個實體UserName,字段有pinyin和name;
- 通過pinyin4j.jar包把中文轉成拼音,然后保存到UserName集合,代碼如下:
HanyuPinyinOutputFormat format = new HanyuPinyinOutputFormat();
format.setToneType(HanyuPinyinToneType.WITHOUT_TONE);//不顯示音標
format.setVCharType(HanyuPinyinVCharType.WITH_V);//“ü”輸出V
format.setCaseType(HanyuPinyinCaseType.LOWERCASE);//拼音輸出小寫
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類有很多轉換的方法,我選擇了toHanyuPinyinStringArray,將單個字符轉成拼音
值得注意的是,HanyuPinyinOutputFormat類,可以用你輸出不同的拼音格式
setToneType 設置音標的顯示方式:
HanyuPinyinToneType.WITH_TONE_MARK:在拼音字母上顯示音標,如“zhòng”
HanyuPinyinToneType.WITH_TONE_NUMBER:在拼音字符串后面通過數(shù)字顯示,如“zhong4”
HanyuPinyinToneType.WITHOUT_TONE:不顯示音標
setCaseType 設置拼音大小寫:
HanyuPinyinCaseType.LOWERCASE:返回的拼音為小寫字母
HanyuPinyinCaseType.UPPERCASE:返回的拼音為大寫字母
setVCharType 設置拼音字母“ü”的顯示方式
漢語拼音中的“ü”不能簡單的通過英文來表示,所以需要單獨定義“ü”的顯示格式
HanyuPinyinVCharType.WITH_U_UNICODE:默認的顯示方式,輸出“ü”
HanyuPinyinVCharType.WITH_V:輸出“v”
HanyuPinyinVCharType.WITH_U_AND_COLON:輸出“u:”
所以過濾的判斷需要改一下,代碼如下:
List<UserName> fList = new ArrayList<>();
String cons = constraint.toString().trim().toLowerCase();
for (UserName userName : originalList) {
//從首位開始匹配
if (userName.getPinyin().startsWith(cons)){
fList.add(userName);
}
}
filterResults.values = fList;
filterResults.count = fList.size();
看看運行效果:

總結:
這個拼音搜索功能還有待改進
1、拼音搜索的準確性,比如王重陽(wangzhongyang,wangchongyang)其實應該有兩種讀音,但是我現(xiàn)在項目只做了一種。
后續(xù)有時間再補吧,項目地址:https://github.com/tongtian00/CustonSearch
到此這篇關于android通過拼音搜索中文的功能實現(xiàn)代碼的文章就介紹到這了,更多相關android搜索中文內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Android編程創(chuàng)建與解析xml的常用方法詳解
這篇文章主要介紹了Android編程創(chuàng)建與解析xml的常用方法,結合具體實例形式較為詳細的分析了Android操作xml文件的步驟、實現(xiàn)技巧與相關注意事項,需要的朋友可以參考下2017-05-05
Android?Framework原理Binder驅動源碼解析
這篇文章主要為大家介紹了Android?Framework原理Binder驅動源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01
移動端android上line-height不居中的問題的解決
現(xiàn)在越來越多的移動界面使用rem適配,最近發(fā)現(xiàn)了移動端android上line-height不居中的問題,今日就來介紹一下解決的方法,非常具有實用價值,需要的朋友可以參考下2018-03-03
Android自定義View實現(xiàn)兩種二維碼的掃描效果
這篇文章主要為大家詳細介紹了Android如何自定義View實現(xiàn)兩種二維碼的掃描效果,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-01-01

