How would you write it vs Smalltalk

i found a description of Smalltalk in

http://www.engin.umd.umich.edu/CIS/course.des/cis400/smalltalk/freq.html

and then I tried it in Ruby:

h = Hash.new(0)

print "Enter line: "

gets.downcase.each_byte {|b| c = “%c” % b; h[c] += 1 if c =~ /[a-z]/}

p h.sort

C:\rails\depot>ruby count_alpha2.rb
Enter line: This is a test of the line count
[[“a”, 1], [“c”, 1], [“e”, 3], [“f”, 1], [“h”, 2], [“i”, 3], [“l”, 1],
[“n”, 2],
[“o”, 2], [“s”, 3], [“t”, 5], [“u”, 1]]

I just wonder how would you write it? There is no each_char it seems,
unless with ActiveSupport.

On Sun, 23 Sep 2007 02:37:28 +0900, SpringFlowers AutumnMoon wrote:

I just wonder how would you write it? There is no each_char it seems,
unless with ActiveSupport.

running away but a quick idea: you could use Integer#chr to get a char
(well, a one-byte-string) from an integer.
And, since you’re dwelling with bytes anyway, you could just do a test
like
if c =~ ?a…?z
(?a being the literal to get the integer value of a character)

So something like:
gets.downcase.each_byte {|b| h[c.chr] += 1 if c =~ ?a…?z}

On 9/22/07, gabriele renzi [email protected] wrote:

On Sun, 23 Sep 2007 02:37:28 +0900, SpringFlowers AutumnMoon wrote:

I just wonder how would you write it? There is no each_char it seems,
unless with ActiveSupport.

So something like:
gets.downcase.each_byte {|b| h[c.chr] += 1 if c =~ ?a…?z}

Here’s another way:

puts “Enter line:”
h = Hash.new(0)
gets.scan(/[a-zA-Z]/) {|ch| h[ch.downcase] += 1 }
p h.sort


Rick DeNatale

My blog on Ruby
http://talklikeaduck.denhaven2.com/

Rick DeNatale schrieb:

Here’s another way:

puts “Enter line:”
h = Hash.new(0)
gets.scan(/[a-zA-Z]/) {|ch| h[ch.downcase] += 1 }
p h.sort

Well, we need an inject solution, right?

p gets.downcase.scan(/[a-z]/).inject({}){|h,c|
h.merge({c,h[c].to_i+1})}.sort

cheers

Simon

And this seems conceptually easier and it’s slightly faster(if I
benchmarked it correctly):

input = gets
h = Hash.new(0)

input.scan(/[a-z]/){|char| h[char] += 1}
p h.sort

============

require “benchmark”

input = “123Hello+#”

h = Hash.new(0)
Benchmark.bm(20) do |br|
br.report(“scan with block:”) do
200_000.times do
input.downcase.scan(/[a-z]/){|char| h[char] += 1}
end
end
end

p h.sort
puts

h = Hash.new(0)
Benchmark.bm(20) do |br|
br.report(“scan with inject:”) do
200_000.times do
input.downcase.scan(/[a-z]/).inject(h) do |h, c|
h[c] += 1
h
end
end
end
end

p h.sort

–output:–

                      user     system      total        real

scan with block: 2.690000 0.010000 2.700000 ( 2.714077)
[[“e”, 200000], [“h”, 200000], [“l”, 400000], [“o”, 200000]]

                      user     system      total        real

scan with inject: 2.830000 0.020000 2.850000 ( 2.856019)
[[“e”, 200000], [“h”, 200000], [“l”, 400000], [“o”, 200000]]

Simon Kröger wrote:

Rick DeNatale schrieb:

Here’s another way:

puts “Enter line:”
h = Hash.new(0)
gets.scan(/[a-zA-Z]/) {|ch| h[ch.downcase] += 1 }
p h.sort

Well, we need an inject solution, right?

p gets.downcase.scan(/[a-z]/).inject({}){|h,c|
h.merge({c,h[c].to_i+1})}.sort

How about:

p gets.downcase.scan(/[a-z]/).inject(Hash.new(0)){|h, c| h[c]+= 1;
h}.sort

7stud – wrote:

And this seems conceptually easier and it’s slightly faster(if I
benchmarked it correctly):

input = gets
h = Hash.new(0)

input.scan(/[a-z]/){|char| h[char] += 1}
p h.sort

Whoops. I didn’t downcase the input in that example, but I did downcase
the input in the benchmark tests. In any case, apologies to Rick
Denatale who already posted that solution.

On Sep 22, 2:39 pm, “Rick DeNatale” [email protected] wrote:

Here’s another way:

puts “Enter line:”
h = Hash.new(0)
gets.scan(/[a-zA-Z]/) {|ch| h[ch.downcase] += 1 }
p h.sort

More verbose but a teensy bit quicker by avoiding regex & downcase:

lambda do |str|
h = Hash.new(0)
str.each_byte do |b|
if (b > 96 && b < 123)
h[b.chr] += 1
elsif (b > 64 && b < 91)
h[(b+32).chr] += 1
end
end
h
end