means 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
data/.document ADDED
@@ -0,0 +1,5 @@
1
+ lib/**/*.rb
2
+ bin/*
3
+ -
4
+ features/**/*.feature
5
+ LICENSE.txt
data/.rspec ADDED
@@ -0,0 +1 @@
1
+ --color
data/Gemfile ADDED
@@ -0,0 +1,13 @@
1
+ source "http://rubygems.org"
2
+ # Add dependencies required to use your gem here.
3
+ # Example:
4
+ # gem "activesupport", ">= 2.3.5"
5
+
6
+ # Add dependencies to develop your gem here.
7
+ # Include everything needed to run rake, tests, features, etc.
8
+ group :development do
9
+ gem "rspec", "~> 2.3.0"
10
+ gem "bundler", "~> 1.0.0"
11
+ gem "jeweler", "~> 1.6.4"
12
+ gem "rcov", ">= 0"
13
+ end
data/Gemfile.lock ADDED
@@ -0,0 +1,28 @@
1
+ GEM
2
+ remote: http://rubygems.org/
3
+ specs:
4
+ diff-lcs (1.1.3)
5
+ git (1.2.5)
6
+ jeweler (1.6.4)
7
+ bundler (~> 1.0)
8
+ git (>= 1.2.5)
9
+ rake
10
+ rake (0.9.2.2)
11
+ rcov (0.9.11)
12
+ rspec (2.3.0)
13
+ rspec-core (~> 2.3.0)
14
+ rspec-expectations (~> 2.3.0)
15
+ rspec-mocks (~> 2.3.0)
16
+ rspec-core (2.3.1)
17
+ rspec-expectations (2.3.0)
18
+ diff-lcs (~> 1.1.2)
19
+ rspec-mocks (2.3.0)
20
+
21
+ PLATFORMS
22
+ ruby
23
+
24
+ DEPENDENCIES
25
+ bundler (~> 1.0.0)
26
+ jeweler (~> 1.6.4)
27
+ rcov
28
+ rspec (~> 2.3.0)
data/LICENSE.txt ADDED
@@ -0,0 +1,20 @@
1
+ Copyright (c) 2011 Nigel Lowry
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.rdoc ADDED
@@ -0,0 +1,19 @@
1
+ = means
2
+
3
+ A very small gem to calculate the different kinds of mean in a data set.
4
+
5
+ == Contributing to means
6
+
7
+ * Check out the latest master to make sure the feature hasn't been implemented or the bug hasn't been fixed yet
8
+ * Check out the issue tracker to make sure someone already hasn't requested it and/or contributed it
9
+ * Fork the project
10
+ * Start a feature/bugfix branch
11
+ * Commit and push until you are happy with your contribution
12
+ * Make sure to add tests for it. This is important so I don't break it in a future version unintentionally.
13
+ * Please try not to mess with the Rakefile, version, or history. If you want to have your own version, or is otherwise necessary, that is fine, but please isolate to its own commit so I can cherry-pick around it.
14
+
15
+ == Copyright
16
+
17
+ Copyright (c) 2011 Nigel Lowry. See LICENSE.txt for
18
+ further details.
19
+
data/Rakefile ADDED
@@ -0,0 +1,49 @@
1
+ # encoding: utf-8
2
+
3
+ require 'rubygems'
4
+ require 'bundler'
5
+ begin
6
+ Bundler.setup(:default, :development)
7
+ rescue Bundler::BundlerError => e
8
+ $stderr.puts e.message
9
+ $stderr.puts "Run `bundle install` to install missing gems"
10
+ exit e.status_code
11
+ end
12
+ require 'rake'
13
+
14
+ require 'jeweler'
15
+ Jeweler::Tasks.new do |gem|
16
+ # gem is a Gem::Specification... see http://docs.rubygems.org/read/chapter/20 for more options
17
+ gem.name = "means"
18
+ gem.homepage = "http://github.com/nigel-lowry/means"
19
+ gem.license = "MIT"
20
+ gem.summary = %Q{Small gem to work out the arithmetic, harmonic and geometric means of a data set}
21
+ gem.description = %Q{Calculates the diferent kinds of mean for a data set. Also has methods to do this cumulatively given an accumulator rather than going over whole data set each time given a single new value}
22
+ gem.email = "nigel-lowry@ultra.eclipse.co.uk"
23
+ gem.authors = ["Nigel Lowry"]
24
+ # dependencies defined in Gemfile
25
+ end
26
+ Jeweler::RubygemsDotOrgTasks.new
27
+
28
+ require 'rspec/core'
29
+ require 'rspec/core/rake_task'
30
+ RSpec::Core::RakeTask.new(:spec) do |spec|
31
+ spec.pattern = FileList['spec/**/*_spec.rb']
32
+ end
33
+
34
+ RSpec::Core::RakeTask.new(:rcov) do |spec|
35
+ spec.pattern = 'spec/**/*_spec.rb'
36
+ spec.rcov = true
37
+ end
38
+
39
+ task :default => :spec
40
+
41
+ require 'rake/rdoctask'
42
+ Rake::RDocTask.new do |rdoc|
43
+ version = File.exist?('VERSION') ? File.read('VERSION') : ""
44
+
45
+ rdoc.rdoc_dir = 'rdoc'
46
+ rdoc.title = "means #{version}"
47
+ rdoc.rdoc_files.include('README*')
48
+ rdoc.rdoc_files.include('lib/**/*.rb')
49
+ end
data/VERSION ADDED
@@ -0,0 +1 @@
1
+ 0.1.0
data/lib/Mean.rb ADDED
@@ -0,0 +1,59 @@
1
+ require 'mathn'
2
+
3
+ # Allows calculation of arithmetic, geometric and harmonic means.
4
+ # Class methods allow the passing in of a data set.
5
+ # Alternatively an object can be created and elements can
6
+ # be added with Mean#push
7
+ class Mean
8
+
9
+ # Calculate the arithmetic mean of _data_
10
+ def Mean.arithmetic(data)
11
+ data.reduce(:+) / data.size unless data.empty?
12
+ end
13
+
14
+ # Calculate the geometric mean of _data_
15
+ def Mean.geometric(data)
16
+ data.reduce(:*) ** (1 / data.size) unless data.empty?
17
+ end
18
+
19
+ # Calculate the harmonic mean of _data_
20
+ def Mean.harmonic(data)
21
+ data.size / data.reduce(0) {|sum, element| sum += (1 / element)} unless data.empty?
22
+ end
23
+
24
+ # Remember the initial state.
25
+ #
26
+ # If you are passing in an initial state:
27
+ # [arithmetic mean] needs :sum and :count
28
+ # [geometric mean] needs :product and :count.
29
+ # [harmonic mean] needs :sum_of_reciprocals and :count
30
+ def initialize(params={})
31
+ @sum ||= params[:sum] ||= 0
32
+ @sum_of_reciprocals ||= params[:sum_of_reciprocals] ||= 0
33
+ @product ||= params[:product] ||= 1
34
+ @count ||= params[:count] ||= 0
35
+ end
36
+
37
+ # Add _element_ to the data set
38
+ def push(element)
39
+ @sum += element
40
+ @sum_of_reciprocals += (1 / element)
41
+ @product *= element
42
+ @count += 1
43
+ end
44
+
45
+ # Calculate the arithmetic mean of the data set
46
+ def arithmetic_mean
47
+ @sum / @count unless @count == 0
48
+ end
49
+
50
+ # Calculate the geometric mean of the data set
51
+ def geometric_mean
52
+ @product ** (1 / @count) unless @count == 0
53
+ end
54
+
55
+ # Calculate the harmonic mean of the data set
56
+ def harmonic_mean
57
+ @count / @sum_of_reciprocals unless @count == 0
58
+ end
59
+ end
data/means.gemspec ADDED
@@ -0,0 +1,60 @@
1
+ # Generated by jeweler
2
+ # DO NOT EDIT THIS FILE DIRECTLY
3
+ # Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
4
+ # -*- encoding: utf-8 -*-
5
+
6
+ Gem::Specification.new do |s|
7
+ s.name = "means"
8
+ s.version = "0.1.0"
9
+
10
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
11
+ s.authors = ["Nigel Lowry"]
12
+ s.date = "2011-11-06"
13
+ s.description = "Calculates the diferent kinds of mean for a data set. Also has methods to do this cumulatively given an accumulator rather than going over whole data set each time given a single new value"
14
+ s.email = "nigel-lowry@ultra.eclipse.co.uk"
15
+ s.extra_rdoc_files = [
16
+ "LICENSE.txt",
17
+ "README.rdoc"
18
+ ]
19
+ s.files = [
20
+ ".document",
21
+ ".rspec",
22
+ "Gemfile",
23
+ "Gemfile.lock",
24
+ "LICENSE.txt",
25
+ "README.rdoc",
26
+ "Rakefile",
27
+ "VERSION",
28
+ "lib/Mean.rb",
29
+ "means.gemspec",
30
+ "spec/Mean_spec.rb",
31
+ "spec/spec_helper.rb"
32
+ ]
33
+ s.homepage = "http://github.com/nigel-lowry/means"
34
+ s.licenses = ["MIT"]
35
+ s.require_paths = ["lib"]
36
+ s.rubygems_version = "1.8.10"
37
+ s.summary = "Small gem to work out the arithmetic, harmonic and geometric means of a data set"
38
+
39
+ if s.respond_to? :specification_version then
40
+ s.specification_version = 3
41
+
42
+ if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
43
+ s.add_development_dependency(%q<rspec>, ["~> 2.3.0"])
44
+ s.add_development_dependency(%q<bundler>, ["~> 1.0.0"])
45
+ s.add_development_dependency(%q<jeweler>, ["~> 1.6.4"])
46
+ s.add_development_dependency(%q<rcov>, [">= 0"])
47
+ else
48
+ s.add_dependency(%q<rspec>, ["~> 2.3.0"])
49
+ s.add_dependency(%q<bundler>, ["~> 1.0.0"])
50
+ s.add_dependency(%q<jeweler>, ["~> 1.6.4"])
51
+ s.add_dependency(%q<rcov>, [">= 0"])
52
+ end
53
+ else
54
+ s.add_dependency(%q<rspec>, ["~> 2.3.0"])
55
+ s.add_dependency(%q<bundler>, ["~> 1.0.0"])
56
+ s.add_dependency(%q<jeweler>, ["~> 1.6.4"])
57
+ s.add_dependency(%q<rcov>, [">= 0"])
58
+ end
59
+ end
60
+
data/spec/Mean_spec.rb ADDED
@@ -0,0 +1,165 @@
1
+ require File.expand_path(File.dirname(__FILE__) + '/spec_helper')
2
+
3
+ describe "Mean" do
4
+
5
+ describe "#arithmetic" do
6
+ it "should be nil for an empty array" do
7
+ Mean.arithmetic([]).should be_nil
8
+ end
9
+
10
+ it "should be the singleton element of a singleton array" do
11
+ element = 5.5
12
+ Mean.arithmetic([element]).should == element
13
+ end
14
+
15
+ it "should be the common value in an array when all values are equal" do
16
+ element = 5.5
17
+ Mean.arithmetic([element, element]).should == element
18
+ end
19
+
20
+ it "should be 1.5 for [1, 2]" do
21
+ data = [1.0, 2.0]
22
+ Mean.arithmetic(data).should be_within(0.01).of(1.5)
23
+ end
24
+
25
+ it "should work with integers" do
26
+ data = [1, 2]
27
+ Mean.arithmetic(data).should be_within(0.01).of(1.5)
28
+ end
29
+
30
+ it "should work with an int then a float" do
31
+ data = [1, 2.0]
32
+ Mean.arithmetic(data).should be_within(0.01).of(1.5)
33
+ end
34
+
35
+ it "should work with a float then an int" do
36
+ data = [1.0, 2]
37
+ Mean.arithmetic(data).should be_within(0.01).of(1.5)
38
+ end
39
+
40
+ it "should work with three numbers" do
41
+ data = [1, 2, 3]
42
+ Mean.arithmetic(data).should == 2
43
+ end
44
+ end
45
+
46
+ describe "#geometric" do
47
+ it "should be nil for an empty array" do
48
+ Mean.geometric([]).should be_nil
49
+ end
50
+
51
+ it "should be the singleton element of a singleton array" do
52
+ element = 5.5
53
+ Mean.geometric([element]).should == element
54
+ end
55
+
56
+ it "should be the common value in a when all values are equal" do
57
+ element = 5.5
58
+ Mean.geometric([element, element]).should == element
59
+ end
60
+
61
+ it "should be zero when there is even a single zero" do
62
+ data = [1.1, 2.2, 0.0]
63
+ Mean.geometric(data).should == 0
64
+ end
65
+
66
+ it "should be the square root of the product of two numbers" do
67
+ d1, d2 = 2, 8
68
+ data = [d1, d2]
69
+ Mean.geometric(data).should == 4
70
+ end
71
+
72
+ it "should be the cube root of the product of three numbers" do
73
+ d1, d2, d3 = 4, 1, 1.0 / 32.0
74
+ data = [d1, d2, d3]
75
+ Mean.geometric(data).should == 1.0 / 2.0
76
+ end
77
+ end
78
+
79
+ describe "#harmonic" do
80
+ it "should be nil for an empty array" do
81
+ Mean.harmonic([]).should be_nil
82
+ end
83
+
84
+ it "should be the singleton element of a singleton array" do
85
+ element = 5.5
86
+ Mean.harmonic([element]).should == element
87
+ end
88
+
89
+ it "should be the common value in a when all values are equal" do
90
+ element = 5.5
91
+ Mean.harmonic([element, element]).should == element
92
+ end
93
+
94
+ it "should be be according to the formula with two elements" do
95
+ d1, d2 = 1.0, 2.0
96
+ data = [d1, d2]
97
+ Mean.harmonic(data).should be_within(0.01).of((2.0 * d1 * d2) / (d1 + d2))
98
+ end
99
+
100
+ it "should be 12/7 for [1, 2, 4]" do
101
+ data = [1.0, 2.0, 4.0]
102
+ Mean.harmonic(data).should be_within(0.01).of(1.7142)
103
+ end
104
+
105
+ it "should be zero if there is even one zero" do
106
+ data = [1.0, 2.0, 0.0]
107
+ Mean.harmonic(data).should == 0
108
+ end
109
+ end
110
+
111
+ describe "accumulators" do
112
+ before(:each) do
113
+ @m = Mean.new()
114
+ end
115
+
116
+ it "should accumulate the arithmetic mean" do
117
+ @m.arithmetic_mean.should be_nil
118
+ @m.push 1
119
+ @m.arithmetic_mean.should == 1
120
+ @m.push 2
121
+ @m.arithmetic_mean.should == 1.5
122
+ @m.push 3
123
+ @m.arithmetic_mean.should == 2
124
+ end
125
+
126
+ it "should accumulate the geometric mean" do
127
+ @m.geometric_mean.should be_nil
128
+ @m.push 2
129
+ @m.geometric_mean.should == 2
130
+ @m.push 8
131
+ @m.geometric_mean.should == 4
132
+ end
133
+
134
+ it "should accumulate the harmonic mean" do
135
+ @m.harmonic_mean.should be_nil
136
+ @m.push 1
137
+ @m.harmonic_mean.should == 1
138
+ @m.push 2
139
+ @m.harmonic_mean.should == 4.0 / 3.0
140
+ end
141
+ end
142
+
143
+ describe "initialising accumulator" do
144
+ it "should resurrect the arithmetic mean using the count and sum" do
145
+ m = Mean.new(count: 1, sum: 1)
146
+ m.arithmetic_mean.should == 1
147
+ m.push 2
148
+ m.arithmetic_mean.should == 1.5
149
+ end
150
+
151
+ it "should resurrect the geometric mean using the count and product" do
152
+ m = Mean.new(count: 1, product: 2)
153
+ m.geometric_mean.should == 2
154
+ m.push 8
155
+ m.geometric_mean.should == 4
156
+ end
157
+
158
+ it "should resurrect the harmonic mean using the count and sum of the reciprocals" do
159
+ m = Mean.new(count: 1, sum_of_reciprocals: 1)
160
+ m.harmonic_mean.should == 1
161
+ m.push 2
162
+ m.harmonic_mean.should == 4.0 / 3.0
163
+ end
164
+ end
165
+ end
@@ -0,0 +1,12 @@
1
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
2
+ $LOAD_PATH.unshift(File.dirname(__FILE__))
3
+ require 'rspec'
4
+ require 'Mean'
5
+
6
+ # Requires supporting files with custom matchers and macros, etc,
7
+ # in ./support/ and its subdirectories.
8
+ Dir["#{File.dirname(__FILE__)}/support/**/*.rb"].each {|f| require f}
9
+
10
+ RSpec.configure do |config|
11
+
12
+ end
metadata ADDED
@@ -0,0 +1,109 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: means
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Nigel Lowry
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2011-11-06 00:00:00.000000000Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: rspec
16
+ requirement: &70157793188260 !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ~>
20
+ - !ruby/object:Gem::Version
21
+ version: 2.3.0
22
+ type: :development
23
+ prerelease: false
24
+ version_requirements: *70157793188260
25
+ - !ruby/object:Gem::Dependency
26
+ name: bundler
27
+ requirement: &70157793187780 !ruby/object:Gem::Requirement
28
+ none: false
29
+ requirements:
30
+ - - ~>
31
+ - !ruby/object:Gem::Version
32
+ version: 1.0.0
33
+ type: :development
34
+ prerelease: false
35
+ version_requirements: *70157793187780
36
+ - !ruby/object:Gem::Dependency
37
+ name: jeweler
38
+ requirement: &70157793187300 !ruby/object:Gem::Requirement
39
+ none: false
40
+ requirements:
41
+ - - ~>
42
+ - !ruby/object:Gem::Version
43
+ version: 1.6.4
44
+ type: :development
45
+ prerelease: false
46
+ version_requirements: *70157793187300
47
+ - !ruby/object:Gem::Dependency
48
+ name: rcov
49
+ requirement: &70157793186820 !ruby/object:Gem::Requirement
50
+ none: false
51
+ requirements:
52
+ - - ! '>='
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ type: :development
56
+ prerelease: false
57
+ version_requirements: *70157793186820
58
+ description: Calculates the diferent kinds of mean for a data set. Also has methods
59
+ to do this cumulatively given an accumulator rather than going over whole data set
60
+ each time given a single new value
61
+ email: nigel-lowry@ultra.eclipse.co.uk
62
+ executables: []
63
+ extensions: []
64
+ extra_rdoc_files:
65
+ - LICENSE.txt
66
+ - README.rdoc
67
+ files:
68
+ - .document
69
+ - .rspec
70
+ - Gemfile
71
+ - Gemfile.lock
72
+ - LICENSE.txt
73
+ - README.rdoc
74
+ - Rakefile
75
+ - VERSION
76
+ - lib/Mean.rb
77
+ - means.gemspec
78
+ - spec/Mean_spec.rb
79
+ - spec/spec_helper.rb
80
+ homepage: http://github.com/nigel-lowry/means
81
+ licenses:
82
+ - MIT
83
+ post_install_message:
84
+ rdoc_options: []
85
+ require_paths:
86
+ - lib
87
+ required_ruby_version: !ruby/object:Gem::Requirement
88
+ none: false
89
+ requirements:
90
+ - - ! '>='
91
+ - !ruby/object:Gem::Version
92
+ version: '0'
93
+ segments:
94
+ - 0
95
+ hash: -1103619696357381530
96
+ required_rubygems_version: !ruby/object:Gem::Requirement
97
+ none: false
98
+ requirements:
99
+ - - ! '>='
100
+ - !ruby/object:Gem::Version
101
+ version: '0'
102
+ requirements: []
103
+ rubyforge_project:
104
+ rubygems_version: 1.8.10
105
+ signing_key:
106
+ specification_version: 3
107
+ summary: Small gem to work out the arithmetic, harmonic and geometric means of a data
108
+ set
109
+ test_files: []