782 lines
39 KiB
Plaintext
782 lines
39 KiB
Plaintext
|
{
|
||
|
"cells": [
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"![Logo 1](https://git.wmi.amu.edu.pl/AITech/Szablon/raw/branch/master/Logotyp_AITech1.jpg)\n",
|
||
|
"<div class=\"alert alert-block alert-info\">\n",
|
||
|
"<h1> Modelowanie języka</h1>\n",
|
||
|
"<h2> 8. <i>Neuronowy model języka</i> [wykład]</h2> \n",
|
||
|
"<h3> Filip Graliński (2022)</h3>\n",
|
||
|
"</div>\n",
|
||
|
"\n",
|
||
|
"![Logo 2](https://git.wmi.amu.edu.pl/AITech/Szablon/raw/branch/master/Logotyp_AITech2.jpg)\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"## Neuronowy n-gramowy model języka\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Omówiony w poprzedniej części neuronowy bigramowy model języka\n",
|
||
|
"warunkuje kolejny wyraz jedynie względem bezpośrednio poprzedzającego\n",
|
||
|
"— jak w każdym bigramowym modelu przyjmujemy założenie, że $w_i$\n",
|
||
|
"zależy tylko od $w_{i-1}$. Rzecz jasna jest to bardzo duże\n",
|
||
|
"ograniczenie, w rzeczywistości bardzo często prawdopodobieństwo\n",
|
||
|
"kolejnego wyrazu zależy od wyrazu dwie, trzy, cztery itd. pozycje\n",
|
||
|
"wstecz czy w ogólności od wszystkich wyrazów poprzedzających (bez\n",
|
||
|
"względu na ich pozycje).\n",
|
||
|
"\n",
|
||
|
"**Pytanie**: Wskaż zależności o zasięgu większym niż 1 wyraz w zdaniu\n",
|
||
|
"/Zatopieni w kłębach dymu cygar i pochyleni nad butelkami z ciemnego\n",
|
||
|
"szkła obywatele tej dzielnicy, jedni zakładali się o wygranę lub\n",
|
||
|
"przegranę Anglii, drudzy o bankructwo Wokulskiego; jedni nazywali\n",
|
||
|
"geniuszem Bismarcka, drudzy — awanturnikiem Wokulskiego; jedni\n",
|
||
|
"krytykowali postępowanie prezydenta MacMahona, inni twierdzili, że\n",
|
||
|
"Wokulski jest zdecydowanym wariatem, jeżeli nie czymś gorszym…/\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Trigramowy neuronowy model języka\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Spróbujmy najpierw rozszerzyć nasz model na trigramy, to znaczy\n",
|
||
|
"będziemy przewidywać słowo $w_i$ na podstawie słów $w_{i-2}$ i\n",
|
||
|
"$w_{i-1}$.\n",
|
||
|
"\n",
|
||
|
"Najprostsze rozwiązanie polegałoby na zanurzeniu pary $(w_{i-2},\n",
|
||
|
"w_{i-1})$ w całości i postępowaniu jak w przypadku modelu bigramowego.\n",
|
||
|
"Byłoby to jednak zupełnie niepraktyczne, jako że:\n",
|
||
|
"\n",
|
||
|
"- liczba zanurzeń do wyuczenia byłaby olbrzymia ($|V|^2$ — byłoby to\n",
|
||
|
" ewentualnie akceptowalne dla modeli operujących na krótszych\n",
|
||
|
" jednostkach niż słowa, np. na znakach),\n",
|
||
|
"- w szczególności zanurzenia dla par $(v, u)$, $(u, v)$, $(u, u)$ i\n",
|
||
|
" $(v, v)$ nie miałyby ze sobą nic wspólnego.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Konketanacja zanurzeń\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Właściwsze rozwiązanie polega na zanurzeniu dalej pojedynczych słów i\n",
|
||
|
"następnie ich **konkatenowaniu**.\n",
|
||
|
"\n",
|
||
|
"Przypomnijmy, że konkatenacja wektorów $\\vec{x_1}$ i $\\vec{x_2}$ to wektor o rozmiarze\n",
|
||
|
"$|\\vec{x_1}| + |\\vec{x_2}|$ powstały ze „sklejania” wektorów $\\vec{x_1}$ i $\\vec{x_2}$.\n",
|
||
|
"Konkatenację wektorów $\\vec{x_1}$ i $\\vec{x_2}$ będziemy oznaczać za pomocą $[\\vec{x_1}, \\vec{x_2}]$.\n",
|
||
|
"\n",
|
||
|
"Przykład: jeśli $\\vec{x_1} = [-1, 2, 0]$ i $\\vec{x_2} = [3, -3]$,\n",
|
||
|
"wówczas $[\\vec{x_1}, \\vec{x_2}] = [-1, 2, 0, 3, -3]$\n",
|
||
|
"\n",
|
||
|
"Oznacza to, że nasza macierz „kontekstowa” $C$ powinna mieć w modelu trigramowym rozmiar nie\n",
|
||
|
"$|V| \\times m$, lecz $|V| \\times (m+m)$ = $|V| \\times 2m$ i wyjście będzie zdefiniowane za pomocą wzoru:\n",
|
||
|
"\n",
|
||
|
"$$\\vec{y} = \\operatorname{softmax}(C[E(w_{i-2}),E(w_{i-1})]),$$\n",
|
||
|
"\n",
|
||
|
"co można przedstawić za pomocą następującego schematu:\n",
|
||
|
"\n",
|
||
|
"![img](./08_Neuronowy_ngramowy_model/trigram1.drawio.png \"Diagram prostego bigramowego neuronowego modelu języka\")\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"##### Rozbicie macierzy $C$\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Zamiast mnożyć macierz $C$ przez konkatenację dwóch wektorów, można\n",
|
||
|
"rozbić macierz $C$ na dwie, powiedzmy $C_{-2}$ i $C_{-1}$, przemnażać\n",
|
||
|
"je osobno przez odpowiadające im wektory i następnie **dodać** macierze,\n",
|
||
|
"tak aby:\n",
|
||
|
"\n",
|
||
|
"$$C[E(w_{i-2}),E(w_{i-1})] = C_{-2}E(w_{i-2}) + C_{-1}E(w_{i-1}).$$\n",
|
||
|
"\n",
|
||
|
"Macierze $C_{-2}$ i $C_{-1}$ będą miały rozmiar $|V| \\times m$.\n",
|
||
|
"\n",
|
||
|
"Przy tym podejściu możemy powiedzieć, że ostatni i przedostatni wyraz\n",
|
||
|
"mają swoje osobne macierze o potencjalnie różnych wagach — co ma sens,\n",
|
||
|
"jako że na inne aspekty zwracamy uwagę przewidując kolejne słowo na\n",
|
||
|
"podstawie wyrazu bezpośrednio poprzedzającego, a na inne — na\n",
|
||
|
"podstawie słowa występującego dwie pozycje wcześniej.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Uogólnienie na $n$-gramowy model języka dla dowolnego $n$\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Łatwo uogólnić opisany wyżej trigramowy model języka dla dowolnego $n$.\n",
|
||
|
"Uogólniony model można przedstawić za pomocą wzoru:\n",
|
||
|
"\n",
|
||
|
"$$\\vec{y} = \\operatorname{softmax}(C[E(w_{i-n+1}),\\dots,E(w_{i-1})]),$$\n",
|
||
|
"\n",
|
||
|
"gdzie macierz $C$ ma rozmiar $|V| \\times nm$ lub za pomocą wzoru:\n",
|
||
|
"\n",
|
||
|
"$$\\vec{y} = \\operatorname{softmax}(C_{-(n-1)}E(w_{i-n+1}) + \\dots + C_{-1}E(w_{i-1}),$$\n",
|
||
|
"\n",
|
||
|
"gdzie macierze $C_{-(n-1)}$, …, $C_{-1}$ mają rozmiary $|V| \\times m$.\n",
|
||
|
"\n",
|
||
|
"Por. diagram:\n",
|
||
|
"\n",
|
||
|
"![img](./08_Neuronowy_ngramowy_model/ngram.drawio.png \"Diagram prostego n-gramowego neuronowego modelu języka\")\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Dodanie kolejnej warstwy\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"W wypadku trigramowego czy — ogólniej — n-gramowego modelu języka dla\n",
|
||
|
"$n \\geq 3$ warto dodać kolejną (**ukrytą**) warstwę, na którą będziemy rzutować\n",
|
||
|
"skonkatenowane embeddingi, zanim zrzutujemy je do długiego wektora\n",
|
||
|
"prawdopodobieństw.\n",
|
||
|
"\n",
|
||
|
"Zakładamy, że warstwa ukryta zawiera $h$ neuronów. Wartość $h$ powinna być mniejsza\n",
|
||
|
"niż $nm$ (a może nawet od $m$).\n",
|
||
|
"\n",
|
||
|
"**Pytanie**: Dlaczego wartość $h > nm$ nie jest racjonalnym wyborem?\n",
|
||
|
"\n",
|
||
|
"**Pytanie**: Dlaczego dodanie kolejnej warstwy nie ma sensu dla modelu bigramowego?\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Funkcja aktywacji\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Aby warstwa ukryta wnosiła coś nowego, na wyjściu z tej funkcji musimy (dlaczego?)\n",
|
||
|
"zastosować nieliniową **funkcji aktywacji**. Zazwyczaj jako funkcji\n",
|
||
|
"aktywacji w sieciach neuronowych używa się funkcji ReLU albo funkcji\n",
|
||
|
"sigmoidalnej. W prostych neuronowych modelach języka sprawdza się też\n",
|
||
|
"**tangens hiperboliczny** (tgh, w literaturze anglojęzycznej tanh):\n",
|
||
|
"\n",
|
||
|
"$$\\operatorname{tgh}(x) = \\frac{e^x - e^{-x}}{e^x + e^{-x}}.$$\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "code",
|
||
|
"execution_count": 2,
|
||
|
"metadata": {},
|
||
|
"outputs": [
|
||
|
{
|
||
|
"name": "stderr",
|
||
|
"output_type": "stream",
|
||
|
"text": [
|
||
|
"/media/kuba/ssdsam/anaconda3/envs/lmzajecia/lib/python3.10/site-packages/torch/nn/functional.py:1933: UserWarning: nn.functional.tanh is deprecated. Use torch.tanh instead.\n",
|
||
|
" warnings.warn(\"nn.functional.tanh is deprecated. Use torch.tanh instead.\")\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"data": {
|
||
|
"text/plain": [
|
||
|
"'08_Neuronowy_ngramowy_model/tanh.png'"
|
||
|
]
|
||
|
},
|
||
|
"execution_count": 2,
|
||
|
"metadata": {},
|
||
|
"output_type": "execute_result"
|
||
|
},
|
||
|
{
|
||
|
"data": {
|
||
|
"image/png": "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
|
||
|
"text/plain": [
|
||
|
"<Figure size 432x288 with 1 Axes>"
|
||
|
]
|
||
|
},
|
||
|
"metadata": {
|
||
|
"needs_background": "light"
|
||
|
},
|
||
|
"output_type": "display_data"
|
||
|
}
|
||
|
],
|
||
|
"source": [
|
||
|
"import matplotlib.pyplot as plt\n",
|
||
|
"import torch\n",
|
||
|
"import torch.nn as nn\n",
|
||
|
"\n",
|
||
|
"x = torch.linspace(-5,5,100)\n",
|
||
|
"plt.xlabel(\"x\")\n",
|
||
|
"plt.ylabel(\"y\")\n",
|
||
|
"a = torch.Tensor(x.size()[0]).fill_(2.)\n",
|
||
|
"m = torch.stack([x, a])\n",
|
||
|
"plt.plot(x, nn.functional.tanh(m)[0])\n",
|
||
|
"fname = '08_Neuronowy_ngramowy_model/tanh.png'\n",
|
||
|
"plt.savefig(fname)\n",
|
||
|
"fname"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"##### Tangens hiperboliczny zastosowany dla wektora\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Tangens hiperboliczny wektora będzie po prostu wektorem tangensów\n",
|
||
|
"hiperbolicznych poszczególnych wartości.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "code",
|
||
|
"execution_count": 3,
|
||
|
"metadata": {},
|
||
|
"outputs": [
|
||
|
{
|
||
|
"name": "stderr",
|
||
|
"output_type": "stream",
|
||
|
"text": [
|
||
|
"/media/kuba/ssdsam/anaconda3/envs/lmzajecia/lib/python3.10/site-packages/torch/nn/functional.py:1933: UserWarning: nn.functional.tanh is deprecated. Use torch.tanh instead.\n",
|
||
|
" warnings.warn(\"nn.functional.tanh is deprecated. Use torch.tanh instead.\")\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"data": {
|
||
|
"text/plain": [
|
||
|
"tensor([-1.0000, -0.9640, 0.0000, 0.4621, 1.0000])"
|
||
|
]
|
||
|
},
|
||
|
"execution_count": 3,
|
||
|
"metadata": {},
|
||
|
"output_type": "execute_result"
|
||
|
}
|
||
|
],
|
||
|
"source": [
|
||
|
"import torch\n",
|
||
|
"import torch.nn as nn\n",
|
||
|
"\n",
|
||
|
"v = torch.Tensor([-100, -2.0, 0.0, 0.5, 1000.0])\n",
|
||
|
"nn.functional.tanh(v)"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"[[[tensor](tensor)([-1.0000, -0.9640, 0.0000, 0.4621, 1.0000])]]\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Wzór i schemat dwuwarstwowego n-gramowego neuronowego modelu języka\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Dwuwarstwowy model języka będzie określony następującym wzorem:\n",
|
||
|
"\n",
|
||
|
"$$\\vec{y} = \\operatorname{softmax}(C\\operatorname{tgh}(W[E(w_{i-n+1}),\\dots,E(w_{i-1})])),$$\n",
|
||
|
"\n",
|
||
|
"gdzie:\n",
|
||
|
"\n",
|
||
|
"- $W$ jest wyuczalną macierzą wag o rozmiarze $h \\times nm$,\n",
|
||
|
"- $C$ będzie macierzą o rozmiarze $|V| \\times h$.\n",
|
||
|
"\n",
|
||
|
"Zmodyfikowaną sieć można przedstawić za pomocą następującego schematu:\n",
|
||
|
"\n",
|
||
|
"![img](./08_Neuronowy_ngramowy_model/ngram-tgh.drawio.png \"Dwuwarstwowy n-gramowy neuronowy model języka\")\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Liczba wag w modelu dwuwarstwowym\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Na wagi w modelu dwuwarstwowym składają się:\n",
|
||
|
"\n",
|
||
|
"- zanurzenia: $m|V|$,\n",
|
||
|
"- wagi warstwy ukrytej: $hnm$,\n",
|
||
|
"- wagi warstwy wyjściowej: $|V|h$,\n",
|
||
|
"\n",
|
||
|
"a zatem łącznie:\n",
|
||
|
"\n",
|
||
|
"$$m|V| + hnm + |V|h$$\n",
|
||
|
"\n",
|
||
|
"Jeśli $h \\approx m$ (co jest realistyczną opcją), wówczas otrzymamy oszacowanie:\n",
|
||
|
"\n",
|
||
|
"$$O(m|V| + nm^2).$$\n",
|
||
|
"\n",
|
||
|
"Zauważmy, że względem $n$ oznacza to bardzo korzystną złożoność\n",
|
||
|
"$O(n)$! Oznacza to, że nasz model może działać dla dużo większych\n",
|
||
|
"wartości $n$ niż tradycyjny, statystyczny n-gramowy model języka (dla którego\n",
|
||
|
"wartości $n > 5$ zazwyczaj nie mają sensu).\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Model worka słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Jak stwierdziliśmy przed chwilą, dwuwarstwowy n-gramowy model języka\n",
|
||
|
"może działać dla stosunkowo dużego $n$. Zauważmy jednak, że istnieje\n",
|
||
|
"pewna słabość tego modelu. Otóż o ile intuicyjnie ma sens odróżniać\n",
|
||
|
"słowo poprzedzające, słowo występujące dwie pozycje wstecz i zapewne\n",
|
||
|
"trzy pozycje wstecz, a zatem uczyć się osobnych macierzy $C_{-1}$,\n",
|
||
|
"$C_{-2}$, $C_{-3}$ to różnica między wpływem słowa\n",
|
||
|
"występującego cztery pozycje wstecz i pięć pozycji wstecz jest już\n",
|
||
|
"raczej nieistotna; innymi słowy różnica między macierzami $C_{-4}$ i\n",
|
||
|
"$C_{-5}$ będzie raczej niewielka i sieć niepotrzebnie będzie uczyła\n",
|
||
|
"się dwukrotnie podobnych wag. Im dalej wstecz, tym różnica wpływu\n",
|
||
|
"będzie jeszcze mniej istotna, można np. przypuszczać, że różnica\n",
|
||
|
"między $C_{-10}$ i $C_{-13}$ nie powinna być duża.\n",
|
||
|
"\n",
|
||
|
"Spróbujmy najpierw zaproponować radykalne podejście, w którym nie\n",
|
||
|
"będziemy w ogóle uwzględniać pozycji słów (lub będziemy je uwzględniać\n",
|
||
|
"w niewielkim stopniu), później połączymy to z omówionym wcześniej\n",
|
||
|
"modelem $n$-gramowym.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Agregacja wektorów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Zamiast patrzeć na kilka poprzedzających słów, można przewidywać na\n",
|
||
|
"podstawie **całego** ciągu słów poprzedzających odgadywane słowo. Zauważmy jednak, że\n",
|
||
|
"sieć neuronowa musi mieć ustaloną strukturę, nie możemy zmieniać jej\n",
|
||
|
"rozmiaru. Musimy zatem najpierw zagregować cały ciąg do wektora o\n",
|
||
|
"**stałej** długości. Potrzebujemy zatem pewnej funkcji agregującej $A$, takiej by\n",
|
||
|
"$A(w_1,\\dots,w_{i-1})$ było wektorem o stałej długości, niezależnie od $i$.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Worek słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Najprostszą funkcją agregującą jest po prostu… suma. Dodajemy po\n",
|
||
|
"prostu zanurzenia słów:\n",
|
||
|
"\n",
|
||
|
"$$A(w_1,\\dots,w_{i-1}) = E(w_1) + \\dots + E(w_{i-1}) = \\sum_{j=1}^{i-1} E(w_j).$$\n",
|
||
|
"\n",
|
||
|
"**Uwaga**: zanurzenia słów nie zależą od pozycji słowa (podobnie było w wypadku n-gramowego modelu!).\n",
|
||
|
"\n",
|
||
|
"Jeśli rozmiar zanurzenia (embeddingu) wynosi $m$, wówczas rozmiar\n",
|
||
|
"wektora uzyskanego dla całego poprzedzającego tekstu wynosi również $m$.\n",
|
||
|
"\n",
|
||
|
"Proste dodawanie wydaje się bardzo „prostacką” metodą, a jednak\n",
|
||
|
"suma wektorów słów jest **zaskakująco skuteczną metodą zanurzenia\n",
|
||
|
"(embedowania) całych tekstów (doc2vec)**. Prostym wariantem dodawania jest obliczanie **średniej wektorów**:\n",
|
||
|
"\n",
|
||
|
"$$A(w_1,\\dots,w_{i-1}) = \\frac{E(w_1) + \\dots + E(w_{i-1})}{i-1} = \\frac{\\sum_{j=1}^{i-1} E(w_j)}{i-1}.$$\n",
|
||
|
"\n",
|
||
|
"Tak czy siak uzyskany wektor **nie zależy od kolejności słów**\n",
|
||
|
"(dodawanie jest przemienne i łączne!). Mówimy więc o **worku słów**\n",
|
||
|
"(*bag of words*, *BoW*) — co ma symbolizować fakt, że słowa są\n",
|
||
|
"przemieszane, niczym produkty w torbie na zakupy.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"##### Schemat graficzny modelu typu worek słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Po zanurzeniu całego poprzedzającego tekstu postępujemy podobnie jak w\n",
|
||
|
"modelu bigramowym — rzutujemy embedding na długi wektor wartości, na\n",
|
||
|
"którym stosujemy funkcję softmax:\n",
|
||
|
"\n",
|
||
|
"![img](./08_Neuronowy_ngramowy_model/bow1.drawio.png \"Model typu worek słów\")\n",
|
||
|
"\n",
|
||
|
"Odpowiada to wzorowi:\n",
|
||
|
"\n",
|
||
|
"$$y = \\operatorname{softmax}(C\\sum_{j=1}^{i-1} E(w_j)).$$\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Jak traktować powtarzające się słowa?\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Według wzoru podanego wyżej, jeśli słowo w poprzedzającym tekście\n",
|
||
|
"pojawia się więcej niż raz, jego embedding zostanie zsumowany odpowiednią liczbę razy.\n",
|
||
|
"Na przykład embedding tekstu *to be or not to be* będzie wynosił:\n",
|
||
|
"\n",
|
||
|
"$$E(\\mathrm{to}) + E(\\mathrm{be}) + E(\\mathrm{or}) + E(\\mathrm{not}) + E(\\mathrm{to}) + E(\\mathrm{be}) = 2E(\\mathrm{to}) + 2E(\\mathrm{be}) + E(\\mathrm{or}) + E(\\mathrm{not}).$$\n",
|
||
|
"\n",
|
||
|
"Innymi słowy, choć w worku słów nie uwzględniamy kolejności słów, to\n",
|
||
|
"**liczba wystąpień** ma dla nas ciągle znaczenie. Można powiedzieć, że\n",
|
||
|
"traktujemy poprzedzający tekst jako **multizbiór** (struktura\n",
|
||
|
"matematyczna, w której nie uwzględnia się kolejności, choć zachowana\n",
|
||
|
"jest informacja o liczbie wystąpień).\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"##### Zbiór słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Oczywiście moglibyśmy przy agregowaniu zanurzeń pomijać powtarzające\n",
|
||
|
"się słowa, a zatem zamiast multizbioru słów rozpatrywać po prostu ich zbiór:\n",
|
||
|
"\n",
|
||
|
"$$A(w_1,\\dots,w_{i-1}) = \\sum_{w \\in \\{w_1,\\dots,w_{i-1}\\}} E(w).$$\n",
|
||
|
"\n",
|
||
|
"Jest kwestią dyskusyjną, czy to lepsze czy gorsze podejście — w końcu\n",
|
||
|
"liczba wystąpień np. słów *Ukraina* czy *Polska* może wpływać w jakimś\n",
|
||
|
"stopniu na prawdopodobieństwo kolejnego słowa (*Kijów* czy\n",
|
||
|
"*Warszawa*?).\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Worek słów a wektoryzacja tf\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Wzór na sumę zanurzeń słów można przekształcić w taki sposób, by\n",
|
||
|
"sumować po wszystkich słowach ze słownika, zamiast po słowach rzeczywiście występujących w tekście:\n",
|
||
|
"\n",
|
||
|
"$$A(w_1,\\dots,w_{i-1}) = \\sum_{j=1}^{i-1} E(w_j) = \\sum_{w \\in V} \\#wE(w)$$\n",
|
||
|
"\n",
|
||
|
"gdzie $\\#w$ to liczba wystąpień słowa $w$ w ciagu $w_1,\\dots,w_{i-1}$ (w wielu przypadkach równa zero!).\n",
|
||
|
"\n",
|
||
|
"Jeśli teraz zanurzenia będziemy reprezentować jako macierz $E$ (por. poprzedni wykład),\n",
|
||
|
"wówczas sumę można przedstawić jako iloczyn macierzy $E$ i pewnego wektora:\n",
|
||
|
"\n",
|
||
|
"$$A(w_1,\\dots,w_{i-1}) = E(w) [\\#w^1,\\dots,\\#w^{|V|}]^T.$$\n",
|
||
|
"\n",
|
||
|
"(Odróżniamy $w^i$ jako $i$-ty wyraz w słowniku $V$ od $w_i$ jako $i$-tego wyraz w rozpatrywanym ciągu).\n",
|
||
|
"\n",
|
||
|
"Zwróćmy uwagę, że wektor $[\\#w_1,\\dots,\\#w_{|V|}]$ to po prostu\n",
|
||
|
"reprezentacja wektora poprzedzającego tekstu (tj. ciągu\n",
|
||
|
"$(w_1,\\dots,w_{i-1})$) przy użyciu schematu wektoryzacji tf (*term\n",
|
||
|
"frequency*). Przypomnijmy, że tf to reprezentacja tekstu przy użyciu\n",
|
||
|
"wektorów o rozmiarze $|V|$ — na każdej pozycji odnotowujemy liczbę wystąpień.\n",
|
||
|
"Wektory tf są **rzadkie**, tj. na wielu pozycjach zawierają zera.\n",
|
||
|
"\n",
|
||
|
"Innymi słowy, nasz model języka *bag of words* można przedstawić za pomocą wzoru:\n",
|
||
|
"\n",
|
||
|
"$$y = \\operatorname{softmax}(C\\operatorname{tf}(w_1,\\dots,w_{i-1})),$$\n",
|
||
|
"\n",
|
||
|
"co można zilustrować w następujący sposób:\n",
|
||
|
"\n",
|
||
|
"![img](./08_Neuronowy_ngramowy_model/bow2.drawio.png \"Model typu worek słów — alternatywna reprezentacja\")\n",
|
||
|
"\n",
|
||
|
"Można stwierdzić, że zanurzenie tekstu przekształca rzadki, długi wektor\n",
|
||
|
"tf w gęsty, krótki wektor.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Ważenie słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Czy wszystkie słowa są tak samo istotne? Rzecz jasna, nie:\n",
|
||
|
"\n",
|
||
|
"- jak już wiemy z naszych rozważań dotyczących n-gramowych modeli języka, słowa bezpośrednio\n",
|
||
|
" poprzedzające odgadywany wyraz mają większy wpływ niż słowa wcześniejsze;\n",
|
||
|
" intuicyjnie, wpływ słów stopniowo spada — tym bardziej, im bardziej słowo jest oddalone od słowa odgadywanego;\n",
|
||
|
"- jak wiemy z wyszukiwania informacji, słowa, które występują w wielu tekstach czy dokumentach, powinny mieć\n",
|
||
|
" mniejsze znaczenie, w skrajnym przypadku słowa występujące w prawie każdym tekście (*że*, *w*, *i* itd.) powinny\n",
|
||
|
" być praktycznie pomijane jako *stop words* (jeśli rozpatrywać je w „masie” worka słów — oczywiście\n",
|
||
|
" to, czy słowo poprzedzające odgadywane słowo to *że*, *w* czy *i* ma olbrzymie znaczenie!).\n",
|
||
|
"\n",
|
||
|
"Zamiast po prostu dodawać zanurzenia, można operować na sumie (bądź średniej) ważonej:\n",
|
||
|
"\n",
|
||
|
"$$\\sum_{j=1}^{i-1} \\omega(j, w_j)E(w_j),$$\n",
|
||
|
"\n",
|
||
|
"gdzie $\\omega(j, w_j)$ jest pewną wagą, która może zależeć od pozycji $j$ lub samego słowa $w_j$.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Uwzględnienie pozycji\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Można w pewnym stopniu złamać „workowatość” naszej sieci przez proste\n",
|
||
|
"uwzględnienie pozycji słowa, np. w taki sposób:\n",
|
||
|
"\n",
|
||
|
"$$\\omega(j, w_j) = \\beta^{i-j-1},$$\n",
|
||
|
"\n",
|
||
|
"dla pewnego hiperparametru $\\beta$. Na przykład jeśli $\\beta=0,9$,\n",
|
||
|
"wówczas słowo bezpośrednio poprzedzające dane słowo ma $1 / 0,9^9 \\approx 2,58$\n",
|
||
|
"większy wpływ niż słowo występujące 10 pozycji wstecz.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Odwrócona częstość dokumentowa\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Aby większą wagę przykładać do słów występujących w mniejszej liczbie\n",
|
||
|
"dokumentów, możemy użyć, znanej z wyszukiwania informacji,\n",
|
||
|
"odwrotnej częstości dokumentowej (*inverted document frequency*, *idf*):\n",
|
||
|
"\n",
|
||
|
"$$\\omega(j, w_j) = \\operatorname{idf}_S(w_j) = \\operatorname{log}\\frac{|S|}{\\operatorname{df}_S(w_j)},$$\n",
|
||
|
"\n",
|
||
|
"gdzie:\n",
|
||
|
"\n",
|
||
|
"- $S$ jest pewną kolekcją dokumentów czy tekstów, z którego pochodzi przedmiotowy ciąg słów,\n",
|
||
|
"- $\\operatorname{df}_S(w)$ to częstość dokumentowa słowa $w$ w kolekcji $S$, tzn. odpowiedź na pytanie,\n",
|
||
|
" w ilu dokumentach występuje $w$.\n",
|
||
|
"\n",
|
||
|
"Rzecz jasna, ten sposób ważenia oznacza tak naprawdę zastosowanie wektoryzacji tf-idf zamiast tf,\n",
|
||
|
"nasza sieć będzie dana zatem wzorem:\n",
|
||
|
"\n",
|
||
|
"$$y = \\operatorname{softmax}(C\\operatorname{tfidf}(w_1,\\dots,w_{i-1})).$$\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"#### Bardziej skomplikowane sposoby ważenia słów\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Można oczywiście połączyć odwrotną częstość dokumentową z uwzględnieniem pozycji słowa:\n",
|
||
|
"\n",
|
||
|
"$$\\omega(j, w_j) = \\beta^{i-j-1}\\operatorname{idf}_S(w_j).$$\n",
|
||
|
"\n",
|
||
|
"**Uwaga**: „wagi” $\\omega(j, w_j)$ nie są tak naprawdę wyuczalnymi\n",
|
||
|
"wagami (parametrami) naszej sieci neuronowej, terminologia może być\n",
|
||
|
"tutaj myląca. Z punktu widzenia sieci neuronowej $\\omega(j, w_j)$ są\n",
|
||
|
"stałe i **nie** są optymalizowane w procesie propagacji wstecznej. Innymi\n",
|
||
|
"słowy, tak zdefiniowane $\\omega(j, w_j)$ zależą tylko od:\n",
|
||
|
"\n",
|
||
|
"- hiperparametru $\\beta$, który może być optymalizowany już poza siecią (w procesie **hiperoptymalizacji**),\n",
|
||
|
"- wartości $\\operatorname{idf}_S(w_j)$ wyliczanych wcześniej na podstawie kolekcji $S$.\n",
|
||
|
"\n",
|
||
|
"**Pytanie**: czy wagi $\\omega(j, w_j)$ mogłyby sensownie uwzględniać\n",
|
||
|
"jakieś parametry wyuczalne z całą siecią?\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Modelowanie języka przy użyciu bardziej złożonych neuronowych sieci *feed-forward*\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Można połączyć zalety obu ogólnych podejść (n-gramowego modelu i worka\n",
|
||
|
"słów) — można **równocześnie** traktować w specjalny sposób (na\n",
|
||
|
"przykład) dwa poprzedzające wyrazy, wszystkie zaś inne wyrazy\n",
|
||
|
"reprezentować jako „tło” modelowane za pomocą worka słów lub podobnej\n",
|
||
|
"reprezentacji. Osiągamy to poprzez konkatenację wektora\n",
|
||
|
"poprzedzającego słowa, słowa występującego dwie pozycje wstecz oraz\n",
|
||
|
"zagregowanego zanurzenia całego wcześniejszego tekstu:\n",
|
||
|
"\n",
|
||
|
"$$y = \\operatorname{softmax}(C[E(w_{i-1}),E(w_{i-2}),A(w_1,\\dots,w_{i-3})]),$$\n",
|
||
|
"\n",
|
||
|
"czy lepiej z dodatkową warstwą ukrytą:\n",
|
||
|
"\n",
|
||
|
"$$y = \\operatorname{softmax}(C\\operatorname{tgh}(W[E(w_{i-1}),E(w_{i-2}),A(w_1,\\dots,w_{i-3})])),$$\n",
|
||
|
"\n",
|
||
|
"W tak uzyskanym dwuwarstwowym neuronowym modelu języka, łączącym model\n",
|
||
|
"trigramowy z workiem słów, macierz $W$ ma rozmiar $h \\times 3m$.\n",
|
||
|
"\n",
|
||
|
"**Pytanie**: jakie mamy możliwości, jeśli zamiast przewidywać kolejne słowo, mamy za zadanie\n",
|
||
|
"odgadywać słowo w luce (jak w wyzwaniach typu *word gap*)?\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"### Literatura\n",
|
||
|
"\n"
|
||
|
]
|
||
|
},
|
||
|
{
|
||
|
"cell_type": "markdown",
|
||
|
"metadata": {},
|
||
|
"source": [
|
||
|
"Skuteczny n-gramowy neuronowy model języka opisano po raz pierwszy\n",
|
||
|
"w pracy [A Neural Probabilistic Language Model](https://www.jmlr.org/papers/volume3/bengio03a/bengio03a.pdf) autorstwa Yoshua Bengio i in.\n",
|
||
|
"\n"
|
||
|
]
|
||
|
}
|
||
|
],
|
||
|
"metadata": {
|
||
|
"kernelspec": {
|
||
|
"display_name": "Python 3 (ipykernel)",
|
||
|
"language": "python",
|
||
|
"name": "python3"
|
||
|
},
|
||
|
"language_info": {
|
||
|
"codemirror_mode": {
|
||
|
"name": "ipython",
|
||
|
"version": 3
|
||
|
},
|
||
|
"file_extension": ".py",
|
||
|
"mimetype": "text/x-python",
|
||
|
"name": "python",
|
||
|
"nbconvert_exporter": "python",
|
||
|
"pygments_lexer": "ipython3",
|
||
|
"version": "3.10.4"
|
||
|
},
|
||
|
"org": null
|
||
|
},
|
||
|
"nbformat": 4,
|
||
|
"nbformat_minor": 1
|
||
|
}
|