Class variables not classy!
(actually)
Class variables caches are not so classy
(umm, in fact)
Class variables are, well, controversial,
class variables used for caching = bad
Class variables bad?
Main Reason
class A
@@x = 2
def say
@@x
end
end
class B < A
def haha=y
@@x=y
end
end
class C < A
def haha=y
@@x=y
end
end
b = B.new
c = C.new
b.say #=> 2
c.say #=> 2
b.haha = 'because I can'
c.say #=> 'because I can'
Other reasons
- Thread safety
- Its a global variable
But still,
they are useful.
Letsbe practical
And my code has no threads !
class A
@@a_variable_shared_across_all_instances_of_A = nil
def my_method
# and it is accessible within the scope
# just like that
@a_variable_shared_across_all_instances_of_A = 'shared-data'
end
end
So whats not classy?
class Shippping
@@countries = Country.all
def label_the_parcel(address)
country = @@countries.detect{|name| name == address.country_name}
print_label country.shipping_code
end
end
So whats not classy?
class Shippping
@@california_code = State.where(country: 'U.S.A').last.shipping_code
def label_for_california(address)
print_label address
print_label @@california_code
end
end
This can fail during class load
This can fail rails s
So whats not classy?
class Shippping
def label_for_california(address)
print_label address
print_label Shipping.california_code
end
def self.california_code
@@california_code ||= State.where(country: 'U.S.A').last.shipping_code
end
end
The instance-variable cache pattern
Because why not
So whats not classy?
class A
def slowly_loaded_data
@slowly_loaded_data ||= slow_queries
end
end
The actual instance-variable cache pattern
So whats not classy?
class Shippping
def label_for_california(address)
print_label address
print_label Shipping.california_code
end
@@california_code = nil
def self.california_code
@@california_code ||= State.where(country: 'U.S.A').last.shipping_code
end
end
That, right there, is a cache
So whats not classy?
And caches have to expire sometime.
Some day, the california_code will change.
So for that day, we have -
So whats not classy?
class Shippping
@@california_code = nil
def self.california_code
@@california_code ||= State.where(country: 'U.S.A').last.shipping_code
end
def self.flush_california_code
@@california_code = nil
end
end
class State
after_save :flush1
def flush1
Shipping.flush_california_cache if name=='california'
end
end
So whats not classy?
And I have no threads
What could possibly go wrong?
So whats not classy?
drum-roll...
Enter Sidekiq
- Cache invalidation will not be carried forward
- The flush1 action will occur only in the web (Unicorn), and not in the Sidekiq
- In fact it will not occur in other unicorn workers too
- It is an in-process cache
What can be done?
- In-process caches can cause subtle bugs
- Using Rails.cache with memcache or File cache (default) is safer
- Instance variable caches are still ok
Who else does it
cd discourse
ag @@ . | wc -l #=>
cd gitlab
ag @@ . | wc -l #=>
cd spree
ag @@ . | wc -l #=>
cd rails
ag @@ . | wc -l #=>
@h6165
Thanks
Class variables not so classy
By Abhishek Yadav
Class variables not so classy
- 1,177