Skip to content

Commit

Permalink
Cheaper attribute reads and respond_to?. Add underscore-prefixed meth…
Browse files Browse the repository at this point in the history
…od aliased to the attribute name so it can be overridden but still called internally.
  • Loading branch information
jeremy committed Sep 9, 2010
1 parent d79b1aa commit e0e3adf
Show file tree
Hide file tree
Showing 3 changed files with 12 additions and 7 deletions.
4 changes: 2 additions & 2 deletions activerecord/lib/active_record/attribute_methods.rb
Expand Up @@ -48,13 +48,13 @@ def method_missing(method_id, *args, &block)
end

def respond_to?(*args)
self.class.define_attribute_methods
self.class.define_attribute_methods unless self.class.attribute_methods_generated?
super
end

protected
def attribute_method?(attr_name)
attr_name == 'id' || attributes.include?(attr_name)
attr_name == 'id' || @attributes.include?(attr_name)
end
end
end
10 changes: 7 additions & 3 deletions activerecord/lib/active_record/attribute_methods/read.rb
Expand Up @@ -70,13 +70,19 @@ def define_read_method(symbol, attr_name, column)
if cache_attribute?(attr_name)
access_code = "@attributes_cache['#{attr_name}'] ||= (#{access_code})"
end
generated_attribute_methods.module_eval("def #{symbol}; #{access_code}; end", __FILE__, __LINE__)
generated_attribute_methods.module_eval("def _#{symbol}; #{access_code}; end; alias #{symbol} _#{symbol}", __FILE__, __LINE__)
end
end

# Returns the value of the attribute identified by <tt>attr_name</tt> after it has been typecast (for example,
# "2004-12-12" in a data column is cast to a date object, like Date.new(2004, 12, 12)).
def read_attribute(attr_name)
send "_#{attr_name}"
rescue NoMethodError
_read_attribute attr_name
end

def _read_attribute(attr_name)
attr_name = attr_name.to_s
attr_name = self.class.primary_key if attr_name == 'id'
if !(value = @attributes[attr_name]).nil?
Expand All @@ -89,8 +95,6 @@ def read_attribute(attr_name)
else
value
end
else
nil
end
end

Expand Down
Expand Up @@ -19,12 +19,13 @@ module ClassMethods
def define_method_attribute(attr_name)
if create_time_zone_conversion_attribute?(attr_name, columns_hash[attr_name])
method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}(reload = false)
def _#{attr_name}(reload = false)
cached = @attributes_cache['#{attr_name}']
return cached if cached && !reload
time = read_attribute('#{attr_name}')
time = _read_attribute('#{attr_name}')
@attributes_cache['#{attr_name}'] = time.acts_like?(:time) ? time.in_time_zone : time
end
alias #{attr_name} _#{attr_name}
EOV
generated_attribute_methods.module_eval(method_body, __FILE__, line)
else
Expand Down

0 comments on commit e0e3adf

Please sign in to comment.