| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Statistics::Basic::Vector; |
|
2
|
|
|
|
|
|
|
|
|
3
|
33
|
|
|
33
|
|
158
|
use strict; |
|
|
33
|
|
|
|
|
45
|
|
|
|
33
|
|
|
|
|
1315
|
|
|
4
|
33
|
|
|
33
|
|
152
|
use warnings; |
|
|
33
|
|
|
|
|
39
|
|
|
|
33
|
|
|
|
|
900
|
|
|
5
|
33
|
|
|
33
|
|
145
|
use Carp; |
|
|
33
|
|
|
|
|
50
|
|
|
|
33
|
|
|
|
|
2079
|
|
|
6
|
33
|
|
|
33
|
|
171
|
use Scalar::Util qw(blessed weaken looks_like_number); |
|
|
33
|
|
|
|
|
115
|
|
|
|
33
|
|
|
|
|
3920
|
|
|
7
|
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
our $tag_number = 0; |
|
9
|
|
|
|
|
|
|
|
|
10
|
33
|
|
|
33
|
|
204
|
use Statistics::Basic; |
|
|
33
|
|
|
|
|
70
|
|
|
|
33
|
|
|
|
|
338
|
|
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
use overload |
|
13
|
0
|
|
|
0
|
|
0
|
'0+' => sub { croak "attempt to use vector as scalar numerical value" }, |
|
14
|
|
|
|
|
|
|
'""' => sub { |
|
15
|
42
|
|
|
42
|
|
691
|
my $this = $_[0]; |
|
16
|
42
|
|
|
|
|
59
|
local $" = ", "; |
|
17
|
42
|
100
|
|
|
|
109
|
my @r = map { defined $_ ? $Statistics::Basic::fmt->format_number($_, $Statistics::Basic::IPRES) : "_" } $this->query; |
|
|
207
|
|
|
|
|
9842
|
|
|
18
|
42
|
100
|
|
|
|
4609
|
$Statistics::Basic::DEBUG ? "vector-$this->{tag}:[@r]" : "[@r]"; |
|
19
|
|
|
|
|
|
|
}, |
|
20
|
656
|
|
|
656
|
|
1464
|
'bool' => sub { 1 }, |
|
21
|
33
|
|
|
33
|
|
207
|
fallback => 1; # tries to do what it would have done if this wasn't present. |
|
|
33
|
|
|
|
|
75
|
|
|
|
33
|
|
|
|
|
678
|
|
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
# new {{{ |
|
24
|
|
|
|
|
|
|
sub new { |
|
25
|
294
|
|
|
294
|
1
|
1610
|
my $class = shift; |
|
26
|
294
|
|
|
|
|
316
|
my $vector = $_[0]; |
|
27
|
|
|
|
|
|
|
|
|
28
|
294
|
100
|
66
|
|
|
2274
|
if( blessed($vector) and $vector->isa(__PACKAGE__) ) { |
|
29
|
182
|
50
|
|
|
|
313
|
warn "vector->new called with blessed argument, returning $vector instead of making another\n" if $Statistics::Basic::DEBUG >= 3; |
|
30
|
182
|
|
|
|
|
604
|
return $vector; |
|
31
|
|
|
|
|
|
|
} |
|
32
|
|
|
|
|
|
|
|
|
33
|
112
|
|
|
|
|
621
|
my $this = bless {tag=>(++$tag_number), s=>0, c=>{}, v=>[]}, $class; |
|
34
|
112
|
|
|
|
|
335
|
$this->set_vector( @_ ); |
|
35
|
|
|
|
|
|
|
|
|
36
|
112
|
50
|
|
|
|
279
|
warn "created new vector $this\n" if $Statistics::Basic::DEBUG >= 3; |
|
37
|
|
|
|
|
|
|
|
|
38
|
112
|
|
|
|
|
689
|
return $this; |
|
39
|
|
|
|
|
|
|
} |
|
40
|
|
|
|
|
|
|
# }}} |
|
41
|
|
|
|
|
|
|
# copy {{{ |
|
42
|
|
|
|
|
|
|
sub copy { |
|
43
|
3
|
|
|
3
|
1
|
7
|
my $this = shift; |
|
44
|
3
|
|
|
|
|
5
|
my $that = __PACKAGE__->new( [@{$this->{v}}] ); |
|
|
3
|
|
|
|
|
10
|
|
|
45
|
|
|
|
|
|
|
|
|
46
|
3
|
50
|
|
|
|
9
|
warn "copied vector($this -> $that)\n" if $Statistics::Basic::DEBUG >= 3; |
|
47
|
|
|
|
|
|
|
|
|
48
|
3
|
|
|
|
|
5
|
return $that; |
|
49
|
|
|
|
|
|
|
} |
|
50
|
|
|
|
|
|
|
# }}} |
|
51
|
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
# _set_computer {{{ |
|
53
|
|
|
|
|
|
|
sub _set_computer { |
|
54
|
243
|
|
|
243
|
|
235
|
my $this = shift; |
|
55
|
|
|
|
|
|
|
|
|
56
|
243
|
|
|
|
|
786
|
while( my ($k,$v) = splice @_, 0, 2 ) { |
|
57
|
243
|
100
|
|
|
|
449
|
warn "$this set_computer($k => " . overload::StrVal($v) . ")\n" if $Statistics::Basic::DEBUG; |
|
58
|
243
|
|
|
|
|
843
|
weaken($this->{c}{$k} = $v); |
|
59
|
243
|
|
|
|
|
904
|
$v->_recalc_needed; |
|
60
|
|
|
|
|
|
|
} |
|
61
|
|
|
|
|
|
|
|
|
62
|
243
|
|
|
|
|
326
|
return; |
|
63
|
|
|
|
|
|
|
} |
|
64
|
|
|
|
|
|
|
# }}} |
|
65
|
|
|
|
|
|
|
# _set_linked_computer {{{ |
|
66
|
|
|
|
|
|
|
sub _set_linked_computer { |
|
67
|
62
|
|
|
62
|
|
56
|
my $this = shift; |
|
68
|
62
|
|
|
|
|
65
|
my $key = shift; |
|
69
|
62
|
|
|
|
|
58
|
my $var = shift; |
|
70
|
|
|
|
|
|
|
|
|
71
|
62
|
|
|
|
|
88
|
my $new_key = join("_", ($key, sort {$a<=>$b} map {$_->{tag}} @_)); |
|
|
0
|
|
|
|
|
0
|
|
|
|
62
|
|
|
|
|
196
|
|
|
72
|
|
|
|
|
|
|
|
|
73
|
62
|
|
|
|
|
106
|
$this->_set_computer( $new_key => $var ); |
|
74
|
|
|
|
|
|
|
|
|
75
|
62
|
|
|
|
|
92
|
return; |
|
76
|
|
|
|
|
|
|
} |
|
77
|
|
|
|
|
|
|
# }}} |
|
78
|
|
|
|
|
|
|
# _get_computer {{{ |
|
79
|
|
|
|
|
|
|
sub _get_computer { |
|
80
|
244
|
|
|
244
|
|
264
|
my $this = shift; |
|
81
|
244
|
|
|
|
|
243
|
my $k = shift; |
|
82
|
|
|
|
|
|
|
|
|
83
|
244
|
100
|
50
|
|
|
427
|
warn "$this get_computer($k): " . overload::StrVal($this->{c}{$k}||"") . "\n" if $Statistics::Basic::DEBUG; |
|
84
|
|
|
|
|
|
|
|
|
85
|
244
|
|
|
|
|
616
|
return $this->{c}{$k}; |
|
86
|
|
|
|
|
|
|
} |
|
87
|
|
|
|
|
|
|
# }}} |
|
88
|
|
|
|
|
|
|
# _get_linked_computer {{{ |
|
89
|
|
|
|
|
|
|
sub _get_linked_computer { |
|
90
|
32
|
|
|
32
|
|
37
|
my $this = shift; |
|
91
|
32
|
|
|
|
|
40
|
my $key = shift; |
|
92
|
|
|
|
|
|
|
|
|
93
|
32
|
|
|
|
|
62
|
my $new_key = join("_", ($key, sort {$a<=>$b} map {$_->{tag}} @_)); |
|
|
0
|
|
|
|
|
0
|
|
|
|
32
|
|
|
|
|
141
|
|
|
94
|
|
|
|
|
|
|
|
|
95
|
32
|
|
|
|
|
81
|
return $this->_get_computer( $new_key ); |
|
96
|
|
|
|
|
|
|
} |
|
97
|
|
|
|
|
|
|
# }}} |
|
98
|
|
|
|
|
|
|
# _inform_computers_of_change {{{ |
|
99
|
|
|
|
|
|
|
sub _inform_computers_of_change { |
|
100
|
175
|
|
|
175
|
|
200
|
my $this = shift; |
|
101
|
|
|
|
|
|
|
|
|
102
|
175
|
|
|
|
|
173
|
for my $k (keys %{ $this->{c} }) { |
|
|
175
|
|
|
|
|
537
|
|
|
103
|
271
|
|
|
|
|
323
|
my $v = $this->{c}{$k}; |
|
104
|
|
|
|
|
|
|
|
|
105
|
271
|
100
|
66
|
|
|
1154
|
if( defined($v) and blessed($v) ) { |
|
106
|
268
|
|
|
|
|
616
|
$v->_recalc_needed; |
|
107
|
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
} else { |
|
109
|
3
|
|
|
|
|
5
|
delete $this->{c}{$k}; |
|
110
|
|
|
|
|
|
|
} |
|
111
|
|
|
|
|
|
|
} |
|
112
|
|
|
|
|
|
|
|
|
113
|
175
|
|
|
|
|
298
|
return; |
|
114
|
|
|
|
|
|
|
} |
|
115
|
|
|
|
|
|
|
# }}} |
|
116
|
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
# _fix_size {{{ |
|
118
|
|
|
|
|
|
|
sub _fix_size { |
|
119
|
42
|
|
|
42
|
|
56
|
my $this = shift; |
|
120
|
|
|
|
|
|
|
|
|
121
|
42
|
|
|
|
|
58
|
my $fixed = 0; |
|
122
|
|
|
|
|
|
|
|
|
123
|
42
|
|
|
|
|
49
|
my $d = @{$this->{v}} - $this->{s}; |
|
|
42
|
|
|
|
|
111
|
|
|
124
|
42
|
100
|
|
|
|
113
|
if( $d > 0 ) { |
|
125
|
25
|
|
|
|
|
44
|
splice @{$this->{v}}, 0, $d; |
|
|
25
|
|
|
|
|
56
|
|
|
126
|
25
|
|
|
|
|
206
|
$fixed = 1; |
|
127
|
|
|
|
|
|
|
} |
|
128
|
|
|
|
|
|
|
|
|
129
|
42
|
100
|
|
|
|
134
|
unless( $Statistics::Basic::NOFILL ) { |
|
130
|
34
|
100
|
|
|
|
88
|
if( $d < 0 ) { |
|
131
|
8
|
|
|
|
|
30
|
unshift @{$this->{v}}, # unshift so the 0s leave first |
|
|
20
|
|
|
|
|
34
|
|
|
132
|
8
|
|
|
|
|
13
|
map {0} $d .. -1; # add $d of them |
|
133
|
|
|
|
|
|
|
|
|
134
|
8
|
|
|
|
|
18
|
$fixed = 1; |
|
135
|
|
|
|
|
|
|
} |
|
136
|
|
|
|
|
|
|
} |
|
137
|
|
|
|
|
|
|
|
|
138
|
42
|
50
|
|
|
|
192
|
warn "[fix_size $this] [@{ $this->{v} }]\n" if $Statistics::Basic::DEBUG >= 2; |
|
|
0
|
|
|
|
|
0
|
|
|
139
|
|
|
|
|
|
|
|
|
140
|
42
|
|
|
|
|
69
|
return $fixed; |
|
141
|
|
|
|
|
|
|
} |
|
142
|
|
|
|
|
|
|
# }}} |
|
143
|
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
# query {{{ |
|
145
|
|
|
|
|
|
|
sub query { |
|
146
|
328
|
|
|
328
|
1
|
350
|
my $this = shift; |
|
147
|
|
|
|
|
|
|
|
|
148
|
328
|
100
|
|
|
|
638
|
return (wantarray ? @{$this->{v}} : $this->{v}); |
|
|
262
|
|
|
|
|
1455
|
|
|
149
|
|
|
|
|
|
|
} |
|
150
|
|
|
|
|
|
|
# }}} |
|
151
|
|
|
|
|
|
|
# query_filled {{{ |
|
152
|
|
|
|
|
|
|
sub query_filled { |
|
153
|
185
|
|
|
185
|
1
|
184
|
my $this = shift; |
|
154
|
|
|
|
|
|
|
|
|
155
|
185
|
50
|
|
|
|
346
|
warn "[query_filled $this $this->{s}]\n" if $Statistics::Basic::DEBUG >= 1; |
|
156
|
|
|
|
|
|
|
|
|
157
|
185
|
100
|
|
|
|
164
|
return if @{$this->{v}} < $this->{s}; |
|
|
185
|
|
|
|
|
478
|
|
|
158
|
181
|
|
|
|
|
482
|
return 1; |
|
159
|
|
|
|
|
|
|
} |
|
160
|
|
|
|
|
|
|
# }}} |
|
161
|
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
# insert {{{ |
|
163
|
|
|
|
|
|
|
sub insert { |
|
164
|
29
|
|
|
29
|
1
|
389
|
my $this = shift; |
|
165
|
|
|
|
|
|
|
|
|
166
|
29
|
50
|
|
|
|
96
|
croak "you must define a vector size before using insert()" unless defined $this->{s}; |
|
167
|
|
|
|
|
|
|
|
|
168
|
29
|
|
|
|
|
63
|
for my $e (@_) { |
|
169
|
31
|
100
|
100
|
|
|
142
|
if( ref($e) and not blessed($e) ) { |
|
170
|
3
|
50
|
|
|
|
6
|
if( ref($e) eq "ARRAY" ) { |
|
171
|
3
|
|
|
|
|
5
|
push @{ $this->{v} }, @$e; |
|
|
3
|
|
|
|
|
6
|
|
|
172
|
3
|
50
|
|
|
|
9
|
warn "[insert $this] @$e\n" if $Statistics::Basic::DEBUG >= 1; |
|
173
|
|
|
|
|
|
|
|
|
174
|
|
|
|
|
|
|
} else { |
|
175
|
0
|
|
|
|
|
0
|
croak "insert() elements do not make sense"; |
|
176
|
|
|
|
|
|
|
} |
|
177
|
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
} else { |
|
179
|
28
|
|
|
|
|
34
|
push @{ $this->{v} }, $e; |
|
|
28
|
|
|
|
|
92
|
|
|
180
|
28
|
50
|
|
|
|
112
|
warn "[insert $this] $e\n" if $Statistics::Basic::DEBUG >= 1; |
|
181
|
|
|
|
|
|
|
} |
|
182
|
|
|
|
|
|
|
} |
|
183
|
|
|
|
|
|
|
|
|
184
|
29
|
|
|
|
|
96
|
$this->_fix_size; |
|
185
|
29
|
|
|
|
|
70
|
$this->_inform_computers_of_change; |
|
186
|
|
|
|
|
|
|
|
|
187
|
29
|
|
|
|
|
64
|
return $this; |
|
188
|
|
|
|
|
|
|
} |
|
189
|
|
|
|
|
|
|
# }}} |
|
190
|
|
|
|
|
|
|
# ginsert {{{ |
|
191
|
|
|
|
|
|
|
sub ginsert { |
|
192
|
39
|
|
|
39
|
1
|
57
|
my $this = shift; |
|
193
|
|
|
|
|
|
|
|
|
194
|
39
|
|
|
|
|
65
|
for my $e (@_) { |
|
195
|
39
|
100
|
66
|
|
|
118
|
if( ref($e) and not blessed($e)) { |
|
196
|
2
|
50
|
|
|
|
5
|
if( ref($e) eq "ARRAY" ) { |
|
197
|
2
|
|
|
|
|
1
|
push @{ $this->{v} }, @$e; |
|
|
2
|
|
|
|
|
4
|
|
|
198
|
2
|
50
|
|
|
|
11
|
warn "[ginsert $this] @$e\n" if $Statistics::Basic::DEBUG >= 1; |
|
199
|
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
} else { |
|
201
|
0
|
|
|
|
|
0
|
croak "insert() elements do not make sense"; |
|
202
|
|
|
|
|
|
|
} |
|
203
|
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
} else { |
|
205
|
37
|
|
|
|
|
36
|
push @{ $this->{v} }, $e; |
|
|
37
|
|
|
|
|
75
|
|
|
206
|
37
|
50
|
|
|
|
104
|
warn "[ginsert $this] $e\n" if $Statistics::Basic::DEBUG >= 1; |
|
207
|
|
|
|
|
|
|
} |
|
208
|
|
|
|
|
|
|
} |
|
209
|
|
|
|
|
|
|
|
|
210
|
39
|
50
|
|
|
|
65
|
$this->{s} = @{$this->{v}} if @{$this->{v}} > $this->{s}; |
|
|
39
|
|
|
|
|
65
|
|
|
|
39
|
|
|
|
|
113
|
|
|
211
|
39
|
|
|
|
|
66
|
$this->_inform_computers_of_change; |
|
212
|
|
|
|
|
|
|
|
|
213
|
39
|
|
|
|
|
77
|
return $this; |
|
214
|
|
|
|
|
|
|
} |
|
215
|
|
|
|
|
|
|
*append = \&ginsert; |
|
216
|
|
|
|
|
|
|
# }}} |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
# query_size {{{ |
|
219
|
|
|
|
|
|
|
sub query_size { |
|
220
|
322
|
|
|
322
|
1
|
888
|
my $this = shift; |
|
221
|
|
|
|
|
|
|
|
|
222
|
322
|
|
|
|
|
314
|
return scalar @{$this->{v}}; |
|
|
322
|
|
|
|
|
758
|
|
|
223
|
|
|
|
|
|
|
} |
|
224
|
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
# maybe deprecate this later |
|
226
|
|
|
|
|
|
|
*size = \&query_size unless $ENV{TEST_AUTHOR}; |
|
227
|
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
# }}} |
|
229
|
|
|
|
|
|
|
# set_size {{{ |
|
230
|
|
|
|
|
|
|
sub set_size { |
|
231
|
13
|
|
|
13
|
1
|
22
|
my $this = shift; |
|
232
|
13
|
|
|
|
|
17
|
my $size = shift; |
|
233
|
|
|
|
|
|
|
|
|
234
|
13
|
50
|
|
|
|
59
|
croak "invalid vector size ($size)" if $size < 0; |
|
235
|
|
|
|
|
|
|
|
|
236
|
13
|
50
|
|
|
|
136
|
if( $this->{s} != $size ) { |
|
237
|
13
|
|
|
|
|
29
|
$this->{s} = $size; |
|
238
|
13
|
|
|
|
|
42
|
$this->_fix_size; |
|
239
|
13
|
|
|
|
|
35
|
$this->_inform_computers_of_change; |
|
240
|
|
|
|
|
|
|
} |
|
241
|
|
|
|
|
|
|
|
|
242
|
13
|
|
|
|
|
42
|
return $this; |
|
243
|
|
|
|
|
|
|
} |
|
244
|
|
|
|
|
|
|
# }}} |
|
245
|
|
|
|
|
|
|
# set_vector {{{ |
|
246
|
|
|
|
|
|
|
sub set_vector { |
|
247
|
160
|
|
|
160
|
1
|
1082
|
my $this = shift; |
|
248
|
160
|
|
|
|
|
195
|
my $vector = $_[0]; |
|
249
|
|
|
|
|
|
|
|
|
250
|
160
|
100
|
|
|
|
904
|
if( ref($vector) eq "ARRAY" ) { |
|
|
|
50
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
251
|
76
|
|
|
|
|
111
|
@{$this->{v}} = @$vector; |
|
|
76
|
|
|
|
|
658
|
|
|
252
|
76
|
|
|
|
|
168
|
$this->{s} = int @$vector; |
|
253
|
76
|
|
|
|
|
170
|
$this->_inform_computers_of_change; |
|
254
|
|
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
} elsif( UNIVERSAL::isa($vector, "Statistics::Basic::ComputedVector") ) { |
|
256
|
0
|
|
|
|
|
0
|
$this->set_vector($vector->{input_vector}); |
|
257
|
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
} elsif( UNIVERSAL::isa($vector, "Statistics::Basic::Vector") ) { |
|
259
|
3
|
|
|
|
|
10
|
$this->{s} = $vector->{s}; |
|
260
|
3
|
|
|
|
|
5
|
@{$this->{v}} = @{$vector->{v}}; # copy the vector |
|
|
3
|
|
|
|
|
17
|
|
|
|
3
|
|
|
|
|
9
|
|
|
261
|
|
|
|
|
|
|
|
|
262
|
|
|
|
|
|
|
# I don't think this is the behavior that we really want, since they |
|
263
|
|
|
|
|
|
|
# stay separate objects, they shouldn't be linked like this. |
|
264
|
|
|
|
|
|
|
# $this->{s} = $vector->{s}; |
|
265
|
|
|
|
|
|
|
# $this->{v} = $vector->{v}; # this links the vectors together |
|
266
|
|
|
|
|
|
|
# $this->{c} = $vector->{c}; # so we should link their computers too |
|
267
|
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
} elsif( @_ ) { |
|
269
|
37
|
|
|
|
|
78
|
@{$this->{v}} = @_; |
|
|
37
|
|
|
|
|
640
|
|
|
270
|
37
|
|
|
|
|
90
|
$this->{s} = int @_; |
|
271
|
|
|
|
|
|
|
|
|
272
|
|
|
|
|
|
|
} elsif( defined $vector ) { |
|
273
|
0
|
|
|
|
|
0
|
croak "argument to set_vector() too strange"; |
|
274
|
|
|
|
|
|
|
} |
|
275
|
|
|
|
|
|
|
|
|
276
|
160
|
50
|
33
|
|
|
471
|
warn "[set_vector $this] [@{ $this->{v} }]\n" if $Statistics::Basic::DEBUG >= 2 and ref($this->{v}); |
|
|
0
|
|
|
|
|
0
|
|
|
277
|
|
|
|
|
|
|
|
|
278
|
160
|
|
|
|
|
256
|
return $this; |
|
279
|
|
|
|
|
|
|
} |
|
280
|
|
|
|
|
|
|
# }}} |
|
281
|
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
1; |