浅析pinyin4j源码 简单利用pinyin4j对中文字符进行自然排序(转)
pinyin4j项目 官网地址 http://pinyin4j.sourceforge.net/
我们先把资源下载下来,连同源码和jar包一起放入工程。如下图:
接下来在demo包下,我们写一个测试类,简单使用pinyin4j对中文字符进行自然排序
新建一个ConvertTest.java


- package demo;
- import java.util.ArrayList;
- import java.util.Collections;
- import java.util.Comparator;
- import java.util.HashMap;
- import java.util.List;
- import java.util.Map;
- import net.sourceforge.pinyin4j.PinyinHelper;
- public class ConvertTest {
- public static void main(String[] args) {
- String src = "我们中间出了一个叛徒";
- char[] arr = src.toCharArray();
- System.out.println("数组长度是:"+arr.length);
- System.out.print("原始顺序:");
- for (char temp : arr) {
- System.out.print(temp+" ");
- }
- System.out.println();
- convertToHanyuPinyin(arr);
- }
- private static List<String> convertToHanyuPinyin(char[] array){
- HashMap<String, String> map = new HashMap<String, String>();
- for (int i = 0; i < array.length; i++) {
- //得到拼音首字母
- String value = (PinyinHelper.toHanyuPinyinStringArray(array[i]))[0].substring(0, 1);
- map.put(String.valueOf(array[i]), value);
- }
- System.out.println(map);
- List<String> list = sort(map);
- return list;
- }
- private static List<String> sort(Map map){
- List<Map.Entry<String, String>> infoIds =
- new ArrayList<Map.Entry<String, String>>(map.entrySet());
- // 对HashMap中的 value 进行排序
- Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
- @Override
- public int compare(Map.Entry<String, String> o1,
- Map.Entry<String, String> o2) {
- return (o1.getValue()).compareTo(o2.getValue());
- }
- });
- List<String> list = new ArrayList<String>();
- /*****************FOR TEST***********************/
- List<String> letterList = new ArrayList<String>();
- /*****************FOR TEST***********************/
- // 对HashMap中的 value 进行排序后 显示排序结果
- for (int i = 0; i < infoIds.size(); i++) {
- Map.Entry<String,String> entry = infoIds.get(i);
- list.add(entry.getKey());
- letterList.add(entry.getValue());
- }
- /*****************FOR TEST***********************/
- System.out.print("自然顺序:");
- for (String string : list) {
- System.out.print(string + " ");
- }
- System.out.println();
- System.out.print("字母顺序:");
- for (String string : letterList) {
- System.out.print(string +" ");
- }
- /*****************FOR TEST***********************/
- return list;
- }
- }


输出结果为:
可以看到最终的输出顺序已经是按照自然顺序排序后的结果了。
简单说一下步骤:
1.我们先将字符串序列转换成 单个字符key, 首字母value 的map形式,
如 {个=g, 徒=t, 我=w, 出=c, 叛=p, 了=l, 中=z, 一=y, 间=j, 们=m}。
2. 然后针对map中的value进行排序,并返回排序过后的key值。
(PS:当然这里也可以对key值进行排序,但是最好还是针对value。
因为我们这里截取的是首字母,并不是整个拼音音节。)
代码缺点:
1.只是针对中文字符的第一个拼音进行排序,但是汉语中存在多音字。
2.只是针对字符的首字母进行排序,并不是整个拼音字节,并不严谨,适合粗略排序的场景。
下面简单分析一下,pinyin4j的转换流程。
如上图,其中核心的类就是PinyinHelper。它可以转换许多类型的拼音,这里我们只看汉语拼音,其他的与之类似。
追踪代码 PinyinHelper.toHanyuPinyinStringArray
按ctrl + 鼠标左键。
- static public String[] toHanyuPinyinStringArray(char ch)
- {
- return getUnformattedHanyuPinyinStringArray(ch);
- }
继续跟踪代码
- private static String[] getUnformattedHanyuPinyinStringArray(char ch)
- {
- return ChineseToPinyinResource.getInstance().getHanyuPinyinStringArray(ch);
- }
调用ChineseToPinyinResource示例的getHanyuPinyinStringArray方法

- String[] getHanyuPinyinStringArray(char ch)
- {
- String pinyinRecord = getHanyuPinyinRecordFromChar(ch);
- if (null != pinyinRecord)
- {
//得到左括号( 的索引值- int indexOfLeftBracket = pinyinRecord.indexOf(Field.LEFT_BRACKET);
//得到右括号) 的索引值- int indexOfRightBracket = pinyinRecord.lastIndexOf(Field.RIGHT_BRACKET);
- //得到字符对应的拼音
- String stripedString = pinyinRecord.substring(indexOfLeftBracket
- + Field.LEFT_BRACKET.length(), indexOfRightBracket);
- //以逗号.为分隔 返回String[] 数组
- return stripedString.split(Field.COMMA);
- } else
- return null; // no record found or mal-formatted record
- }

关键的方法getHanyuPinyinRecordFromChar

- private String getHanyuPinyinRecordFromChar(char ch)
- {
- int codePointOfChar = ch;
- //转换成unicode对应的字符
- String codepointHexStr = Integer.toHexString(codePointOfChar).toUpperCase();
- //从表中查询字符
- // fetch from hashtable
- String foundRecord = getUnicodeToHanyuPinyinTable().getProperty(codepointHexStr);
- //如果是合法的字符就返回,否则返回null
- return isValidRecord(foundRecord) ? foundRecord : null;
- }

就是如下图的资源:
http://www.cnblogs.com/sphere/p/4738888.html
浅析pinyin4j源码 简单利用pinyin4j对中文字符进行自然排序(转)的更多相关文章
- 【转】Android 源码下利用jni编译自己的项目(参考系统development/samples/SimpleJNI)
原文网址:http://blog.csdn.net/qiuxiaolong007/article/details/7860481 记于正文前:环境是ubuntu10.10,android 源码是2.0 ...
- negroni-gzip源码简单分析解读
negroni-gzip源码简单分析解读 这是一个为Negroni设计的gzip压缩处理中间件,需要用到已有的compress中的gzip,阅读了不长的源码之后,总结了一些关键要点和注意点. 检查是否 ...
- FFmpeg的HEVC解码器源码简单分析:解析器(Parser)部分
===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...
- FFmpeg源码简单分析:libswscale的sws_scale()
===================================================== FFmpeg的库函数源码分析文章列表: [架构图] FFmpeg源码结构图 - 解码 FFm ...
- APIView源码简单分析图
APIView源码简单分析 !声明:下面这个dispatch分发方法不在是父类View里的dispatch了,APIView重新封装了这个dispatch.(整个核心就是initialize_requ ...
- Django-session中间件源码简单分析
Django-session中间件源码简单分析 settings里有关中间件的配置 MIDDLEWARE = [ 'django.middleware.security.SecurityMiddlew ...
- FFmpeg源码简单分析:结构体成员管理系统-AVOption
===================================================== FFmpeg的库函数源码分析文章列表: [架构图] FFmpeg源码结构图 - 解码 FFm ...
- FFmpeg的HEVC解码器源码简单分析:概述
===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...
- FFmpeg的HEVC解码器源码简单分析:解码器主干部分
===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...
随机推荐
- 建立qemu桥接的网络连接
转载请注明出处谢谢:http://www.openext.org/2014/07/qemu-kvm-bridge-00 安装桥接工具:sudo apt-get install bridge-u ...
- poj2761(treap入门)
给n个数,然后m个询问,询问任意区间的第k小的数,特别的,任意两个区间不存在包含关系, 也就是说,将所有的询问按L排序之后, 对于i<j , Li < Lj 且 Ri < Rj ...
- Html5 Device API详解
三.四月曾学习过html5相关知识,并就html5 device api做过一次讲解 课程时长一个小时,预期达到level 200目标,即知道html5 device api是什么,且知道怎么实现 面 ...
- mysql如何更新一个表中的某个字段值等于另一个表的某个字段值
表A和表B,现在希望更新A表,当 A.bid = B.id时,a.x = b.x, a.y=b.y,请问mysql中sql语句 update a inner join b on a.bid=b.id ...
- iOS游戏开发游戏功能之外的东西
对于一个游戏的开发,我们除了完毕游戏的功能之外,还有多少东西我们须要考虑呢? 非常多.也非常烦! 但做过一遍之后下一次就会非常easy. 都有什么东西我们想加入到游戏其中呢? (1)分享功能 (2)评 ...
- Windows Phone开发(30):图形
原文:Windows Phone开发(30):图形 图形如矩形.椭圆.路径等都从Shape类派生,它们一般表示规则或不规则图形,这些图形都是简单的二维图形,我相信大家都能理解的. 例一:矩形. 请看下 ...
- hadoop调度程序时出现“Error opening job jar”错误
提示出现的问题: Exception in thread "main" java.io.IOException: Error opening job jar: /home/depl ...
- 自己写shell命令pwd
思维:(1)得到"."的i节点号,叫n(使用stat) (2)chdir ..(使用chdir) (3)找到inode号为n的节点,得到其文件名称. 反复上述操作直到当前文件夹&q ...
- Mahout推荐算法ItemBased
Mahout推荐的ItemBased 一. 算法原理 (一) 基本的 下面的例子,参见图评分矩阵:表现user,归类为item. 图(1) 该算法的原理: 1. 计算Item之间的相似度. ...
- centos7 设备 mariadb-10
下载地址: http://mirrors.ustc.edu.cn/mariadb/mariadb-10.0.19/source/mariadb-10.0.19.tar.gz 由于用cmake所以线安装 ...