Python实现单词拼写检查


Posted in Python onApril 25, 2015

这几天在翻旧代码时发现以前写的注释部分有很多单词拼写错误,这些单词错得不算离谱,应该可以用工具自动纠错绝大部分。用 Python 写个拼写检查脚本很容易,如果能很好利用 aspell/ispell 这些现成的小工具就更简单了。

要点

1、输入一个拼写错误的单词,调用 aspell -a 后得到一些候选正确单词,然后用距离编辑进一步?鹧〕龈??返拇省1热缭诵 aspell -a,输入 ‘hella' 后得到如下结果:
hell, Helli, hello, heal, Heall, he'll, hells, Heller, Ella, Hall, Hill, Hull, hall, heel, hill, hula, hull, Helga, Helsa, Bella, Della, Mella, Sella, fella, Halli, Hally, Hilly, Holli, Holly, hallo, hilly, holly, hullo, Hell's, hell's

2、什么是距离编辑(Edit-Distance,也叫 Levenshtein algorithm)呢?就是说给定一个单词,通过多次插入、删除、交换、替换单字符的操作后枚举出所有可能的正确拼写,比如输入 ‘hella',经过多次插入、删除、交换、替换单字符的操作后变成:
‘helkla', ‘hjlla', ‘hylla', ‘hellma', ‘khella', ‘iella', ‘helhla', ‘hellag', ‘hela', ‘vhella', ‘hhella', ‘hell', ‘heglla', ‘hvlla', ‘hellaa', ‘ghella', ‘hellar', ‘heslla', ‘lhella', ‘helpa', ‘hello', …

3、综合上面2个集合的结果,并且考虑到一些理论知识可以提高拼写检查的准确度,比如一般来说写错单词都是无意的或者误打,完全错的单词可能性很小,而且单词的第一个字母一般不会拼错。所以可以在上面集合里去掉第一个字母不符合的单词,比如:'Sella', ‘Mella', khella', ‘iella' 等,这里 VPSee 不删除单词,而把这些单词从队列里取出来放到队列最后(优先级降低),所以实在匹配不了以 h 开头的单词才去匹配那些以其他字母开头的单词。

4、程序中用到了外部工具 aspell,如何在 Python 里捕捉外部程序的输入和输出以便在 Python 程序里处理这些输入和输出呢?Python 2.4 以后引入了 subprocess 模块,可以用 subprocess.Popen 来处理。

5、Google 大牛 Peter Norvig 写了一篇 How to Write a Spelling Corrector 很值得一看,大牛就是大牛,21行 Python 就解决拼写问题,而且还不用外部工具,只需要事先读入一个词典文件。本文程序的 edits1 函数就是从牛人家那里 copy 的。

代码

 

#!/usr/bin/python
# A simple spell checker

import os, sys, subprocess, signal

alphabet = 'abcdefghijklmnopqrstuvwxyz'

def found(word, args, cwd = None, shell = True):
  child = subprocess.Popen(args, 
    shell = shell, 
    stdin = subprocess.PIPE, 
    stdout = subprocess.PIPE, 
    cwd = cwd, 
    universal_newlines = True) 
  child.stdout.readline()
  (stdout, stderr) = child.communicate(word)
  if ": " in stdout:
    # remove \n\n
    stdout = stdout.rstrip("\n")
    # remove left part until :
    left, candidates = stdout.split(": ", 1) 
    candidates = candidates.split(", ")
    # making an error on the first letter of a word is less 
    # probable, so we remove those candidates and append them 
    # to the tail of queue, make them less priority
    for item in candidates:
      if item[0] != word[0]: 
        candidates.remove(item)
        candidates.append(item)
    return candidates
  else:
    return None

# copy from http://norvig.com/spell-correct.html
def edits1(word):
  n = len(word)
  return set([word[0:i]+word[i+1:] for i in range(n)] +           
    [word[0:i]+word[i+1]+word[i]+word[i+2:] for i in range(n-1)] +
    [word[0:i]+c+word[i+1:] for i in range(n) for c in alphabet] +
    [word[0:i]+c+word[i:] for i in range(n+1) for c in alphabet])

def correct(word):
  candidates1 = found(word, 'aspell -a')
  if not candidates1:
    print "no suggestion"
    return 

  candidates2 = edits1(word)
  candidates = []
  for word in candidates1:
    if word in candidates2:
      candidates.append(word)
  if not candidates:
    print "suggestion: %s" % candidates1[0]
  else:
    print "suggestion: %s" % max(candidates)

def signal_handler(signal, frame):
  sys.exit(0)

if __name__ == '__main__':
  signal.signal(signal.SIGINT, signal_handler)
  while True:
    input = raw_input()
    correct(input)

更简单的方法

当然直接在程序里调用相关模块最简单了,有个叫做 PyEnchant 的库支持拼写检查,安装 PyEnchant 和 Enchant 后就可以直接在 Python 程序里 import 了:

>>> import enchant
>>> d = enchant.Dict("en_US")
>>> d.check("Hello")
True
>>> d.check("Helo")
False
>>> d.suggest("Helo")
['He lo', 'He-lo', 'Hello', 'Helot', 'Help', 'Halo', 'Hell', 'Held', 'Helm', 'Hero', "He'll"]
>>>
Python 相关文章推荐
python中requests模块的使用方法
Apr 08 Python
Python实现的最近最少使用算法
Jul 10 Python
简单讲解Python中的字符串与字符串的输入输出
Mar 13 Python
详解Python中的相对导入和绝对导入
Jan 06 Python
Python操作SQLite数据库的方法详解【导入,创建,游标,增删改查等】
Jul 11 Python
python 3.0 模拟用户登录功能并实现三次错误锁定
Nov 01 Python
python实现AES加密解密
Mar 28 Python
Django中间件基础用法详解
Jul 18 Python
window7下的python2.7版本和python3.5版本的opencv-python安装过程
Oct 24 Python
使用python快速在局域网内搭建http传输文件服务的方法
Nov 14 Python
pytorch 实现查看网络中的参数
Jan 06 Python
Python eval函数原理及用法解析
Nov 14 Python
在Debian下配置Python+Django+Nginx+uWSGI+MySQL的教程
Apr 25 #Python
使用PDB简单调试Python程序简明指南
Apr 25 #Python
Python脚本判断 Linux 是否运行在虚拟机上
Apr 25 #Python
在Python中使用cookielib和urllib2配合PyQuery抓取网页信息
Apr 25 #Python
使用Python的Tornado框架实现一个一对一聊天的程序
Apr 25 #Python
使用Python发送邮件附件以定时备份MySQL的教程
Apr 25 #Python
安装Python的web.py框架并从hello world开始编程
Apr 25 #Python
You might like
发挥语言的威力--融合PHP与ASP
2006/10/09 PHP
PHP chmod 函数与批量修改文件目录权限
2010/05/10 PHP
PHP对象递归引用造成内存泄漏分析
2014/08/28 PHP
深入浅析php中sprintf与printf函数的用法及区别
2016/01/08 PHP
找到了一篇jQuery与Prototype并存的冲突的解决方法
2007/08/29 Javascript
ExtJS PropertyGrid中使用Combobox选择值问题
2010/06/13 Javascript
xss文件页面内容读取(解决)
2010/11/28 Javascript
JQuery.Ajax之错误调试帮助信息介绍
2013/07/04 Javascript
js 时间函数应用加、减、比较、格式转换的示例代码
2013/08/23 Javascript
借助JavaScript脚本判断浏览器Flash Player信息的方法
2014/07/09 Javascript
JavaScript函数定义的常见注意事项小结
2014/09/16 Javascript
基于javascript简单实现对身份证校验
2021/01/25 Javascript
nodejs 中模拟实现 emmiter 自定义事件
2016/02/22 NodeJs
Bootstrap框架的学习教程详解(二)
2016/10/18 Javascript
Vue axios 中提交表单数据(含上传文件)
2017/07/06 Javascript
JavaScript创建对象的七种方式全面总结
2017/08/21 Javascript
jQuery实现小火箭返回顶部特效
2020/02/03 jQuery
js实现可爱的气泡特效
2020/09/05 Javascript
闭包在python中的应用之translate和maketrans用法详解
2014/08/27 Python
python类和继承用法实例
2015/07/07 Python
Python基于pygame实现的font游戏字体(附源码)
2015/11/11 Python
Python程序中的观察者模式结构编写示例
2016/05/27 Python
python 性能提升的几种方法
2016/07/15 Python
python获取微信企业号打卡数据并生成windows计划任务
2019/04/30 Python
python pandas写入excel文件的方法示例
2019/06/25 Python
Python面向对象程序设计之继承、多态原理与用法详解
2020/03/23 Python
python把一个字符串切开的实例方法
2020/09/27 Python
美国大型的健身社区和补充商店:Bodybuilding.com
2016/09/06 全球购物
WoolOvers澳洲官方网站:英国针织服装公司
2018/05/13 全球购物
饿了么订餐官网:外卖、网上订餐
2019/06/28 全球购物
2013年最新自荐信范文
2014/06/23 职场文书
社会发展项目建议书
2014/08/25 职场文书
2014标准社保办理委托书
2014/10/06 职场文书
恰同学少年观后感
2015/06/08 职场文书
电工实训心得体会
2016/01/14 职场文书
详解JVM系列之内存模型
2021/06/10 Javascript