-
Notifications
You must be signed in to change notification settings - Fork 41
/
Copy pathLangModel.py
206 lines (177 loc) · 9.46 KB
/
LangModel.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
import re
import math
# used for unseen words in training vocabularies
UNK = None
# sentence start and end
SENTENCE_START = "<s>"
SENTENCE_END = "</s>"
def read_sentences_from_file(file_path):
with open(file_path, "r") as f:
return [re.split("\s+", line.rstrip('\n')) for line in f]
class UnigramLanguageModel:
def __init__(self, sentences, smoothing=False):
self.unigram_frequencies = dict()
self.corpus_length = 0
for sentence in sentences:
for word in sentence:
self.unigram_frequencies[word] = self.unigram_frequencies.get(word, 0) + 1
if word != SENTENCE_START and word != SENTENCE_END:
self.corpus_length += 1
# subtract 2 because unigram_frequencies dictionary contains values for SENTENCE_START and SENTENCE_END
self.unique_words = len(self.unigram_frequencies) - 2
self.smoothing = smoothing
def calculate_unigram_probability(self, word):
word_probability_numerator = self.unigram_frequencies.get(word, 0)
word_probability_denominator = self.corpus_length
if self.smoothing:
word_probability_numerator += 1
# add one more to total number of seen unique words for UNK - unseen events
word_probability_denominator += self.unique_words + 1
return float(word_probability_numerator) / float(word_probability_denominator)
def calculate_sentence_probability(self, sentence, normalize_probability=True):
sentence_probability_log_sum = 0
for word in sentence:
if word != SENTENCE_START and word != SENTENCE_END:
word_probability = self.calculate_unigram_probability(word)
sentence_probability_log_sum += math.log(word_probability, 2)
return math.pow(2, sentence_probability_log_sum) if normalize_probability else sentence_probability_log_sum
def sorted_vocabulary(self):
full_vocab = list(self.unigram_frequencies.keys())
full_vocab.remove(SENTENCE_START)
full_vocab.remove(SENTENCE_END)
full_vocab.sort()
full_vocab.append(UNK)
full_vocab.append(SENTENCE_START)
full_vocab.append(SENTENCE_END)
return full_vocab
class BigramLanguageModel(UnigramLanguageModel):
def __init__(self, sentences, smoothing=False):
UnigramLanguageModel.__init__(self, sentences, smoothing)
self.bigram_frequencies = dict()
self.unique_bigrams = set()
for sentence in sentences:
previous_word = None
for word in sentence:
if previous_word != None:
self.bigram_frequencies[(previous_word, word)] = self.bigram_frequencies.get((previous_word, word),
0) + 1
if previous_word != SENTENCE_START and word != SENTENCE_END:
self.unique_bigrams.add((previous_word, word))
previous_word = word
# we subtracted two for the Unigram model as the unigram_frequencies dictionary
# contains values for SENTENCE_START and SENTENCE_END but these need to be included in Bigram
self.unique__bigram_words = len(self.unigram_frequencies)
def calculate_bigram_probabilty(self, previous_word, word):
bigram_word_probability_numerator = self.bigram_frequencies.get((previous_word, word), 0)
bigram_word_probability_denominator = self.unigram_frequencies.get(previous_word, 0)
if self.smoothing:
bigram_word_probability_numerator += 1
bigram_word_probability_denominator += self.unique__bigram_words
return 0.0 if bigram_word_probability_numerator == 0 or bigram_word_probability_denominator == 0 else float(
bigram_word_probability_numerator) / float(bigram_word_probability_denominator)
def calculate_bigram_sentence_probability(self, sentence, normalize_probability=True):
bigram_sentence_probability_log_sum = 0
previous_word = None
for word in sentence:
if previous_word != None:
bigram_word_probability = self.calculate_bigram_probabilty(previous_word, word)
bigram_sentence_probability_log_sum += math.log(bigram_word_probability, 2)
previous_word = word
return math.pow(2,
bigram_sentence_probability_log_sum) if normalize_probability else bigram_sentence_probability_log_sum
# calculate number of unigrams & bigrams
def calculate_number_of_unigrams(sentences):
unigram_count = 0
for sentence in sentences:
# remove two for <s> and </s>
unigram_count += len(sentence) - 2
return unigram_count
def calculate_number_of_bigrams(sentences):
bigram_count = 0
for sentence in sentences:
# remove one for number of bigrams in sentence
bigram_count += len(sentence) - 1
return bigram_count
# print unigram and bigram probs
def print_unigram_probs(sorted_vocab_keys, model):
for vocab_key in sorted_vocab_keys:
if vocab_key != SENTENCE_START and vocab_key != SENTENCE_END:
print("{}: {}".format(vocab_key if vocab_key != UNK else "UNK",
model.calculate_unigram_probability(vocab_key)), end=" ")
print("")
def print_bigram_probs(sorted_vocab_keys, model):
print("\t\t", end="")
for vocab_key in sorted_vocab_keys:
if vocab_key != SENTENCE_START:
print(vocab_key if vocab_key != UNK else "UNK", end="\t\t")
print("")
for vocab_key in sorted_vocab_keys:
if vocab_key != SENTENCE_END:
print(vocab_key if vocab_key != UNK else "UNK", end="\t\t")
for vocab_key_second in sorted_vocab_keys:
if vocab_key_second != SENTENCE_START:
print("{0:.5f}".format(model.calculate_bigram_probabilty(vocab_key, vocab_key_second)), end="\t\t")
print("")
print("")
# calculate perplexty
def calculate_unigram_perplexity(model, sentences):
unigram_count = calculate_number_of_unigrams(sentences)
sentence_probability_log_sum = 0
for sentence in sentences:
try:
sentence_probability_log_sum -= math.log(model.calculate_sentence_probability(sentence), 2)
except:
sentence_probability_log_sum -= float('-inf')
return math.pow(2, sentence_probability_log_sum / unigram_count)
def calculate_bigram_perplexity(model, sentences):
number_of_bigrams = calculate_number_of_bigrams(sentences)
bigram_sentence_probability_log_sum = 0
for sentence in sentences:
try:
bigram_sentence_probability_log_sum -= math.log(model.calculate_bigram_sentence_probability(sentence), 2)
except:
bigram_sentence_probability_log_sum -= float('-inf')
return math.pow(2, bigram_sentence_probability_log_sum / number_of_bigrams)
if __name__ == '__main__':
toy_dataset = read_sentences_from_file("./sampledata.txt")
toy_dataset_test = read_sentences_from_file("./sampletest.txt")
toy_dataset_model_unsmoothed = BigramLanguageModel(toy_dataset)
toy_dataset_model_smoothed = BigramLanguageModel(toy_dataset, smoothing=True)
sorted_vocab_keys = toy_dataset_model_unsmoothed.sorted_vocabulary()
print("---------------- Toy dataset ---------------\n")
print("=== UNIGRAM MODEL ===")
print("- Unsmoothed -")
print_unigram_probs(sorted_vocab_keys, toy_dataset_model_unsmoothed)
print("\n- Smoothed -")
print_unigram_probs(sorted_vocab_keys, toy_dataset_model_smoothed)
print("")
print("=== BIGRAM MODEL ===")
print("- Unsmoothed -")
print_bigram_probs(sorted_vocab_keys, toy_dataset_model_unsmoothed)
print("- Smoothed -")
print_bigram_probs(sorted_vocab_keys, toy_dataset_model_smoothed)
print("")
print("== SENTENCE PROBABILITIES == ")
longest_sentence_len = max([len(" ".join(sentence)) for sentence in toy_dataset_test]) + 5
print("sent", " " * (longest_sentence_len - len("sent") - 2), "uprob\t\tbiprob")
for sentence in toy_dataset_test:
sentence_string = " ".join(sentence)
print(sentence_string, end=" " * (longest_sentence_len - len(sentence_string)))
print("{0:.5f}".format(toy_dataset_model_smoothed.calculate_sentence_probability(sentence)), end="\t\t")
print("{0:.5f}".format(toy_dataset_model_smoothed.calculate_bigram_sentence_probability(sentence)))
print("")
print("== TEST PERPLEXITY == ")
print("unigram: ", calculate_unigram_perplexity(toy_dataset_model_smoothed, toy_dataset_test))
print("bigram: ", calculate_bigram_perplexity(toy_dataset_model_smoothed, toy_dataset_test))
print("")
actual_dataset = read_sentences_from_file("./train.txt")
actual_dataset_test = read_sentences_from_file("./test.txt")
actual_dataset_model_smoothed = BigramLanguageModel(actual_dataset, smoothing=True)
print("---------------- Actual dataset ----------------\n")
print("PERPLEXITY of train.txt")
print("unigram: ", calculate_unigram_perplexity(actual_dataset_model_smoothed, actual_dataset))
print("bigram: ", calculate_bigram_perplexity(actual_dataset_model_smoothed, actual_dataset))
print("")
print("PERPLEXITY of test.txt")
print("unigram: ", calculate_unigram_perplexity(actual_dataset_model_smoothed, actual_dataset_test))
print("bigram: ", calculate_bigram_perplexity(actual_dataset_model_smoothed, actual_dataset_test))