enumerable-statistics 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +11 -0
- data/.rspec +3 -0
- data/.travis.yml +19 -0
- data/Gemfile +4 -0
- data/README.md +36 -0
- data/Rakefile +9 -0
- data/bin/console +14 -0
- data/bin/rspec +16 -0
- data/bin/setup +8 -0
- data/enumerable-statistics.gemspec +27 -0
- data/ext/enumerable/statistics/extension/extconf.rb +5 -0
- data/ext/enumerable/statistics/extension/statistics.c +765 -0
- data/lib/enumerable/statistics.rb +2 -0
- data/lib/enumerable/statistics/version.rb +5 -0
- metadata +129 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 9bbba060f2298fae081d18782b237d693663558c
|
4
|
+
data.tar.gz: f4b5aa2e3bdaed71938a0bb4fb699f9d04daf3e5
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 6b2cc2848db0d8103e09bf770d565be0fcfe360f6dbb45b18ebbbca48027d0f96ae6488a90f3f0ddf3111a82200e14e3b07509f3f7e6cd5ce2fa115350f34224
|
7
|
+
data.tar.gz: 2fdc7b8b992a2194aabd6aa82af286d93d03772f385ed1927f2c412eaf2667dd9c9472a65b438c9a1a9660012a014bd879ec16e5dc934ac76d9a543a936539cd
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.travis.yml
ADDED
@@ -0,0 +1,19 @@
|
|
1
|
+
---
|
2
|
+
language: ruby
|
3
|
+
|
4
|
+
rvm:
|
5
|
+
- ruby-head
|
6
|
+
- 2.3.0
|
7
|
+
- 2.2.4
|
8
|
+
- 2.1
|
9
|
+
|
10
|
+
before_install:
|
11
|
+
- gem update --system
|
12
|
+
- gem update bundler
|
13
|
+
|
14
|
+
install:
|
15
|
+
- bundle install
|
16
|
+
|
17
|
+
script:
|
18
|
+
- bundle exec rake clobber compile
|
19
|
+
- bundle exec rake spec
|
data/Gemfile
ADDED
data/README.md
ADDED
@@ -0,0 +1,36 @@
|
|
1
|
+
# Enumerable::Statistics
|
2
|
+
|
3
|
+
Welcome to your new gem! In this directory, you'll find the files you need to be able to package up your Ruby library into a gem. Put your Ruby code in the file `lib/enumerable/statistics`. To experiment with that code, run `bin/console` for an interactive prompt.
|
4
|
+
|
5
|
+
TODO: Delete this and the text above, and describe your gem
|
6
|
+
|
7
|
+
## Installation
|
8
|
+
|
9
|
+
Add this line to your application's Gemfile:
|
10
|
+
|
11
|
+
```ruby
|
12
|
+
gem 'enumerable-statistics'
|
13
|
+
```
|
14
|
+
|
15
|
+
And then execute:
|
16
|
+
|
17
|
+
$ bundle
|
18
|
+
|
19
|
+
Or install it yourself as:
|
20
|
+
|
21
|
+
$ gem install enumerable-statistics
|
22
|
+
|
23
|
+
## Usage
|
24
|
+
|
25
|
+
TODO: Write usage instructions here
|
26
|
+
|
27
|
+
## Development
|
28
|
+
|
29
|
+
After checking out the repo, run `bin/setup` to install dependencies. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
|
30
|
+
|
31
|
+
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
32
|
+
|
33
|
+
## Contributing
|
34
|
+
|
35
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/enumerable-statistics.
|
36
|
+
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
|
3
|
+
require "bundler/setup"
|
4
|
+
require "enumerable/statistics"
|
5
|
+
|
6
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
7
|
+
# with your gem easier. You can also use a different console, if you like.
|
8
|
+
|
9
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
10
|
+
# require "pry"
|
11
|
+
# Pry.start
|
12
|
+
|
13
|
+
require "irb"
|
14
|
+
IRB.start
|
data/bin/rspec
ADDED
@@ -0,0 +1,16 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
#
|
3
|
+
# This file was generated by Bundler.
|
4
|
+
#
|
5
|
+
# The application 'rspec' is installed as part of a gem, and
|
6
|
+
# this file is here to facilitate running it.
|
7
|
+
#
|
8
|
+
|
9
|
+
require "pathname"
|
10
|
+
ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile",
|
11
|
+
Pathname.new(__FILE__).realpath)
|
12
|
+
|
13
|
+
require "rubygems"
|
14
|
+
require "bundler/setup"
|
15
|
+
|
16
|
+
load Gem.bin_path("rspec-core", "rspec")
|
data/bin/setup
ADDED
@@ -0,0 +1,27 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'enumerable/statistics/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = "enumerable-statistics"
|
8
|
+
spec.version = Enumerable::Statistics::VERSION
|
9
|
+
spec.authors = ["Kenta Murata"]
|
10
|
+
spec.email = ["mrkn@mrkn.jp"]
|
11
|
+
|
12
|
+
spec.summary = %q{Statistics features for Enumerable}
|
13
|
+
spec.description = %q{This library provides statistics features for Enumerable}
|
14
|
+
spec.homepage = "https://github.com/mrkn/enumerable-statistics"
|
15
|
+
|
16
|
+
spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
|
17
|
+
spec.bindir = "exe"
|
18
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
19
|
+
spec.require_paths = ["ext", "lib"]
|
20
|
+
spec.extensions = Dir['ext/**/extconf.rb']
|
21
|
+
|
22
|
+
spec.add_development_dependency "bundler", "~> 1.11"
|
23
|
+
spec.add_development_dependency "rake", "~> 10.0"
|
24
|
+
spec.add_development_dependency "rake-compiler", "~> 0.9.8"
|
25
|
+
spec.add_development_dependency "rspec", "~> 3.4"
|
26
|
+
spec.add_development_dependency "fuubar"
|
27
|
+
end
|
@@ -0,0 +1,765 @@
|
|
1
|
+
#include <ruby/ruby.h>
|
2
|
+
#include <ruby/version.h>
|
3
|
+
#include <assert.h>
|
4
|
+
|
5
|
+
#if RUBY_API_VERSION_CODE >= 20400
|
6
|
+
/* for 2.4.0 or higher */
|
7
|
+
# define HAVE_ARRAY_SUM
|
8
|
+
# undef HAVE_RB_FIX_PLUS
|
9
|
+
# undef HAVE_RB_RATIONAL_PLUS
|
10
|
+
#elif RUBY_API_VERSION_CODE >= 20200
|
11
|
+
/* for 2.3.0 and 2.2.0 */
|
12
|
+
# undef HAVE_ARRAY_SUM
|
13
|
+
# undef HAVE_RB_FIX_PLUS
|
14
|
+
# undef HAVE_RB_RATIONAL_PLUS
|
15
|
+
#endif
|
16
|
+
|
17
|
+
#ifndef HAVE_TYPE_STRUCT_RRATIONAL
|
18
|
+
struct RRational {
|
19
|
+
struct RBasic basic;
|
20
|
+
const VALUE num;
|
21
|
+
const VALUE den;
|
22
|
+
};
|
23
|
+
#endif
|
24
|
+
|
25
|
+
#ifndef RRATIONAL
|
26
|
+
# define RRATIONAL(obj) (R_CAST(RRational)(obj))
|
27
|
+
#endif
|
28
|
+
|
29
|
+
#ifndef RRATIONAL_SET_NUM
|
30
|
+
# define RRATIONAL_SET_NUM(rat, n) RB_OBJ_WRITE((rat), &((struct RRational *)(rat))->num,(n))
|
31
|
+
#endif
|
32
|
+
|
33
|
+
#ifndef RRATIONAL_SET_DEN
|
34
|
+
# define RRATIONAL_SET_DEN(rat, d) RB_OBJ_WRITE((rat), &((struct RRational *)(rat))->den,(d))
|
35
|
+
#endif
|
36
|
+
|
37
|
+
#ifndef HAVE_TYPE_STRUCT_RCOMPLEX
|
38
|
+
struct RComplex {
|
39
|
+
struct RBasic basic;
|
40
|
+
const VALUE real;
|
41
|
+
const VALUE imag;
|
42
|
+
};
|
43
|
+
#endif
|
44
|
+
|
45
|
+
#ifndef RCOMPLEX
|
46
|
+
# define RCOMPLEX(obj) (R_CAST(RComplex)(obj))
|
47
|
+
#endif
|
48
|
+
|
49
|
+
#ifndef RCOMPLEX_SET_REAL
|
50
|
+
# define RCOMPLEX_SET_REAL(cmp, r) RB_OBJ_WRITE((cmp), &((struct RComplex *)(cmp))->real,(r))
|
51
|
+
#endif
|
52
|
+
|
53
|
+
#ifndef RCOMPLEX_SET_IMAG
|
54
|
+
# define RCOMPLEX_SET_IMAG(cmp, i) RB_OBJ_WRITE((cmp), &((struct RComplex *)(cmp))->imag,(i))
|
55
|
+
#endif
|
56
|
+
|
57
|
+
#ifndef MUL_OVERFLOW_SIGNED_INTEGER_P
|
58
|
+
# define MUL_OVERFLOW_SIGNED_INTEGER_P(a, b, min, max) ( \
|
59
|
+
(a) == 0 ? 0 : \
|
60
|
+
(a) == -1 ? (b) < -(max) : \
|
61
|
+
(a) > 0 ? \
|
62
|
+
((b) > 0 ? (max) / (a) < (b) : (min) / (a) > (b)) : \
|
63
|
+
((b) > 0 ? (min) / (a) < (b) : (max) / (a) > (b)))
|
64
|
+
#endif
|
65
|
+
|
66
|
+
#ifndef MUL_OVERFLOW_LONG_P
|
67
|
+
# define MUL_OVERFLOW_LONG_P(a, b) MUL_OVERFLOW_SIGNED_INTEGER_P(a, b, LONG_MIN, LONG_MAX)
|
68
|
+
#endif
|
69
|
+
|
70
|
+
static ID idPow, idPLUS, idMINUS, idSTAR, id_eqeq_p, id_idiv, id_negate, id_to_f, id_cmp;
|
71
|
+
static ID id_each, id_real_p;
|
72
|
+
|
73
|
+
inline static VALUE
|
74
|
+
f_add(VALUE x, VALUE y)
|
75
|
+
{
|
76
|
+
if (FIXNUM_P(y) && FIX2LONG(y) == 0)
|
77
|
+
return x;
|
78
|
+
else if (FIXNUM_P(x) && FIX2LONG(x) == 0)
|
79
|
+
return y;
|
80
|
+
return rb_funcall(x, idPLUS, 1, y);
|
81
|
+
}
|
82
|
+
|
83
|
+
inline static VALUE
|
84
|
+
f_sub(VALUE x, VALUE y)
|
85
|
+
{
|
86
|
+
if (FIXNUM_P(y) && FIX2LONG(y) == 0)
|
87
|
+
return x;
|
88
|
+
return rb_funcall(x, idMINUS, 1, y);
|
89
|
+
}
|
90
|
+
|
91
|
+
inline static VALUE
|
92
|
+
f_real_p(VALUE x)
|
93
|
+
{
|
94
|
+
if (FIXNUM_P(x) || RB_TYPE_P(x, T_BIGNUM) || RB_TYPE_P(x, T_RATIONAL) || RB_FLOAT_TYPE_P(x))
|
95
|
+
return Qtrue;
|
96
|
+
else if (RB_TYPE_P(x, T_COMPLEX))
|
97
|
+
return Qfalse;
|
98
|
+
else
|
99
|
+
return rb_funcall(x, id_real_p, 0);
|
100
|
+
}
|
101
|
+
|
102
|
+
#ifndef HAVE_RB_FIX_PLUS
|
103
|
+
static VALUE
|
104
|
+
complex_new(VALUE klass, VALUE real, VALUE imag)
|
105
|
+
{
|
106
|
+
assert(!RB_TYPE_P(real, T_COMPLEX));
|
107
|
+
|
108
|
+
NEWOBJ_OF(obj, struct RComplex, klass, T_COMPLEX | (RGENGC_WB_PROTECTED_COMPLEX ? FL_WB_PROTECTED : 0));
|
109
|
+
|
110
|
+
RCOMPLEX_SET_REAL(obj, real);
|
111
|
+
RCOMPLEX_SET_IMAG(obj, imag);
|
112
|
+
|
113
|
+
return (VALUE)obj;
|
114
|
+
}
|
115
|
+
|
116
|
+
static VALUE
|
117
|
+
complex_caonicalize_new(VALUE klass, VALUE real, VALUE imag)
|
118
|
+
{
|
119
|
+
if (f_real_p(real) && f_real_p(imag))
|
120
|
+
return complex_new(klass, real, imag);
|
121
|
+
else if (f_real_p(imag)) {
|
122
|
+
VALUE new_imag;
|
123
|
+
|
124
|
+
new_imag = f_add(RCOMPLEX(real)->imag, imag);
|
125
|
+
|
126
|
+
return complex_new(klass, RCOMPLEX(real)->real, new_imag);
|
127
|
+
}
|
128
|
+
else {
|
129
|
+
VALUE new_real, new_imag;
|
130
|
+
|
131
|
+
new_real = f_sub(RCOMPLEX(real)->real, RCOMPLEX(imag)->imag);
|
132
|
+
new_imag = f_add(RCOMPLEX(real)->imag, RCOMPLEX(imag)->real);
|
133
|
+
|
134
|
+
return complex_new(klass, new_real, new_imag);
|
135
|
+
}
|
136
|
+
}
|
137
|
+
|
138
|
+
static VALUE
|
139
|
+
complex_add(VALUE self, VALUE other)
|
140
|
+
{
|
141
|
+
if (RB_TYPE_P(other, T_COMPLEX)) {
|
142
|
+
VALUE real, imag;
|
143
|
+
|
144
|
+
real = f_add(RCOMPLEX(self)->real, RCOMPLEX(other)->real);
|
145
|
+
imag = f_add(RCOMPLEX(self)->imag, RCOMPLEX(other)->imag);
|
146
|
+
|
147
|
+
return complex_new(CLASS_OF(self), real, imag);
|
148
|
+
}
|
149
|
+
else if (rb_obj_is_kind_of(other, rb_cNumeric) && RTEST(f_real_p(other))) {
|
150
|
+
VALUE real;
|
151
|
+
|
152
|
+
real = f_add(RCOMPLEX(self)->real, other);
|
153
|
+
|
154
|
+
return complex_new(CLASS_OF(self), real, RCOMPLEX(other)->imag);
|
155
|
+
}
|
156
|
+
return rb_num_coerce_bin(self, other, idPLUS);
|
157
|
+
}
|
158
|
+
|
159
|
+
static VALUE
|
160
|
+
rb_fix_plus(VALUE x, VALUE y)
|
161
|
+
{
|
162
|
+
if (FIXNUM_P(y)) {
|
163
|
+
long a, b, c;
|
164
|
+
VALUE r;
|
165
|
+
|
166
|
+
a = FIX2LONG(x);
|
167
|
+
b = FIX2LONG(y);
|
168
|
+
c = a + b;
|
169
|
+
r = LONG2NUM(c);
|
170
|
+
|
171
|
+
return r;
|
172
|
+
}
|
173
|
+
else if (RB_TYPE_P(y, T_BIGNUM)) {
|
174
|
+
return rb_big_plus(y, x);
|
175
|
+
}
|
176
|
+
else if (RB_FLOAT_TYPE_P(y)) {
|
177
|
+
return DBL2NUM((double)FIX2LONG(x) + RFLOAT_VALUE(y));
|
178
|
+
}
|
179
|
+
else if (RB_TYPE_P(y, T_COMPLEX)) {
|
180
|
+
return complex_add(y, x);
|
181
|
+
}
|
182
|
+
else {
|
183
|
+
return rb_num_coerce_bin(x, y, '+');
|
184
|
+
}
|
185
|
+
}
|
186
|
+
#endif
|
187
|
+
|
188
|
+
#ifndef HAVE_RB_RATIONAL_PLUS
|
189
|
+
# define ZERO INT2FIX(0)
|
190
|
+
# define ONE INT2FIX(1)
|
191
|
+
|
192
|
+
# define f_boolcast(x) ((x) ? Qtrue : Qfalse)
|
193
|
+
#define rb_raise_zerodiv() rb_raise(rb_eZeroDivError, "divided by 0")
|
194
|
+
|
195
|
+
inline static VALUE
|
196
|
+
f_cmp(VALUE x, VALUE y)
|
197
|
+
{
|
198
|
+
if (FIXNUM_P(x) && FIXNUM_P(y)) {
|
199
|
+
long c = FIX2LONG(x) - FIX2LONG(y);
|
200
|
+
if (c > 0)
|
201
|
+
c = 1;
|
202
|
+
else if (c < 0)
|
203
|
+
c = -1;
|
204
|
+
return INT2FIX(c);
|
205
|
+
}
|
206
|
+
return rb_funcall(x, id_cmp, 1, y);
|
207
|
+
}
|
208
|
+
|
209
|
+
inline static VALUE
|
210
|
+
f_negative_p(VALUE x)
|
211
|
+
{
|
212
|
+
if (FIXNUM_P(x))
|
213
|
+
return f_boolcast(FIX2LONG(x) < 0);
|
214
|
+
return rb_funcall(x, '<', 1, ZERO);
|
215
|
+
}
|
216
|
+
|
217
|
+
inline static VALUE
|
218
|
+
f_zero_p(VALUE x)
|
219
|
+
{
|
220
|
+
if (FIXNUM_P(x)) {
|
221
|
+
return f_boolcast(FIX2LONG(x) == 0);
|
222
|
+
}
|
223
|
+
else if (RB_TYPE_P(x, T_BIGNUM)) {
|
224
|
+
return Qfalse;
|
225
|
+
}
|
226
|
+
else if (RB_TYPE_P(x, T_RATIONAL)) {
|
227
|
+
VALUE num = RRATIONAL(x)->num;
|
228
|
+
return f_boolcast(FIXNUM_P(num) && FIX2LONG(num) == 0);
|
229
|
+
}
|
230
|
+
return rb_funcall(x, id_eqeq_p, 1, ZERO);
|
231
|
+
}
|
232
|
+
|
233
|
+
inline static VALUE
|
234
|
+
f_negate(VALUE x)
|
235
|
+
{
|
236
|
+
return rb_funcall(x, id_negate, 0);
|
237
|
+
}
|
238
|
+
|
239
|
+
inline static VALUE
|
240
|
+
f_to_f(VALUE x)
|
241
|
+
{
|
242
|
+
if (RB_TYPE_P(x, T_STRING))
|
243
|
+
return DBL2NUM(rb_str_to_dbl(x, 0));
|
244
|
+
return rb_funcall(x, id_to_f, 0);
|
245
|
+
}
|
246
|
+
|
247
|
+
inline static long
|
248
|
+
i_gcd(long x, long y)
|
249
|
+
{
|
250
|
+
if (x < 0)
|
251
|
+
x = -x;
|
252
|
+
if (y < 0)
|
253
|
+
y = -y;
|
254
|
+
|
255
|
+
if (x == 0)
|
256
|
+
return y;
|
257
|
+
if (y == 0)
|
258
|
+
return x;
|
259
|
+
|
260
|
+
while (x > 0) {
|
261
|
+
long t = x;
|
262
|
+
x = y % x;
|
263
|
+
y = t;
|
264
|
+
}
|
265
|
+
return y;
|
266
|
+
}
|
267
|
+
|
268
|
+
inline static VALUE
|
269
|
+
f_imul(long a, long b)
|
270
|
+
{
|
271
|
+
VALUE r;
|
272
|
+
|
273
|
+
if (a == 0 || b == 0)
|
274
|
+
return ZERO;
|
275
|
+
else if (a == 1)
|
276
|
+
return LONG2NUM(b);
|
277
|
+
else if (b == 1)
|
278
|
+
return LONG2NUM(a);
|
279
|
+
|
280
|
+
if (MUL_OVERFLOW_LONG_P(a, b))
|
281
|
+
r = rb_big_mul(rb_int2big(a), rb_int2big(b));
|
282
|
+
else
|
283
|
+
r = LONG2NUM(a * b);
|
284
|
+
|
285
|
+
return r;
|
286
|
+
}
|
287
|
+
|
288
|
+
inline static VALUE
|
289
|
+
f_mul(VALUE x, VALUE y)
|
290
|
+
{
|
291
|
+
if (FIXNUM_P(y)) {
|
292
|
+
long iy = FIX2LONG(y);
|
293
|
+
if (iy == 0) {
|
294
|
+
if (FIXNUM_P(x) || RB_TYPE_P(x, T_BIGNUM))
|
295
|
+
return ZERO;
|
296
|
+
}
|
297
|
+
else if (iy == 1)
|
298
|
+
return x;
|
299
|
+
}
|
300
|
+
if (FIXNUM_P(x)) {
|
301
|
+
long ix = FIX2LONG(x);
|
302
|
+
if (ix == 0) {
|
303
|
+
if (FIXNUM_P(y) || RB_TYPE_P(y, T_BIGNUM))
|
304
|
+
return ZERO;
|
305
|
+
}
|
306
|
+
else if (ix == 1)
|
307
|
+
return y;
|
308
|
+
}
|
309
|
+
return rb_funcall(x, '*', 1, y);
|
310
|
+
}
|
311
|
+
|
312
|
+
inline static VALUE
|
313
|
+
f_idiv(VALUE x, VALUE y)
|
314
|
+
{
|
315
|
+
return rb_funcall(x, id_idiv, 1, y);
|
316
|
+
}
|
317
|
+
|
318
|
+
inline static VALUE
|
319
|
+
f_mod(VALUE x, VALUE y)
|
320
|
+
{
|
321
|
+
return rb_funcall(x, '%', 1, y);
|
322
|
+
}
|
323
|
+
|
324
|
+
inline static VALUE
|
325
|
+
f_gcd_normal(VALUE x, VALUE y)
|
326
|
+
{
|
327
|
+
VALUE z;
|
328
|
+
|
329
|
+
if (FIXNUM_P(x) && FIXNUM_P(y))
|
330
|
+
return LONG2NUM(i_gcd(FIX2LONG(x), FIX2LONG(y)));
|
331
|
+
|
332
|
+
if (f_negative_p(x))
|
333
|
+
x = f_negate(x);
|
334
|
+
if (f_negative_p(y))
|
335
|
+
y = f_negate(y);
|
336
|
+
|
337
|
+
if (f_zero_p(x))
|
338
|
+
return y;
|
339
|
+
if (f_zero_p(y))
|
340
|
+
return x;
|
341
|
+
|
342
|
+
for (;;) {
|
343
|
+
if (FIXNUM_P(x)) {
|
344
|
+
if (FIX2LONG(x) == 0)
|
345
|
+
return y;
|
346
|
+
if (FIXNUM_P(y))
|
347
|
+
return LONG2NUM(i_gcd(FIX2LONG(x), FIX2LONG(y)));
|
348
|
+
}
|
349
|
+
z = x;
|
350
|
+
x = f_mod(y, x);
|
351
|
+
y = z;
|
352
|
+
}
|
353
|
+
/* NOTREACHED */
|
354
|
+
}
|
355
|
+
|
356
|
+
inline static VALUE
|
357
|
+
f_gcd(VALUE x, VALUE y)
|
358
|
+
{
|
359
|
+
return f_gcd_normal(x, y);
|
360
|
+
}
|
361
|
+
|
362
|
+
inline static VALUE
|
363
|
+
nurat_s_new_internal(VALUE klass, VALUE num, VALUE den)
|
364
|
+
{
|
365
|
+
NEWOBJ_OF(obj, struct RRational, klass, T_RATIONAL | (RGENGC_WB_PROTECTED_RATIONAL ? FL_WB_PROTECTED : 0));
|
366
|
+
RRATIONAL_SET_NUM(obj, num);
|
367
|
+
RRATIONAL_SET_DEN(obj, den);
|
368
|
+
return (VALUE)obj;
|
369
|
+
}
|
370
|
+
|
371
|
+
inline static VALUE
|
372
|
+
nurat_s_canonicalize_internal_no_reduce(VALUE klass, VALUE num, VALUE den)
|
373
|
+
{
|
374
|
+
switch (FIX2INT(f_cmp(den, ZERO))) {
|
375
|
+
case -1:
|
376
|
+
num = f_negate(num);
|
377
|
+
den = f_negate(den);
|
378
|
+
break;
|
379
|
+
case 0:
|
380
|
+
rb_raise_zerodiv();
|
381
|
+
break;
|
382
|
+
}
|
383
|
+
|
384
|
+
return nurat_s_new_internal(klass, num, den);
|
385
|
+
}
|
386
|
+
|
387
|
+
inline static VALUE
|
388
|
+
f_addsub(VALUE self, VALUE anum, VALUE aden, VALUE bnum, VALUE bden, int k)
|
389
|
+
{
|
390
|
+
VALUE num, den;
|
391
|
+
|
392
|
+
if (FIXNUM_P(anum) && FIXNUM_P(aden) && FIXNUM_P(bnum) && FIXNUM_P(bden)) {
|
393
|
+
long an = FIX2LONG(anum);
|
394
|
+
long ad = FIX2LONG(aden);
|
395
|
+
long bn = FIX2LONG(bnum);
|
396
|
+
long bd = FIX2LONG(bden);
|
397
|
+
long ig = i_gcd(ad, bd);
|
398
|
+
|
399
|
+
VALUE g = LONG2NUM(ig);
|
400
|
+
VALUE a = f_imul(an, bd / ig);
|
401
|
+
VALUE b = f_imul(bn, ad / ig);
|
402
|
+
|
403
|
+
VALUE c;
|
404
|
+
if (k == '+')
|
405
|
+
c = f_add(a, b);
|
406
|
+
else
|
407
|
+
c = f_sub(a, b);
|
408
|
+
|
409
|
+
b = f_idiv(aden, g);
|
410
|
+
g = f_gcd(c, g);
|
411
|
+
num = f_idiv(c, g);
|
412
|
+
a = f_idiv(bden, g);
|
413
|
+
den = f_mul(a, b);
|
414
|
+
}
|
415
|
+
else {
|
416
|
+
VALUE g = f_gcd(aden, bden);
|
417
|
+
VALUE a = f_mul(anum, f_idiv(bden, g));
|
418
|
+
VALUE b = f_mul(bnum, f_idiv(aden, g));
|
419
|
+
|
420
|
+
VALUE c;
|
421
|
+
if (k == '+')
|
422
|
+
c = f_add(a, b);
|
423
|
+
else
|
424
|
+
c = f_sub(a, b);
|
425
|
+
|
426
|
+
b = f_idiv(aden, g);
|
427
|
+
g = f_gcd(c, g);
|
428
|
+
num = f_idiv(c, g);
|
429
|
+
a = f_idiv(bden, g);
|
430
|
+
den = f_mul(a, b);
|
431
|
+
}
|
432
|
+
|
433
|
+
return nurat_s_canonicalize_internal_no_reduce(CLASS_OF(self), num, den);
|
434
|
+
}
|
435
|
+
|
436
|
+
static VALUE
|
437
|
+
rb_rational_plus(VALUE self, VALUE other)
|
438
|
+
{
|
439
|
+
if (RB_TYPE_P(other, T_FIXNUM) || RB_TYPE_P(other, T_BIGNUM)) {
|
440
|
+
VALUE num = RRATIONAL(self)->num;
|
441
|
+
VALUE den = RRATIONAL(self)->den;
|
442
|
+
|
443
|
+
return f_addsub(self, num, den, other, ONE, idPLUS);
|
444
|
+
}
|
445
|
+
else if (RB_TYPE_P(other, T_FLOAT)) {
|
446
|
+
return f_add(f_to_f(self), other);
|
447
|
+
}
|
448
|
+
else if (RB_TYPE_P(other, T_RATIONAL)) {
|
449
|
+
VALUE anum = RRATIONAL(self)->num;
|
450
|
+
VALUE aden = RRATIONAL(self)->den;
|
451
|
+
VALUE bnum = RRATIONAL(other)->num;
|
452
|
+
VALUE bden = RRATIONAL(other)->den;
|
453
|
+
|
454
|
+
return f_addsub(self, anum, aden, bnum, bden, '+');
|
455
|
+
}
|
456
|
+
else {
|
457
|
+
return rb_num_coerce_bin(self, other, idPLUS);
|
458
|
+
}
|
459
|
+
}
|
460
|
+
#endif
|
461
|
+
|
462
|
+
#ifndef HAVE_ARRAY_SUM
|
463
|
+
static VALUE
|
464
|
+
ary_sum(int argc, VALUE* argv, VALUE ary)
|
465
|
+
{
|
466
|
+
VALUE e, v, r;
|
467
|
+
long i, n;
|
468
|
+
int block_given;
|
469
|
+
|
470
|
+
if (rb_scan_args(argc, argv, "01", &v) == 0)
|
471
|
+
v = LONG2FIX(0);
|
472
|
+
|
473
|
+
block_given = rb_block_given_p();
|
474
|
+
|
475
|
+
if (RARRAY_LEN(ary) == 0)
|
476
|
+
return v;
|
477
|
+
|
478
|
+
n = 0;
|
479
|
+
r = Qundef;
|
480
|
+
for (i = 0; i < RARRAY_LEN(ary); i++) {
|
481
|
+
e = RARRAY_AREF(ary, i);
|
482
|
+
if (block_given)
|
483
|
+
e = rb_yield(e);
|
484
|
+
if (FIXNUM_P(e)) {
|
485
|
+
n += FIX2LONG(e); /* should not overflow long type */
|
486
|
+
if (!FIXABLE(n)) {
|
487
|
+
v = rb_big_plus(LONG2NUM(n), v);
|
488
|
+
n = 0;
|
489
|
+
}
|
490
|
+
}
|
491
|
+
else if (RB_TYPE_P(e, T_BIGNUM))
|
492
|
+
v = rb_big_plus(e, v);
|
493
|
+
else if (RB_TYPE_P(e, T_RATIONAL)) {
|
494
|
+
if (r == Qundef)
|
495
|
+
r = e;
|
496
|
+
else
|
497
|
+
r = rb_rational_plus(r, e);
|
498
|
+
}
|
499
|
+
else
|
500
|
+
goto not_exact;
|
501
|
+
}
|
502
|
+
|
503
|
+
if (n != 0)
|
504
|
+
v = rb_fix_plus(LONG2FIX(n), v);
|
505
|
+
if (r != Qundef)
|
506
|
+
v = rb_rational_plus(r, v);
|
507
|
+
return v;
|
508
|
+
|
509
|
+
not_exact:
|
510
|
+
if (n != 0)
|
511
|
+
v = rb_fix_plus(LONG2FIX(n), v);
|
512
|
+
if (r != Qundef)
|
513
|
+
v = rb_rational_plus(r, v);
|
514
|
+
|
515
|
+
if (RB_FLOAT_TYPE_P(e)) {
|
516
|
+
/* Kahan's compensated summation algorithm */
|
517
|
+
double f, c;
|
518
|
+
|
519
|
+
f = NUM2DBL(v);
|
520
|
+
c = 0.0;
|
521
|
+
goto has_float_value;
|
522
|
+
for (; i < RARRAY_LEN(ary); i++) {
|
523
|
+
double x, y, t;
|
524
|
+
e = RARRAY_AREF(ary, i);
|
525
|
+
if (block_given)
|
526
|
+
e = rb_yield(e);
|
527
|
+
if (RB_FLOAT_TYPE_P(e))
|
528
|
+
has_float_value:
|
529
|
+
x = RFLOAT_VALUE(e);
|
530
|
+
else if (FIXNUM_P(e))
|
531
|
+
x = FIX2LONG(e);
|
532
|
+
else if (RB_TYPE_P(e, T_BIGNUM))
|
533
|
+
x = rb_big2dbl(e);
|
534
|
+
else if (RB_TYPE_P(e, T_RATIONAL))
|
535
|
+
x = rb_num2dbl(e);
|
536
|
+
else
|
537
|
+
goto not_float;
|
538
|
+
|
539
|
+
y = x - c;
|
540
|
+
t = f + y;
|
541
|
+
c = (t - f) - y;
|
542
|
+
f = t;
|
543
|
+
}
|
544
|
+
return DBL2NUM(f);
|
545
|
+
|
546
|
+
not_float:
|
547
|
+
v = DBL2NUM(f);
|
548
|
+
}
|
549
|
+
|
550
|
+
goto has_some_value;
|
551
|
+
for (; i < RARRAY_LEN(ary); i++) {
|
552
|
+
e = RARRAY_AREF(ary, i);
|
553
|
+
if (block_given)
|
554
|
+
e = rb_yield(e);
|
555
|
+
has_some_value:
|
556
|
+
v = rb_funcall(v, idPLUS, 1, e);
|
557
|
+
}
|
558
|
+
|
559
|
+
return v;
|
560
|
+
}
|
561
|
+
#endif
|
562
|
+
|
563
|
+
#define ENUM_WANT_SVALUE() do { \
|
564
|
+
e = rb_enum_values_pack(argc, argv); \
|
565
|
+
} while (0)
|
566
|
+
|
567
|
+
struct enum_sum_memo {
|
568
|
+
VALUE v, r;
|
569
|
+
long n;
|
570
|
+
double f, c;
|
571
|
+
int block_given;
|
572
|
+
int float_value;
|
573
|
+
};
|
574
|
+
|
575
|
+
static VALUE
|
576
|
+
enum_sum_iter_i(RB_BLOCK_CALL_FUNC_ARGLIST(e, args))
|
577
|
+
{
|
578
|
+
struct enum_sum_memo *memo = (struct enum_sum_memo *)args;
|
579
|
+
long n = memo->n;
|
580
|
+
VALUE v = memo->v;
|
581
|
+
VALUE r = memo->r;
|
582
|
+
double f = memo->f;
|
583
|
+
double c = memo->c;
|
584
|
+
|
585
|
+
ENUM_WANT_SVALUE();
|
586
|
+
|
587
|
+
if (memo->block_given)
|
588
|
+
e = rb_yield(e);
|
589
|
+
|
590
|
+
if (memo->float_value)
|
591
|
+
goto float_value;
|
592
|
+
|
593
|
+
if (FIXNUM_P(v) || RB_TYPE_P(v, T_BIGNUM) || RB_TYPE_P(v, T_RATIONAL)) {
|
594
|
+
if (FIXNUM_P(e)) {
|
595
|
+
n += FIX2LONG(e); /* should not overflow long type */
|
596
|
+
if (!FIXABLE(n)) {
|
597
|
+
v = rb_big_plus(LONG2NUM(n), v);
|
598
|
+
n = 0;
|
599
|
+
}
|
600
|
+
}
|
601
|
+
else if (RB_TYPE_P(e, T_BIGNUM))
|
602
|
+
v = rb_big_plus(e, v);
|
603
|
+
else if (RB_TYPE_P(e, T_RATIONAL)) {
|
604
|
+
if (r == Qundef)
|
605
|
+
r = e;
|
606
|
+
else
|
607
|
+
r = rb_rational_plus(r, e);
|
608
|
+
}
|
609
|
+
else {
|
610
|
+
if (n != 0) {
|
611
|
+
v = rb_fix_plus(LONG2FIX(n), v);
|
612
|
+
n = 0;
|
613
|
+
}
|
614
|
+
if (r != Qundef) {
|
615
|
+
v = rb_rational_plus(r, v);
|
616
|
+
r = Qundef;
|
617
|
+
}
|
618
|
+
if (RB_FLOAT_TYPE_P(e)) {
|
619
|
+
f = NUM2DBL(v);
|
620
|
+
c = 0.0;
|
621
|
+
memo->float_value = 1;
|
622
|
+
goto float_value;
|
623
|
+
}
|
624
|
+
else
|
625
|
+
goto some_value;
|
626
|
+
}
|
627
|
+
}
|
628
|
+
else if (RB_FLOAT_TYPE_P(v)) {
|
629
|
+
/* Kahan's compensated summation algorithm */
|
630
|
+
double x, y, t;
|
631
|
+
|
632
|
+
float_value:
|
633
|
+
if (RB_FLOAT_TYPE_P(e))
|
634
|
+
x = RFLOAT_VALUE(e);
|
635
|
+
else if (FIXNUM_P(e))
|
636
|
+
x = FIX2LONG(e);
|
637
|
+
else if (RB_TYPE_P(e, T_BIGNUM))
|
638
|
+
x = rb_big2dbl(e);
|
639
|
+
else if (RB_TYPE_P(e, T_RATIONAL))
|
640
|
+
x = rb_num2dbl(e);
|
641
|
+
else {
|
642
|
+
v = DBL2NUM(f);
|
643
|
+
memo->float_value = 0;
|
644
|
+
goto some_value;
|
645
|
+
}
|
646
|
+
|
647
|
+
y = x - c;
|
648
|
+
t = f + y;
|
649
|
+
c = (t - f) - y;
|
650
|
+
f = t;
|
651
|
+
}
|
652
|
+
else {
|
653
|
+
some_value:
|
654
|
+
v = rb_funcall(v, idPLUS, 1, e);
|
655
|
+
}
|
656
|
+
|
657
|
+
memo->v = v;
|
658
|
+
memo->n = n;
|
659
|
+
memo->r = r;
|
660
|
+
memo->f = f;
|
661
|
+
memo->c = c;
|
662
|
+
|
663
|
+
return Qnil;
|
664
|
+
}
|
665
|
+
|
666
|
+
static VALUE
|
667
|
+
enum_stat_sum(int argc, VALUE* argv, VALUE obj)
|
668
|
+
{
|
669
|
+
struct enum_sum_memo memo;
|
670
|
+
|
671
|
+
if (rb_scan_args(argc, argv, "01", &memo.v) == 0)
|
672
|
+
memo.v = LONG2FIX(0);
|
673
|
+
|
674
|
+
memo.block_given = rb_block_given_p();
|
675
|
+
|
676
|
+
memo.n = 0;
|
677
|
+
memo.r = Qundef;
|
678
|
+
|
679
|
+
if ((memo.float_value = RB_FLOAT_TYPE_P(memo.v))) {
|
680
|
+
memo.f = RFLOAT_VALUE(memo.v);
|
681
|
+
memo.c = 0.0;
|
682
|
+
}
|
683
|
+
|
684
|
+
rb_block_call(obj, id_each, 0, 0, enum_sum_iter_i, (VALUE)&memo);
|
685
|
+
|
686
|
+
if (memo.float_value) {
|
687
|
+
return DBL2NUM(memo.f);
|
688
|
+
}
|
689
|
+
else {
|
690
|
+
if (memo.n != 0)
|
691
|
+
memo.v = rb_fix_plus(LONG2FIX(memo.n), memo.v);
|
692
|
+
if (memo.r != Qundef)
|
693
|
+
memo.v = rb_rational_plus(memo.r, memo.v);
|
694
|
+
return memo.v;
|
695
|
+
}
|
696
|
+
}
|
697
|
+
|
698
|
+
static VALUE
|
699
|
+
enum_stat_mean_variance(VALUE obj)
|
700
|
+
{
|
701
|
+
return rb_assoc_new(INT2FIX(0), INT2FIX(0));
|
702
|
+
}
|
703
|
+
|
704
|
+
static VALUE
|
705
|
+
enum_stat_mean(VALUE obj)
|
706
|
+
{
|
707
|
+
VALUE ary = enum_stat_mean_variance(obj);
|
708
|
+
return RARRAY_AREF(ary, 0);
|
709
|
+
}
|
710
|
+
|
711
|
+
static VALUE
|
712
|
+
enum_stat_variance(VALUE obj)
|
713
|
+
{
|
714
|
+
VALUE ary = enum_stat_mean_variance(obj);
|
715
|
+
return RARRAY_AREF(ary, 1);
|
716
|
+
}
|
717
|
+
|
718
|
+
static VALUE
|
719
|
+
enum_stat_mean_stddev(VALUE obj)
|
720
|
+
{
|
721
|
+
VALUE ary = enum_stat_mean_variance(obj);
|
722
|
+
VALUE var = RARRAY_AREF(ary, 1);
|
723
|
+
VALUE stddev = rb_funcall(var, idPow, 1, DBL2NUM(0.5));
|
724
|
+
RARRAY_ASET(ary, 1, stddev);
|
725
|
+
return ary;
|
726
|
+
}
|
727
|
+
|
728
|
+
static VALUE
|
729
|
+
enum_stat_stddev(VALUE obj)
|
730
|
+
{
|
731
|
+
VALUE ary = enum_stat_mean_stddev(obj);
|
732
|
+
return RARRAY_AREF(ary, 1);
|
733
|
+
}
|
734
|
+
|
735
|
+
void
|
736
|
+
Init_extension(void)
|
737
|
+
{
|
738
|
+
rb_define_method(rb_mEnumerable, "sum", enum_stat_sum, -1);
|
739
|
+
|
740
|
+
#if 0
|
741
|
+
rb_define_method(rb_mEnumerable, "mean_variance", enum_stat_mean_variance, 0);
|
742
|
+
rb_define_method(rb_mEnumerable, "mean", enum_stat_mean, 0);
|
743
|
+
rb_define_alias(rb_mEnumerable, "average", "mean");
|
744
|
+
rb_define_method(rb_mEnumerable, "variance", enum_stat_variance, 0);
|
745
|
+
rb_define_alias(rb_mEnumerable, "var", "variance");
|
746
|
+
rb_define_method(rb_mEnumerable, "mean_stddev", enum_stat_mean_stddev, 0);
|
747
|
+
rb_define_method(rb_mEnumerable, "stddev", enum_stat_stddev, 0);
|
748
|
+
#endif
|
749
|
+
|
750
|
+
#ifndef HAVE_ARRAY_SUM
|
751
|
+
rb_define_method(rb_cArray, "sum", ary_sum, -1);
|
752
|
+
#endif
|
753
|
+
|
754
|
+
idPLUS = '+';
|
755
|
+
idMINUS = '-';
|
756
|
+
idSTAR = '*';
|
757
|
+
idPow = rb_intern("**");
|
758
|
+
id_eqeq_p = rb_intern("==");
|
759
|
+
id_idiv = rb_intern("div");
|
760
|
+
id_negate = rb_intern("-@");
|
761
|
+
id_to_f = rb_intern("to_f");
|
762
|
+
id_cmp = rb_intern("<=>");
|
763
|
+
id_each = rb_intern("each");
|
764
|
+
id_real_p = rb_intern("real?");
|
765
|
+
}
|
metadata
ADDED
@@ -0,0 +1,129 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: enumerable-statistics
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Kenta Murata
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2016-05-01 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: bundler
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - "~>"
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '1.11'
|
20
|
+
type: :development
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - "~>"
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '1.11'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: rake
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '10.0'
|
34
|
+
type: :development
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '10.0'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: rake-compiler
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - "~>"
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: 0.9.8
|
48
|
+
type: :development
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: 0.9.8
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: rspec
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - "~>"
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '3.4'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - "~>"
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '3.4'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: fuubar
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - ">="
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '0'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - ">="
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '0'
|
83
|
+
description: This library provides statistics features for Enumerable
|
84
|
+
email:
|
85
|
+
- mrkn@mrkn.jp
|
86
|
+
executables: []
|
87
|
+
extensions:
|
88
|
+
- ext/enumerable/statistics/extension/extconf.rb
|
89
|
+
extra_rdoc_files: []
|
90
|
+
files:
|
91
|
+
- ".gitignore"
|
92
|
+
- ".rspec"
|
93
|
+
- ".travis.yml"
|
94
|
+
- Gemfile
|
95
|
+
- README.md
|
96
|
+
- Rakefile
|
97
|
+
- bin/console
|
98
|
+
- bin/rspec
|
99
|
+
- bin/setup
|
100
|
+
- enumerable-statistics.gemspec
|
101
|
+
- ext/enumerable/statistics/extension/extconf.rb
|
102
|
+
- ext/enumerable/statistics/extension/statistics.c
|
103
|
+
- lib/enumerable/statistics.rb
|
104
|
+
- lib/enumerable/statistics/version.rb
|
105
|
+
homepage: https://github.com/mrkn/enumerable-statistics
|
106
|
+
licenses: []
|
107
|
+
metadata: {}
|
108
|
+
post_install_message:
|
109
|
+
rdoc_options: []
|
110
|
+
require_paths:
|
111
|
+
- ext
|
112
|
+
- lib
|
113
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
114
|
+
requirements:
|
115
|
+
- - ">="
|
116
|
+
- !ruby/object:Gem::Version
|
117
|
+
version: '0'
|
118
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
119
|
+
requirements:
|
120
|
+
- - ">="
|
121
|
+
- !ruby/object:Gem::Version
|
122
|
+
version: '0'
|
123
|
+
requirements: []
|
124
|
+
rubyforge_project:
|
125
|
+
rubygems_version: 2.6.3
|
126
|
+
signing_key:
|
127
|
+
specification_version: 4
|
128
|
+
summary: Statistics features for Enumerable
|
129
|
+
test_files: []
|