`
ahuaxuan
  • 浏览: 640687 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

使用DFA实现文字过滤

阅读更多
/**
  * author:ahuaxuan(张荣华)
  * date:2009-02-21
  */

Dfa和文字过滤


文字过滤是一般大型网站必不可少的一个功能,而且很多文字类网站更是需要。那么如何设计一个高效的文字过滤系统就是非常重要的了。

文字过滤需求简要描述:判断集合A中哪些子集属于集合B,拿javaeye来说,如果用户发表一篇文章(集合A),我们需要判断这篇文章里是否存在一些关键字是属于集合B,B一般来说就是违禁词列表。

看到这里,没有接触过的同学可能会想到contains,正则之类的方法,但是很遗憾,这些方法都是行不通的。唯一比较好的算法是DFA。

一,DFA简介:
学过编译原理的同学们一定知道,在词法分析阶段将源代码中的文本变成语法的集合就是通过确定有限自动机实现的。但是DFA并不只是词法分析里用到,DFA的用途非常的广泛,并不局限在计算机领域。

DFA的基本功能是可以通过event和当前的state得到下一个state,即event+state=nextstate,
我们来看一张到处都能找到的状态图:


---------------------------------------


-------------------------------------





大写字母是状态,小写字母是动作:我们可以看到S+a=U,U+a=Q,S+b=V等等。一般情况下我们可以用矩阵来表示整个状态转移过程:
---------------
状态\字符  a       b
S        U       V
U        Q       V
V        U       Q
Q        Q       Q

但是表示状态图可以有很多数据结构,上面的矩阵只是一个便于理解的简单例子。而接下来在本文提到的文字过滤系统中会使用另外的数据结构来实现自动机模型

二,文字过滤
在文字过滤系统中,为了能够应付较高的并发,有一个目标比较重要,就是尽量的减少计算,而在DFA中,基本没有什么计算,有的只是状态的转移。而要把违禁文字列表构造成一个状态机,用矩阵来实现是比较麻烦的,下面介绍一种比较简单的实现方式,就是树结构。

所有的违禁词其本质来说是有ascii码组成的,而待过滤文本其本质也是ascii码的集合,比如说:
输入是A=[101,102,105,97,98,112,110]
违禁词列表:
[102,105]
[98,112]
那么我们的任务就是把上面两个违禁词构造成一个DFA,这样输入的A就可以通过在这个DFA上的转移来实现违禁词查找的功能。

树结构实现这个DFA的基于的基本方法是数组的index和数组value之间的关系(在双数组trie中同样是基于这一基本方法)
那么102其实可以看作一个数组索引,而105是102这个索引指向的下一个数组中的一个索引,105后面没有值了,那就代表这个违禁词结束了。

通过这样一种方式,就可以构造出一颗DFA的树结构表示。

接着遍历输入文本中的每一个byte,然后在DFA中作状态转移就可以判断出一个违禁词是否出现在输入文本中。

下面贴出ahuaxuan基于以上理论用python写的一段文字过滤脚本:
#encoding:UTF-8
import sys
from time import time
'''
@author: ahuaxuan 
@date: 2009-02-20
'''

wordTree = [None for x in range(256)]
wordTree.append(0)
nodeTree = [wordTree, 0]
def readInputText():
    txt = ''
    for line in open('text.txt', 'rb'):
        txt = txt + line
    return txt

def createWordTree():
    awords = []
    for b in open('words.txt', 'rb'):
        awords.append(b.strip())
    
    for word in awords:
        temp = wordTree
        for a in range(0,len(word)):
            index = ord(word[a])
            if a < (len(word) - 1):
                if temp[index] == None:
                    node = [[None for x in range(256)],0]
                    temp[index] = node
                elif temp[index] == 1:
                    node = [[None for x in range(256)],1]
                    temp[index] = node
                
                temp = temp[index][0]
            else:
                temp[index] = 1
    

def searchWord(str):
    temp = nodeTree
    words = []
    word = []
    a = 0
    while a < len(str):
        index = ord(str[a])
        temp = temp[0][index]
        if temp == None:
            temp = nodeTree
            a = a - len(word)
            word = []
        elif temp == 1 or temp[1] == 1:
            word.append(index)
            words.append(word)
            a = a - len(word) + 1 
            word = []
            temp = nodeTree
        else:
            word.append(index)
        a = a + 1
    
    return words

if __name__ == '__main__':
    #reload(sys)  
    #sys.setdefaultencoding('GBK')  
    input2 = readInputText()
    createWordTree();
    beign=time()
    list2 = searchWord(input2)
    print "cost time : ",time()-beign
    strLst = []
    print 'I have find some words as ', len(list2)
    map = {}
    for w in list2:
        word = "".join([chr(x) for x in w])
        if not map.__contains__(word):
            map[word] = 1
        else:
            map[word] = map[word] + 1
    
    for key, value in map.items():
        print key, value


输入文本就是本文(不包含下面的示例结果文本)
运行结果示例:
python 5
违禁词 12
DFA 12
ahuaxuan 3

       



当然用python实现以上算法只是为了便于理解,事实上python的速度实在是太慢了,同样的违禁词列表,同样的输入文本,python写的比用java写的差了40倍左右。理论上来讲在这个功能上,用python调用c写的功能比较合适。

而这种方式比较大的缺点是内存使用虑较大,因为有很多数组上的元素是None,引用的空间会消耗大量的内存,这个和违禁词的长度和个数成正比。比较好的方式还是用双数组实现DFA,这个方式使用内存空间较小,而基本原理还是一样,通过两个数组的index和value之间的数学关系来实现状态机的转移。

附件中附带违禁词和输入文本的测试,大家可以运行一下看看效果。

由于ahuaxuan水平有限,如代码中出现问题,还望不吝指出,谢谢。
  • 大小: 51.4 KB
分享到:
评论
5 楼 汪兆铭 2009-02-23  
<p>啥也不说了,看图</p>
<p><br /><img src="/upload/attachment/77758/552361d2-4856-372c-9304-11c5680751cd.jpg" alt="" /></p>
<p>第一次运行是原程序,第二次是全部改用xrang以后的。</p>
<p> </p>
4 楼 ahuaxuan 2009-02-23  
tanleihaoren 写道
要实现文字过滤有很多很好的算法,自己网上查查,用不着这么“高级”。

还有哪些算法,我能查到的只有一篇什么通过散列记录首字的方法,这个方法太小学生了.

看到你回复中"高级"两个字给了引号,让我感到汗颜,想必你有更好的算法,能否讨教一二(别跟我说双数组之类的,那个算法我早搞懂了)

本文给出的方法在时间复杂度上和双数组是一样的,但是空间复杂度比双数组要高,当然如果你能介绍一些比trie或者双数组trie更好的算法给我学习学习我也是非常感谢的,期待中
3 楼 tanleihaoren 2009-02-23  
要实现文字过滤有很多很好的算法,自己网上查查,用不着这么“高级”。
2 楼 ahuaxuan 2009-02-21  
汪兆铭 写道
40倍?
用xrange试试?
现在手头没环境,周一测下看。应该不会差这么多,也许是程序有问题。

xrange用在什么地方?把while替换掉吗?

在输入文本达到1m违禁词超过2000的时候确实相差40倍。python需要1600毫秒,而java只要40-50毫秒

而如果用附件中的输入文本和违禁词列表两者差距则只有5倍,在我的机器上,python是5毫秒,而java是1毫秒
1 楼 汪兆铭 2009-02-21  
40倍?
用xrange试试?
现在手头没环境,周一测下看。应该不会差这么多,也许是程序有问题。

相关推荐

    DFA算法实现敏感词过滤

    本篇将详细介绍如何使用DFA(Deterministic Finite Automaton,确定有限状态自动机)算法实现高效敏感词过滤。 DFA是一种状态机模型,它由一组状态、一个起始状态、一组输入符号以及在状态之间基于输入符号的转移...

    java利用DFA算法实现敏感词过滤功能

    在本文中,我们将探讨如何使用DFA(有穷自动机)算法在Java中实现敏感词过滤功能。敏感词过滤在许多应用程序中都是必要的,例如社交媒体、论坛或博客平台,以防止用户发布不当或有害的内容。以下是对DFA算法及其在...

    java。dfa算法实现敏感词过滤

    本篇将重点介绍如何使用Java实现基于DFA(Deterministic Finite Automaton,确定有限状态自动机)算法的敏感词过滤。 首先,DFA算法是一种图论概念,它可以被视作一种特殊的有向图,每个节点代表一个状态,每条边...

    java使用DFA算法实现敏感词过滤

    本文将详细讲解如何使用Java语言结合DFA(Deterministic Finite Automaton,确定有限状态自动机)算法来实现高效敏感词过滤。 首先,让我们了解什么是DFA算法。DFA是一种特殊的图论模型,它由有限个状态和一些输入...

    Java使用DFA算法实现过滤多家公司自定义敏感字功能详解

    Java使用DFA算法实现过滤多家公司自定义敏感字功能详解主要介绍了Java使用DFA算法实现过滤多家公司自定义敏感字功能,结合实例形式分析了DFA算法的实现原理及过滤敏感字的相关操作技巧。 DFA算法简介 DFA...

    Qt实现DFA敏感词过滤

    文件"filter_DFA"可能包含了实现DFA敏感词过滤的具体代码和资源,包括DFA状态的定义、状态转移函数、敏感词库处理、以及Qt界面与DFA逻辑的集成。分析这些文件可以帮助你更深入地理解实际项目的实现细节。 总的来说...

    高效敏感词过滤JAVA实现(DFA算法) 5000字2ms

    高效敏感词过滤JAVA实现(DFA算法) 5000字2ms 节点 + 2进制标识(节省空间/提高查询效率) 附源码、注释,附带专业敏感词库(3396个敏感词) 看得上就拿去用,替换下一两处util方法、改个路径即可 不求什么,...

    java中DFA算法过滤敏感词

    DFA全称为:Deterministic Finite Automaton,即确定有穷自动机。其特征为:有一个有限状态集合和一些从一个状态通向另一个状态的边,每条边上标记有一个符号,其中一个状态是初态,某些状态是终态。但不同于不确定的...

    基于PHP的DFA算法敏感词过滤器

    **PHP的DFA敏感词过滤器**是利用PHP的字符串处理能力和数据结构来实现DFA的状态转换。这种过滤器的实现可能包括以下几个步骤: 1. **敏感词库构建**:首先,需要收集并整理敏感词列表,然后根据这些词构建DFA的状态...

    使用DFA算法实现的内容安全反垃圾智能鉴黄敏感词过滤

    当用户输入的文字通过这个系统时,DFA会逐字符检查,一旦发现匹配到敏感词的开始,就会立即触发警告或删除该内容。 具体到实现上,这个项目可能包含以下几个关键部分: 1. **敏感词库构建**:首先,需要建立一个...

    DFA实例敏感词过滤实例,附带关键词文件

    总结一下,这个实例展示了如何使用DFA算法和Java编程实现一个敏感词过滤系统。`SensitiveWordUtil`作为核心工具类,负责构建和操作DFA,`censorwords.prop`文件则是敏感词的来源。这样的系统能够有效地在文本处理...

    Golang基于DFA算法实现的敏感词过滤

    【作品名称】:Golang基于DFA算法实现的敏感词过滤 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。 【项目介绍】: 基于DFA算法; ...

    编译原理NFA转DFA实现(python).zip

    NFA转DFA是编译器设计的一个关键步骤,它有助于理解和实现正则表达式的求解过程。本实验是基于Python语言进行的NFA到DFA的转换实践。 首先,我们需要理解NFA和DFA的基本概念。NFA是一种允许存在多个可选路径的...

    编译原理 DFA编程实现

    在这个主题中,“DFA编程实现”是关于确定有限自动机(Deterministic Finite Automaton,DFA)的编程实践。DFA是一种状态机模型,用于识别或接受特定的字符串,常用于词法分析阶段,即编译器的第一步。 DFA由一组...

    过滤敏感词汇的laravel包使用DFA算法

    1. DFA类:实现DFA算法的类,用于构建和执行状态转移。 2. 敏感词库:包含了敏感词汇,可能以数组或者数据库形式存储。 3. 过滤器接口:提供一个方便的接口,让开发者可以轻松地在路由、控制器、视图等不同位置调用...

    DFA,NFA实现

    例如,编译器的词法分析阶段通常使用有限状态自动机来识别源代码中的各种符号和关键字。 7. **学习资源** 提供的压缩包“FSA”可能包含实现这些算法的源代码,这对于学习和理解FSA的内部工作原理非常有帮助。通过...

    编译原理Java实现NFA到DFA的等价变换

    本文将介绍使用Java语言实现NFA到DFA的等价变换。 第一部分:NFA和DFA的概念 Deterministic Finite Automaton(DFA)是一种特殊的自动机,它的状态变迁函数是一个确定的函数,即给定当前状态和输入符号,下一个...

    DFA算法实现的敏感词过滤.zip

    DFA算法实现的敏感词过滤.zip DFA算法实现的敏感词过滤工具,支持Skip参数控制敏感词干扰噪音,支持白名单跳过白名单词汇,支持在线添加和删除敏感词,管理敏感词库。 程序会跳过不同的距离,查找敏感词,距离越长,...

    NFA转DFA,并将DFA最小化

    本项目通过VS2015和C++语言,实现了一个完整的流程,即从正则表达式构建NFA,再将其转换为DFA,并对DFA进行最小化。 首先,我们需要理解NFA(非确定性有限自动机)。NFA是一种状态机,其中每个状态可以有多个出边,...

    基于dfa的敏感词过滤的小商城系统,我自己开发的。。莫得版权问题哇,为啥不通过

    我开发的这个小商城系统,采用了创新的基于确定有限状态自动机(DFA)的敏感词过滤机制,并巧妙融合了协同过滤算法来实现商品推荐功能。系统后端主要采用JavaWeb技术栈,包含但不限于Servlet、JSP和Spring框架,前端...

Global site tag (gtag.js) - Google Analytics