当前位置:   article > 正文

android开发之使用拼音搜索汉字

安卓arrylist 拼音搜索

国庆回了趟家,昨天真不想走,离家近的感觉太好。唉,不扯这些,说说今天的正事吧。
上篇博客中介绍了自定义AutoCompleteTextView ,但是用到了一个很蹩脚的技术,就是我们事先把每个汉字的拼音当作一个字段输入进去了,在实际的开发中肯定不会这样做,我们要通过代码自动生成汉字的拼音,就像我们的手机通讯录,比如我们要查找“张三”这个人,我们只需要输入“zs”、“cs”或者“zhangsan”、“changsan”就能搜索到该人,那么我们该怎么来实现这样的功能呢?

本文所述案例是在上篇博客的基础上实现的,如果还没阅读上篇博客,请看android开发之自定义AutoCompleteTextView
本文要实现的整体效果如下图所示:
这里写图片描述


在上篇博客中我们自定义了AutoCompleteTextView的Adapter,本文中,我们继续对这个Adapter进行深化改造。

主要改造两个地方,第一个地方是在构造方法中初始化拼音集合:

改造后的构造方法:

  1. public MyActAdapter(Context context, List<Book> books, int maxMatch) {
  2. this.books = books;
  3. this.context = context;
  4. this.maxMatch = maxMatch;
  5. initPinYinList();
  6. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

这个方法主要是初始化两个List集合,一个是pinYinList 另一个是pinYinAllList ,前者是所有书的作者姓名拼音的首字母集合,后者是所有书的作者姓名拼音全拼集合。

  1. private void initPinYinList() {
  2. pinYinList = new ArrayList<Set<String>>();
  3. pinYinAllList = new ArrayList<Set<String>>();
  4. PinYin4j pinyin = new PinYin4j();
  5. for (int i = 0; i < books.size(); i++) {
  6. pinYinList.add(pinyin.getPinyin(books.get(i).getAuthor().toString()));
  7. pinYinAllList.add(pinyin.getAllPinyin(books.get(i).getAuthor().toString()));
  8. }
  9. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

这里还涉及到两个类,如下:

PinYin4j.java

  1. package com.example.myact;
  2. import java.util.Arrays;
  3. import java.util.HashSet;
  4. import java.util.Set;
  5. public class PinYin4j {
  6. public PinYin4j(){
  7. }
  8. /**
  9. * 字符串集合转换字符串(逗号分隔)
  10. *
  11. * @author wangsong
  12. * @param stringSet
  13. * @return
  14. */
  15. public String makeStringByStringSet(Set<String> stringSet) {
  16. StringBuilder str = new StringBuilder();
  17. int i = 0;
  18. for (String s : stringSet) {
  19. if (i == stringSet.size() - 1) {
  20. str.append(s);
  21. } else {
  22. str.append(s + ",");
  23. }
  24. i++;
  25. }
  26. return str.toString().toLowerCase();
  27. }
  28. /**
  29. * 获取汉字拼音全拼
  30. *
  31. * @author wangsong
  32. * @param src
  33. * @return Set<String>
  34. */
  35. public Set<String> getAllPinyin(String src) {
  36. char[] srcChar;
  37. srcChar = src.toCharArray();
  38. String[][] temp = new String[src.length()][];
  39. for (int i = 0; i < srcChar.length; i++) {
  40. char c = srcChar[i];
  41. if (String.valueOf(c).matches("[\\u4E00-\\u9FA5]+")) {
  42. String[] t = PinyinHelper.getUnformattedHanyuPinyinStringArray(c);
  43. temp[i] = new String[t.length];
  44. for(int j=0;j<t.length;j++){
  45. temp[i][j]=t[j].replaceAll("\\d", "");//获取全拼
  46. }
  47. } else if (((int) c >= 65 && (int) c <= 90)
  48. || ((int) c >= 97 && (int) c <= 122)||c>=48&&c<=57||c==42) {
  49. temp[i] = new String[] { String.valueOf(srcChar[i]) };
  50. } else {
  51. temp[i] = new String[] {"null!"};
  52. }
  53. }
  54. String[] pingyinArray = paiLie(temp);
  55. return array2Set(pingyinArray);
  56. }
  57. /**
  58. * 获取汉字拼音首字母集合
  59. *
  60. * @author wangsong
  61. * @param src
  62. * @return Set<String>
  63. */
  64. public Set<String> getPinyin(String src) {
  65. char[] srcChar;
  66. srcChar = src.toCharArray();
  67. String[][] temp = new String[src.length()][];
  68. for (int i = 0; i < srcChar.length; i++) {
  69. char c = srcChar[i];
  70. if (String.valueOf(c).matches("[\\u4E00-\\u9FA5]+")) {
  71. String[] t = PinyinHelper.getUnformattedHanyuPinyinStringArray(c);
  72. temp[i] = new String[t.length];
  73. for(int j=0;j<t.length;j++){
  74. temp[i][j]=t[j].substring(0,1);
  75. }
  76. } else if (((int) c >= 65 && (int) c <= 90)
  77. || ((int) c >= 97 && (int) c <= 122)||c>=48&&c<=57||c==42) {
  78. temp[i] = new String[] { String.valueOf(srcChar[i]) };
  79. } else {
  80. temp[i] = new String[] {"null!"};
  81. }
  82. }
  83. String[] pingyinArray = paiLie(temp);
  84. return array2Set(pingyinArray);
  85. }
  86. /*
  87. * 求2维数组所有排列组合情况
  88. * 比如:{{1,2},{3},{4},{5,6}}共有2中排列,为:1345,1346,2345,2346
  89. */
  90. private String[] paiLie(String[][] str){
  91. int max=1;
  92. for(int i=0;i<str.length;i++){
  93. max*=str[i].length;
  94. }
  95. String[] result=new String[max];
  96. for(int i = 0; i < max; i++){
  97. String s = "";
  98. int temp = 1; //注意这个temp的用法。
  99. for(int j = 0; j < str.length; j++){
  100. temp *= str[j].length;
  101. s += str[j][i / (max / temp) % str[j].length];
  102. }
  103. result[i]=s;
  104. }
  105. return result;
  106. }
  107. /**
  108. * 去掉重复项
  109. * @param tArray
  110. * @return
  111. */
  112. public static <T extends Object> Set<T> array2Set(T[] tArray) {
  113. Set<T> tSet = new HashSet<T>(Arrays.asList(tArray));
  114. // TODO 没有一步到位的方法,根据具体的作用,选择合适的Set的子类来转换。
  115. return tSet;
  116. }
  117. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127

PinyinHelper.java

  1. package com.example.myact;
  2. import java.io.BufferedInputStream;
  3. import java.io.FileNotFoundException;
  4. import java.io.IOException;
  5. import java.util.Properties;
  6. public class PinyinHelper {
  7. private static PinyinHelper instance;
  8. private Properties properties = null;
  9. public static String[] getUnformattedHanyuPinyinStringArray(char ch) {
  10. return getInstance().getHanyuPinyinStringArray(ch);
  11. }
  12. private PinyinHelper() {
  13. initResource();
  14. }
  15. public static PinyinHelper getInstance() {
  16. if (instance == null) {
  17. instance = new PinyinHelper();
  18. }
  19. return instance;
  20. }
  21. private void initResource() {
  22. try {
  23. final String resourceName = "/assets/unicode_to_hanyu_pinyin.txt";
  24. // final String resourceName = "/assets/unicode_py.ini";
  25. properties = new Properties();
  26. properties.load(getResourceInputStream(resourceName));
  27. } catch (FileNotFoundException ex) {
  28. ex.printStackTrace();
  29. } catch (IOException ex) {
  30. ex.printStackTrace();
  31. }
  32. }
  33. private BufferedInputStream getResourceInputStream(String resourceName) {
  34. return new BufferedInputStream(
  35. PinyinHelper.class.getResourceAsStream(resourceName));
  36. }
  37. private String[] getHanyuPinyinStringArray(char ch) {
  38. String pinyinRecord = getHanyuPinyinRecordFromChar(ch);
  39. if (null != pinyinRecord) {
  40. int indexOfLeftBracket = pinyinRecord.indexOf(Field.LEFT_BRACKET);
  41. int indexOfRightBracket = pinyinRecord
  42. .lastIndexOf(Field.RIGHT_BRACKET);
  43. String stripedString = pinyinRecord.substring(indexOfLeftBracket
  44. + Field.LEFT_BRACKET.length(), indexOfRightBracket);
  45. return stripedString.split(Field.COMMA);
  46. } else
  47. return null;
  48. }
  49. private String getHanyuPinyinRecordFromChar(char ch) {
  50. int codePointOfChar = ch;
  51. String codepointHexStr = Integer.toHexString(codePointOfChar)
  52. .toUpperCase();
  53. String foundRecord = properties.getProperty(codepointHexStr);
  54. return foundRecord;
  55. }
  56. class Field {
  57. static final String LEFT_BRACKET = "(";
  58. static final String RIGHT_BRACKET = ")";
  59. static final String COMMA = ",";
  60. }
  61. public static String[] toHanyuPinyinStringArray(char ch) {
  62. return getUnformattedHanyuPinyinStringArray(ch);
  63. }
  64. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82

这里是初始化拼音集合。
第二个改造的地方就是在过滤器中增加过滤的条件。

这是最新的过滤器,和上文相比,这里只是增加了一个else分支,在else分支中判断搜索条件是否符合要求。

  1. private class ArrayFilter extends Filter {
  2. @Override
  3. protected FilterResults performFiltering(CharSequence constraint) {
  4. FilterResults results = new FilterResults();
  5. if (mFilterBooks == null) {
  6. mFilterBooks = new ArrayList<Book>(books);
  7. }
  8. // 如果没有过滤条件则不过滤
  9. if (constraint == null || constraint.length() == 0) {
  10. results.values = mFilterBooks;
  11. results.count = mFilterBooks.size();
  12. } else {
  13. List<Book> retList = new ArrayList<Book>();
  14. // 过滤条件
  15. String str = constraint.toString().toLowerCase();
  16. Book book;
  17. // 循环变量数据源,如果有属性满足过滤条件,则添加到result中
  18. for (int i = 0; i < mFilterBooks.size(); i++) {
  19. book = mFilterBooks.get(i);
  20. if (book.getAuthor().contains(str)
  21. || book.getName().contains(str)
  22. || (book.getId() + "").contains(str)
  23. || (book.getPrice() + "").contains(str)
  24. || book.getPinyin().contains(str)) {
  25. retList.add(book);
  26. } else {
  27. //查看作者姓名拼音首字母是否符合过滤条件
  28. Set<String> pinyinSet = pinYinList.get(i);
  29. Iterator<String> pinyin = pinyinSet.iterator();
  30. while (pinyin.hasNext()) {
  31. if (pinyin.next().toString().contains(str)) {
  32. retList.add(book);
  33. break;
  34. }
  35. }
  36. //查看作者姓名拼音全拼是否符合过滤条件
  37. Set<String> pinyinAllSet = pinYinAllList.get(i);
  38. Iterator<String> pinyinAll = pinyinAllSet.iterator();
  39. while (pinyinAll.hasNext()) {
  40. if (pinyinAll.next().toString().contains(str)) {
  41. retList.add(book);
  42. break;
  43. }
  44. }
  45. }
  46. // if (maxMatch > 0) {
  47. // if (retList.size() > maxMatch - 1) {
  48. // break;
  49. // }
  50. // }
  51. }
  52. results.values = retList;
  53. results.count = retList.size();
  54. }
  55. return results;
  56. }
  57. // 在这里返回过滤结果
  58. @Override
  59. protected void publishResults(CharSequence constraint,
  60. FilterResults results) {
  61. // notifyDataSetInvalidated(),会重绘控件(还原到初始状态)
  62. // notifyDataSetChanged(),重绘当前可见区域
  63. books = (List<Book>) results.values;
  64. if (results.count > 0) {
  65. notifyDataSetChanged();
  66. } else {
  67. notifyDataSetInvalidated();
  68. }
  69. }
  70. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73

其实还是很简单的,有问题欢迎留言讨论。

完整代码下载点这里。

版权声明:本文为博主原创文章,未经博主允许不得转载。若有错误地方,还望批评指正,不胜感激。

转载于:https://www.cnblogs.com/lenve/p/4889318.html

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号