https://stackoverflow.com/users/1515832/generic-human 의 답변 은 훌륭합니다. 그러나 내가 본 최고의 구현은 Peter Norvig가 자신의 책 'Beautiful Data'에 직접 썼습니다.
그의 코드를 붙여 넣기 전에 Norvig의 방법이 더 정확한 이유를 확장 해 보겠습니다 (코드 측면에서 약간 느리고 길지만).
1) 데이터는 크기와 정밀도 측면에서 조금 더 좋습니다 (단순한 순위보다는 단어 수를 사용합니다) .2) 더 중요한 것은 접근 방식을 매우 정확하게 만드는 것은 n-gram의 논리입니다. .
그가 책에서 제공하는 예는 'sitdown'문자열을 분할하는 문제입니다. 이제 비 그램 문자열 분할 방법은 p ( 'sit') * p ( 'down')를 고려할 것이며, 이것이 p ( 'sitdown')보다 작 으면-매우 자주 발생합니다-분할되지 않습니다 하지만 우리는 (대부분의 경우) 원합니다.
그러나 bigram 모델을 사용하면 p ( 'sit down')를 bigram 대 p ( 'sitdown')으로 평가할 수 있으며 전자가 이깁니다. 기본적으로 bigrams를 사용하지 않으면 분할하는 단어의 확률을 독립적으로 처리합니다. 그렇지 않은 경우 일부 단어가 차례로 나타날 가능성이 더 높습니다. 불행히도 그것들은 종종 많은 경우에 함께 붙어서 스플리터를 혼동하는 단어입니다.
다음은 데이터 링크입니다 (3 개의 개별 문제에 대한 데이터이며 세분화는 단 하나입니다. 자세한 내용은 해당 장을 참조하십시오). http://norvig.com/ngrams/
다음은 코드 링크입니다. http://norvig.com/ngrams/ngrams.py
이 링크는 한동안 작동했지만 어쨌든 여기에 코드의 분할 부분을 복사하여 붙여 넣겠습니다.
import re, string, random, glob, operator, heapq
from collections import defaultdict
from math import log10
def memo(f):
"Memoize function f."
table = {}
def fmemo(*args):
if args not in table:
table[args] = f(*args)
return table[args]
fmemo.memo = table
return fmemo
def test(verbose=None):
"""Run some tests, taken from the chapter.
Since the hillclimbing algorithm is randomized, some tests may fail."""
import doctest
print 'Running tests...'
doctest.testfile('ngrams-test.txt', verbose=verbose)
################ Word Segmentation (p. 223)
@memo
def segment(text):
"Return a list of words that is the best segmentation of text."
if not text: return []
candidates = ([first]+segment(rem) for first,rem in splits(text))
return max(candidates, key=Pwords)
def splits(text, L=20):
"Return a list of all possible (first, rem) pairs, len(first)<=L."
return [(text[:i+1], text[i+1:])
for i in range(min(len(text), L))]
def Pwords(words):
"The Naive Bayes probability of a sequence of words."
return product(Pw(w) for w in words)
#### Support functions (p. 224)
def product(nums):
"Return the product of a sequence of numbers."
return reduce(operator.mul, nums, 1)
class Pdist(dict):
"A probability distribution estimated from counts in datafile."
def __init__(self, data=[], N=None, missingfn=None):
for key,count in data:
self[key] = self.get(key, 0) + int(count)
self.N = float(N or sum(self.itervalues()))
self.missingfn = missingfn or (lambda k, N: 1./N)
def __call__(self, key):
if key in self: return self[key]/self.N
else: return self.missingfn(key, self.N)
def datafile(name, sep='\t'):
"Read key,value pairs from file."
for line in file(name):
yield line.split(sep)
def avoid_long_words(key, N):
"Estimate the probability of an unknown word."
return 10./(N * 10**len(key))
N = 1024908267229 ## Number of tokens
Pw = Pdist(datafile('count_1w.txt'), N, avoid_long_words)
#### segment2: second version, with bigram counts, (p. 226-227)
def cPw(word, prev):
"Conditional probability of word, given previous word."
try:
return P2w[prev + ' ' + word]/float(Pw[prev])
except KeyError:
return Pw(word)
P2w = Pdist(datafile('count_2w.txt'), N)
@memo
def segment2(text, prev='<S>'):
"Return (log P(words), words), where words is the best segmentation."
if not text: return 0.0, []
candidates = [combine(log10(cPw(first, prev)), first, segment2(rem, first))
for first,rem in splits(text)]
return max(candidates)
def combine(Pfirst, first, (Prem, rem)):
"Combine first and rem results into one (probability, words) pair."
return Pfirst+Prem, [first]+rem