java版本结巴分词算法bug[通俗易懂]

java版本结巴分词算法bug[通俗易懂]Nevertolate。所以明天再做也不会晚结巴分词的过程是:1、根据dict.txt中的词库构建一棵trie树,这棵树的实例只有一个,采取单例模式。2、每来一次分词构造,就顺着trie树进行分词,这将产生很多种结果,于是就生成了一个DGA,分词的有向无环图,终点是句子的左边或者右边(实际上应该分别以左边和右边为终点来做处理)。3、利用动态规划,从句子的终点开始,到这算回去(这个在动态…

大家好,又见面了,我是你们的朋友全栈君。

Never to late。所以明天再做也不会晚

结巴分词的过程是:
1、根据dict.txt中的词库构建一棵trie树,这棵树的实例只有一个,采取单例模式。
2、每来一次分词构造,就顺着trie树进行分词,这将产生很多种结果,于是就生成了一个DGA,分词的有向无环图,终点是句子的左边或者右边(实际上应该分别以左边和右边为终点来做处理)。
3、利用动态规划,从句子的终点开始,到这算回去(这个在动态规划中很常见,概率dp):对DGA中查找最大的概率的分词路径,路径上的词语就是分词结果。
4、返回分词结果。

bug1:在实现单例模式的时候,作者用的如下方法

public class WordDictionary{ 
   
	private static WordDictionary singleton;
	public static WordDictionary getInstance() { 
   
        if (singleton == null) { 
   
            synchronized (WordDictionary.class) { 
   
                if (singleton == null) { 
   
                    singleton = new WordDictionary();
                    return singleton;
                }
            }
        }
        return singleton;
    }
}

这种双重锁的方式,在并发场景下,是不安全的,为了避免java编译器对代码进行重排序,应该改为如下形式

private static volatile WordDictionary singleton;
public static WordDictionary getInstance() { 
   
   if (singleton == null) { 
   
        synchronized (WordDictionary.class) { 
   
            if (singleton == null) { 
   
                singleton = new WordDictionary();
                return singleton;
            }
        }
    }
    return singleton;
}

bug2:使用trie树对待分词句子建立DGA的时候采取递归建树,使得大量DictSegment和DictSegment[]堆积,对内存消耗特别严重。使用visual vm进行测试可以发现,将该分词加入到项目中一段时间后,在内存中可以看见DictSegment和DictSegment[]的占比非常高,如果老年代不够大,很有可能会引起OutOfMemory的异常

 Hit match(char[] charArray, int begin, int length, Hit searchHit) { 
   

        if (searchHit == null) { 
   
            // 如果hit为空,新建
            searchHit = new Hit();
            // 设置hit的起始文本位置
            searchHit.setBegin(begin);
        } else { 
   
            // 否则要将HIT状态重置
            searchHit.setUnmatch();
        }
        // 设置hit的当前处理位置
        searchHit.setEnd(begin);
        //设置起始字符为当前字典树的根节点
        Character   keyChar = new Character(charArray[begin]);
        //该keyChar对应的DictSegment
        DictSegment ds      = null;

        // 引用实例变量为本地变量,避免查询时遇到更新的同步问题
        DictSegment[]               segmentArray = this.childrenArray;
        Map<Character, DictSegment> segmentMap   = this.childrenMap;

        // STEP1 在节点中查找keyChar对应的DictSegment
        if (segmentArray != null) { 
   
            // 在数组中查找
            DictSegment keySegment = new DictSegment(keyChar);
            int         position   = Arrays.binarySearch(segmentArray, 0, this.storeSize, keySegment);
            if (position >= 0) { 
   
                ds = segmentArray[position];
            }

        } else if (segmentMap != null) { 
   
            // 在map中查找
            ds = (DictSegment) segmentMap.get(keyChar);
        }

        // STEP2 找到DictSegment,判断词的匹配状态,是否继续递归,还是返回结果
        if (ds != null) { 
   
            if (length > 1) { 
   
                // 词未匹配完,继续往下搜索
                return ds.match(charArray, begin + 1, length - 1, searchHit);
            } else if (length == 1) { 
   

                // 搜索最后一个char
                if (ds.nodeState == 1) { 
   
                    // 添加HIT状态为完全匹配
                    searchHit.setMatch();
                }
                if (ds.hasNextNode()) { 
   
                    // 添加HIT状态为前缀匹配
                    searchHit.setPrefix();
                    // 记录当前位置的DictSegment
                    searchHit.setMatchedDictSegment(ds);
                }
                return searchHit;
            }

        }
        // STEP3 没有找到DictSegment, 将HIT设置为不匹配
        return searchHit;
    }
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/149054.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 使用网站地图六大好处

    使用网站地图六大好处

  • 如何免费申请博客 用WordPress建设网站

    如何免费申请博客 用WordPress建设网站如何用WordPress搭建博客  10年前左右开始兴起第一波互联网浪潮,博客作为一种新型的社交和阅读方式进入人们的视野,那一段时期是博客的黄金时代。现在,人们说“博客已死”。因为大概在7、8年前第二波互联网浪潮突然出现,且来势汹汹——移动互联网的时代到来了。人们被各种新型的社交媒体所吸引,单单手机APP就使人们忙的目不暇接。人们的选择越来越多,于是碎片化的娱乐逐渐成为人们社交休闲方式。人们不…

  • 前端vscode必备插件推荐(墙裂推荐)「建议收藏」

    前端vscode必备插件推荐(墙裂推荐)「建议收藏」前言:vscode是一款强大的前端编辑软件,有些人说ws(webstorm)更好用,但是vs重在轻量级啊!!!而且根据自己的开发习惯安装适合自己的插件后,用起来简直不要太舒服了好嘛!!!首先呢,我先推荐的就是最基础的语言包,没办法,英语水平太捞了哈哈哈,弄起来后就舒服多了,汉语yyds~《Chinese(Simplified)(简体中文)Language》注释工具《ColorfulComments》不同的注释符能带来很多高亮的显示快速找到css定义位置并小窗口展示

  • integer转string java_Integer转换为String类型[通俗易懂]

    integer转string java_Integer转换为String类型[通俗易懂]在学习泛型时,遇到了一个小问题:Integeri=2;Strings=(String)i;Integer类型转换为String类型,本来想直接用强制转换,结果报错:Exceptioninthread”main”java.lang.ClassCastException:java.lang.Integercannotbecasttojava.lang.String经过…

    2022年10月24日
  • DirectSound的应用

    DirectSound的应用

    2021年12月14日
  • redis分布式锁的应用场景_分布式锁redis实现方式

    redis分布式锁的应用场景_分布式锁redis实现方式RedLock分布式锁 基于Redis实现分布式锁的方式名叫Redlock 安全特性:互斥访问,即永远只有一个client能拿到锁 避免死锁:最终client都可能拿到锁,不会出现死锁的情况,即使原本锁住某资源的clientcrash了或者出现了网络分区(两个完全不连通的区域,美国的网咱们登不上去) 容错性:只要大部分Redis节点存活就可以正常提供服务 Redis集群及应用场景热点数据存取数据优先从Redis操作,如果不存在再从文

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号