前綴樹過濾敏感詞算法

前綴樹的基本性質
1.根節(jié)點不包含字符熔掺,除根節(jié)點外每一個節(jié)點都只包含一個字符架曹。
2.從根節(jié)點到某一節(jié)點,路徑上經(jīng)過的字符連接起來,為該節(jié)點對應的字符串。
3.每個節(jié)點的所有子節(jié)點包含的字符都不相同。

數(shù)據(jù)結構:每一個TrieNode中定義一個哈希表,含有孩子節(jié)點的,即下一個字符的TrieNode炬称。

    private class TrieNode {

        // true 關鍵詞的終結 ; false 繼續(xù)
        private boolean end = false;

        // key下一個字符鸡号,value是對應的節(jié)點
        private Map<Character, TrieNode> subNodes = new HashMap<>();

        // 向指定位置添加節(jié)點樹
        void addSubNode(Character key, TrieNode node) {
            subNodes.put(key, node);
        }

        // 獲取下個節(jié)點
        TrieNode getSubNode(Character key) {
            return subNodes.get(key);
        }

        boolean isKeywordEnd() {
            return end;
        }

        void setKeywordEnd(boolean end) {
            this.end = end;
        }

        public int getSubNodeCount() {
            return subNodes.size();
        }
    }

類中設置成員變量转砖,根節(jié)點,不包含任何字符鲸伴。

    private TrieNode rootNode = new TrieNode();
添加敏感詞到前綴樹中

注意構造樹時如果根節(jié)點沒有孩子節(jié)點府蔗,要初始化再添加到哈希表中。

    private void addWord(String lineTxt) {
        TrieNode tempNode = rootNode;
        // 循環(huán)每個字節(jié)
        for (int i = 0; i < lineTxt.length(); ++i) {
            Character c = lineTxt.charAt(i);
            // 過濾空格
            if (isSymbol(c)) {
                continue;
            }
            TrieNode node = tempNode.getSubNode(c);

            if (node == null) { // 沒初始化
                node = new TrieNode();
                tempNode.addSubNode(c, node);
            }

            tempNode = node;

            if (i == lineTxt.length() - 1) {
                // 關鍵詞結束汞窗, 設置結束標志
                tempNode.setKeywordEnd(true);
            }
        }
    }
過濾敏感詞

設置了三個指針幫助判斷姓赤。
如果字符是空格類型,并且是根節(jié)點情況仲吏,直接添加字符不铆,移動begin,pos指針裹唆,temp不變誓斥。相當于直接跳過再接著尋找。非根節(jié)點則只用移動pos指針许帐。因為非根節(jié)點情況相當于已經(jīng)再尋找敏感詞的過程中了劳坑,只需要跳過該空格繼續(xù)尋找即可。
在當前節(jié)點的哈希表中尋找是否有對應字符的孩子節(jié)點成畦,如果未找到距芬,當前字符不是敏感詞。直接添加當前字符循帐。pos框仔,begin指針后移一位,temp指針回到根節(jié)點拄养。
如果找到并且不是關鍵字結尾离斩,只需要pos指針移動。
如果找到并且是關鍵字結尾瘪匿,添加關鍵字跛梗。pos指針后移一位,begin指向pos柿顶,temp返回根節(jié)點茄袖。
注意最后還要添加result.append(text.substring(begin));因為當最后循環(huán)不是敏感詞時候操软,只會移動Pos而沒有添加嘁锯。所以最后一次遍歷不能漏掉。

    public String filter(String text) {
        if (StringUtils.isBlank(text)) {
            return text;
        }
        String replacement = DEFAULT_REPLACEMENT;
        StringBuilder result = new StringBuilder();

        TrieNode tempNode = rootNode;
        int begin = 0; // 回滾數(shù)
        int position = 0; // 當前比較的位置

        while (position < text.length()) {
            char c = text.charAt(position);
            // 空格直接跳過
            if (isSymbol(c)) {
                if (tempNode == rootNode) {
                    result.append(c);
                    ++begin;
                }
                ++position;
                continue;
            }

            tempNode = tempNode.getSubNode(c);

            // 當前位置的匹配結束
            if (tempNode == null) {
                // 以begin開始的字符串不存在敏感詞
                result.append(text.charAt(begin));
                // 跳到下一個字符開始測試
                position = begin + 1;
                begin = position;
                // 回到樹初始節(jié)點
                tempNode = rootNode;
            } else if (tempNode.isKeywordEnd()) {
                // 發(fā)現(xiàn)敏感詞, 從begin到position的位置用replacement替換掉
                result.append(replacement);
                position = position + 1;
                begin = position;
                tempNode = rootNode;
            } else {
                ++position;
            }
        }
        result.append(text.substring(begin));
        return result.toString();
    }

因為有可能存在敏感詞中間夾著宮格家乘,非法字符等形式蝗羊,所以必須對這種類型進行判斷。判斷邏輯可以不同仁锯。

    private boolean isSymbol(char c) {
        int ic = (int) c;
        // 0x2E80-0x9FFF 東亞文字范圍
        return !CharUtils.isAsciiAlphanumeric(c) && (ic < 0x2E80 || ic > 0x9FFF);
    }

具體項目中耀找,可以將關鍵字保存到文件,然后讀取文件構建前綴樹

    @Override
    public void afterPropertiesSet() throws Exception {
        rootNode = new TrieNode();

        try {
            InputStream is = Thread.currentThread().getContextClassLoader()
                    .getResourceAsStream("SensitiveWords.txt");
            InputStreamReader read = new InputStreamReader(is);
            BufferedReader bufferedReader = new BufferedReader(read);
            String lineTxt;
            while ((lineTxt = bufferedReader.readLine()) != null) {
                lineTxt = lineTxt.trim();
                addWord(lineTxt);
            }
            read.close();
        } catch (Exception e) {
            logger.error("讀取敏感詞文件失敗" + e.getMessage());
        }
    }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末业崖,一起剝皮案震驚了整個濱河市野芒,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌双炕,老刑警劉巖狞悲,帶你破解...
    沈念sama閱讀 216,744評論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異妇斤,居然都是意外死亡摇锋,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,505評論 3 392
  • 文/潘曉璐 我一進店門站超,熙熙樓的掌柜王于貴愁眉苦臉地迎上來荸恕,“玉大人,你說我怎么就攤上這事死相∪谇螅” “怎么了?”我有些...
    開封第一講書人閱讀 163,105評論 0 353
  • 文/不壞的土叔 我叫張陵媳纬,是天一觀的道長双肤。 經(jīng)常有香客問我,道長钮惠,這世上最難降的妖魔是什么茅糜? 我笑而不...
    開封第一講書人閱讀 58,242評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮素挽,結果婚禮上蔑赘,老公的妹妹穿的比我還像新娘。我一直安慰自己预明,他們只是感情好缩赛,可當我...
    茶點故事閱讀 67,269評論 6 389
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著撰糠,像睡著了一般酥馍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上阅酪,一...
    開封第一講書人閱讀 51,215評論 1 299
  • 那天旨袒,我揣著相機與錄音汁针,去河邊找鬼。 笑死砚尽,一個胖子當著我的面吹牛施无,可吹牛的內容都是我干的。 我是一名探鬼主播必孤,決...
    沈念sama閱讀 40,096評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼猾骡,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了敷搪?” 一聲冷哼從身側響起兴想,我...
    開封第一講書人閱讀 38,939評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎赡勘,沒想到半個月后襟企,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,354評論 1 311
  • 正文 獨居荒郊野嶺守林人離奇死亡狮含,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,573評論 2 333
  • 正文 我和宋清朗相戀三年顽悼,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片几迄。...
    茶點故事閱讀 39,745評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡蔚龙,死狀恐怖,靈堂內的尸體忽然破棺而出映胁,到底是詐尸還是另有隱情木羹,我是刑警寧澤,帶...
    沈念sama閱讀 35,448評論 5 344
  • 正文 年R本政府宣布解孙,位于F島的核電站坑填,受9級特大地震影響,放射性物質發(fā)生泄漏弛姜。R本人自食惡果不足惜脐瑰,卻給世界環(huán)境...
    茶點故事閱讀 41,048評論 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望廷臼。 院中可真熱鬧苍在,春花似錦、人聲如沸荠商。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,683評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽莱没。三九已至初肉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間饰躲,已是汗流浹背牙咏。 一陣腳步聲響...
    開封第一講書人閱讀 32,838評論 1 269
  • 我被黑心中介騙來泰國打工井佑, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人眠寿。 一個月前我還...
    沈念sama閱讀 47,776評論 2 369
  • 正文 我出身青樓,卻偏偏與公主長得像焦蘑,于是被迫代替她去往敵國和親盯拱。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,652評論 2 354

推薦閱讀更多精彩內容

  • 字典樹介紹 又稱單詞查找樹例嘱,Trie樹狡逢,是一種樹形結構,是一種哈希樹的變種拼卵。典型應用是用于統(tǒng)計奢浑,排序和保存大量的字...
    遠o_O閱讀 5,773評論 1 5
  • Spring Cloud為開發(fā)人員提供了快速構建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務發(fā)現(xiàn)腋腮,斷路器雀彼,智...
    卡卡羅2017閱讀 134,652評論 18 139
  • 3. 在我日復一日的備課講課中,我深深感受到即寡,學生和老師之間是由心靈感應的徊哑,你的行為舉止,他們都會感受到聪富,同樣莺丑,他...
    靜夢園閱讀 563評論 0 1
  • 我相信這不是我一個人的經(jīng)歷,傍晚時分墩蔓,你坐在屋檐下梢莽,看著天慢慢地黑下去。心里寂寞而凄涼奸披,感覺自己的生命被剝奪了昏名。當...
    阿格雷先生閱讀 557評論 5 5