New Optimization idea.

J

John Carter

I have a couple of classes like so...

class Foo
def step1
@Mine = Hash.new
# Perhaps stuff things into @Mine
end

def step2
@mine.each_pair do |key,value|
# Do stuff
end
end
end

Profiling with my "new" profiler shows that I'm creating many many Hash
objects, probably way more than I need. In fact most of them are empty.

Optimization trick...


class Object
FROZEN_EMPTY_HASH = Hash.new.freeze
end

class Foo
@@hash_cache = Hash.new

def step1
@Mine = @@hash_cache
# Perhaps stuff things into @Mine
if @mine.empty?
@Mine = FROZEN_EMPTY_HASH
else
@@hash_cache = Hash.new
end
end

def step2
@mine.each_pair do |key,value|
# Do stuff
end
end
end


John Carter Phone : (64)(3) 358 6639
Tait Electronics Fax : (64)(3) 359 4632
PO Box 1645 Christchurch Email : (e-mail address removed)
New Zealand

Carter's Clarification of Murphy's Law.

"Things only ever go right so that they may go more spectacularly wrong later."

From this principle, all of life and physics may be deduced.
 
R

Robert Klemme

John said:
I have a couple of classes like so...

class Foo
def step1
@mine = Hash.new
# Perhaps stuff things into @mine
end

def step2
@mine.each_pair do |key,value|
# Do stuff
end
end
end

Profiling with my "new" profiler shows that I'm creating many many Hash
objects, probably way more than I need. In fact most of them are empty.

Optimization trick...


class Object
FROZEN_EMPTY_HASH = Hash.new.freeze
end

class Foo
@@hash_cache = Hash.new

def step1
@mine = @@hash_cache
# Perhaps stuff things into @mine
if @mine.empty?
@mine = FROZEN_EMPTY_HASH
else
@@hash_cache = Hash.new
end
end

def step2
@mine.each_pair do |key,value|
# Do stuff
end
end
end

You could as well leave the member nil and change the getter appropriately:

class Foo
def mine() @mine || FROZEN_EMPTY_HASH end
def step2() mine.each_pair ...
end

Or do the test in step2...
@mine and @mine.each_pair ...

Kind regards

robert
 
J

John Carter

You could as well leave the member nil and change the getter appropriately:

class Foo
def mine() @mine || FROZEN_EMPTY_HASH end
def step2() mine.each_pair ...
end

Or do the test in step2...
@mine and @mine.each_pair ...

I didn't want to do that since @mine.each and .has_key? was going to be
evaluated often and in many places.
ie. The optimization tweak would have been a shot gun
hack all over the place.

However, if anybody had listened to me about the Null Object pattern and
nil....

I wouldn't have had to do anything.


John Carter Phone : (64)(3) 358 6639
Tait Electronics Fax : (64)(3) 359 4632
PO Box 1645 Christchurch Email : (e-mail address removed)
New Zealand

Carter's Clarification of Murphy's Law.

"Things only ever go right so that they may go more spectacularly wrong later."

From this principle, all of life and physics may be deduced.
 
R

Ryan Davis

I have a couple of classes like so...

class Foo
def step1
@mine = Hash.new
# Perhaps stuff things into @mine
end

def step2
@mine.each_pair do |key,value|
# Do stuff
end
end
end

Profiling with my "new" profiler shows that I'm creating many many
Hash objects, probably way more than I need. In fact most of them
are empty.

Optimization trick...

Maybe it is just me, but I find your original code much easier to
read and intuitive than your second version. There is just too much
wonky-thought going on in the second version and it'll make it easy
to get it wrong (esp if this pattern spreads throughout a large code
base). I don't know what your real code is doing, but creating empty
hashes seems fine to me. They don't actually cost all that much:

% time ruby -e '1_000_000.times do Hash.new; end'

real 0m7.107s
user 0m3.432s
sys 0m0.101s

(with itunes running in the background no less)

One suggestion I could make is to recycle if you really are making
too many empty hashes. If you are running step1/2 in a loop and doing
a lot of such work, only reinstantiate @mine if you need to:

def initialize
@mine = Hash.new
end

def step1
@mine = Hash.new unless @mine.empty?
# ...
end

That is only a teeny change to your original logic and meets your
goals of creating less hashes. I can read that and grok the intent
immediately. I can't with your rewrite.
 
J

John Carter

Maybe it is just me, but I find your original code much easier to read and
intuitive than your second version. There is just too much wonky-thought
going on in the second version and it'll make it easy to get it wrong (esp if

Like any optimization, premature, it's the root of all programming evil.
In my case profiling showed up a case where I was starting to consume
significant non-GC'able memory resources which were slowing the system as
a whole.

In this case it was worth it. In general, no.



John Carter Phone : (64)(3) 358 6639
Tait Electronics Fax : (64)(3) 359 4632
PO Box 1645 Christchurch Email : (e-mail address removed)
New Zealand

Carter's Clarification of Murphy's Law.

"Things only ever go right so that they may go more spectacularly wrong later."

From this principle, all of life and physics may be deduced.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
473,774
Messages
2,569,599
Members
45,175
Latest member
Vinay Kumar_ Nevatia
Top