-
Notifications
You must be signed in to change notification settings - Fork 0
/
ud.pm
130 lines (116 loc) · 4.04 KB
/
ud.pm
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
#!/usr/bin/perl
# Nízkoúrovňové funkce pro ukládání událostí s alternativami.
# (c) 2007 Dan Zeman <[email protected]>
# Licence: GNU GPL
use utf8;
use open ":utf8";
binmode(STDIN, ":utf8");
binmode(STDOUT, ":utf8");
binmode(STDERR, ":utf8");
package ud;
#------------------------------------------------------------------------------
# Převede událost s alternativami na pole událostí bez alternativ. Pole
# obsahuje všechny kombinace alternativ. V původní události mají zvláštný
# význam znaky " " (mezera - odděluje části události, které mohou mít
# alternativy) a "|" (svislítko - odděluje alternativy uvnitř části).
#------------------------------------------------------------------------------
sub alt
{
my $ud = shift; # událost s alternativami
my @alt; # seznam alternativních událostí
# Jestliže je zpracování alternativ v konfiguraci vypnuté, pouze vrátit událost.
unless($main::konfig{alternativy})
{
push(@alt, $ud);
return \@alt;
}
# Rozdělit alternativy do samostatných událostí.
# Rozdělit událost na části, které mohou být každá zvlášť rozdělené na alternativy.
# Části jsou oddělené mezerami. Pozor, tentokrát nemůžeme za oddělovač považovat
# posloupnost mezer nebo tabulátor, protože po opětovném slepení by nám vyšla jiná
# událost, která by se v hashi nenašla.
my @casti = split(/ /, $ud);
my @dilky;
for(my $i = 0; $i<=$#casti; $i++)
{
# Rozdělit část na alternativy.
my @altcasti = split(/\|/, $casti[$i]);
# Nechceme prázdné pole. I prádzná část má jednu prázdnou alternativu.
if(!scalar(@altcasti))
{
$altcasti[0] = "";
}
for(my $j = 0; $j<=$#altcasti; $j++)
{
$dilky[$i][$j] = $altcasti[$j];
}
}
# Sestavit z dílků všechny kombinace.
my @cesty = (""); # Pole má jeden prvek, a tím je prázdná cesta.
for(my $i = 0; $i<=$#dilky; $i++)
{
my @nove_cesty;
for(my $j = 0; $j<=$#{$dilky[$i]}; $j++)
{
for(my $k = 0; $k<=$#cesty; $k++)
{
my @kopie_cesty = @{$cesty[$k]};
push(@kopie_cesty, $dilky[$i][$j]);
push(@nove_cesty, \@kopie_cesty);
}
}
@cesty = @nove_cesty;
}
# Poslepovat cesty do alternativních událostí.
foreach my $cesta (@cesty)
{
push(@alt, join(" ", @{$cesta}));
}
return \@alt;
}
#------------------------------------------------------------------------------
# Zjistí četnost události.
#------------------------------------------------------------------------------
sub zjistit
{
my $ud = shift; # událost, jejíž četnost chceme znát
my $statref = shift; # odkaz na hash, v němž se má hledat
# Jestliže volající nedodal statistický model, použít globální proměnnou.
if(!$statref)
{
$statref = \%main::stat;
}
# Rozdělit událost na alternativy.
my $alts = alt($ud);
# Sečíst výskyty jednotlivých dílčích událostí.
my $n;
foreach my $alt (@{$alts})
{
$n += $statref->{$alt};
}
return $n;
}
#------------------------------------------------------------------------------
# Uloží výskyt události.
#------------------------------------------------------------------------------
sub ulozit
{
my $ud = shift; # událost, jejíž četnost chceme zvýšit
my $n = shift; # počet výskytů, o který chceme zvýšit četnost
my $statref = shift; # odkaz na hash, do nějž se četnosti ukládají
# Jestliže volající nedodal statistický model, použít globální proměnnou.
if(!$statref)
{
$statref = \%main::stat;
}
$n = 1 if($n eq "");
# Rozdělit událost na alternativy.
my $alts = alt($ud);
# Každé dílčí události započítat poměrnou část výskytu.
my $dil = $n/scalar(@{$alts});
foreach my $alt (@{$alts})
{
$statref->{$alt} += $dil;
}
}
1;