mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
27a1ca2bfe
Delegating to just one line method is to not be worth it. Avoiding the delegation makes `read_attribute` about 15% faster. ```ruby ActiveRecord::Schema.define do create_table :users, force: true do |t| t.string :name end end class User < ActiveRecord::Base def fast_read_attribute(attr_name, &block) name = attr_name.to_s name = self.class.attribute_aliases[name] || name name = @primary_key if name == "id" && @primary_key @attributes.fetch_value(name, &block) end end user = User.create!(name: "user name") Benchmark.ips do |x| x.report("read_attribute('id')") { user.read_attribute('id') } x.report("read_attribute('name')") { user.read_attribute('name') } x.report("fast_read_attribute('id')") { user.fast_read_attribute('id') } x.report("fast_read_attribute('name')") { user.fast_read_attribute('name') } end ``` ``` Warming up -------------------------------------- read_attribute('id') 165.744k i/100ms read_attribute('name') 162.229k i/100ms fast_read_attribute('id') 192.543k i/100ms fast_read_attribute('name') 191.209k i/100ms Calculating ------------------------------------- read_attribute('id') 1.648M (± 1.7%) i/s - 8.287M in 5.030170s read_attribute('name') 1.636M (± 3.9%) i/s - 8.274M in 5.065356s fast_read_attribute('id') 1.918M (± 1.8%) i/s - 9.627M in 5.021271s fast_read_attribute('name') 1.928M (± 0.9%) i/s - 9.752M in 5.058820s ```
146 lines
4 KiB
Ruby
146 lines
4 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require "active_model/attribute_set"
|
|
require "active_model/attribute/user_provided_default"
|
|
|
|
module ActiveModel
|
|
module Attributes #:nodoc:
|
|
extend ActiveSupport::Concern
|
|
include ActiveModel::AttributeMethods
|
|
|
|
included do
|
|
attribute_method_suffix "="
|
|
class_attribute :attribute_types, :_default_attributes, instance_accessor: false
|
|
self.attribute_types = Hash.new(Type.default_value)
|
|
self._default_attributes = AttributeSet.new({})
|
|
end
|
|
|
|
module ClassMethods
|
|
def attribute(name, type = Type::Value.new, **options)
|
|
name = name.to_s
|
|
if type.is_a?(Symbol)
|
|
type = ActiveModel::Type.lookup(type, **options.except(:default))
|
|
end
|
|
self.attribute_types = attribute_types.merge(name => type)
|
|
define_default_attribute(name, options.fetch(:default, NO_DEFAULT_PROVIDED), type)
|
|
define_attribute_method(name)
|
|
end
|
|
|
|
# Returns an array of attribute names as strings
|
|
#
|
|
# class Person
|
|
# include ActiveModel::Attributes
|
|
#
|
|
# attribute :name, :string
|
|
# attribute :age, :integer
|
|
# end
|
|
#
|
|
# Person.attribute_names
|
|
# # => ["name", "age"]
|
|
def attribute_names
|
|
attribute_types.keys
|
|
end
|
|
|
|
private
|
|
def define_method_attribute=(name, owner:)
|
|
ActiveModel::AttributeMethods::AttrNames.define_attribute_accessor_method(
|
|
owner, name, writer: true,
|
|
) do |temp_method_name, attr_name_expr|
|
|
owner <<
|
|
"def #{temp_method_name}(value)" <<
|
|
" _write_attribute(#{attr_name_expr}, value)" <<
|
|
"end"
|
|
end
|
|
end
|
|
|
|
NO_DEFAULT_PROVIDED = Object.new # :nodoc:
|
|
private_constant :NO_DEFAULT_PROVIDED
|
|
|
|
def define_default_attribute(name, value, type)
|
|
self._default_attributes = _default_attributes.deep_dup
|
|
if value == NO_DEFAULT_PROVIDED
|
|
default_attribute = _default_attributes[name].with_type(type)
|
|
else
|
|
default_attribute = Attribute::UserProvidedDefault.new(
|
|
name,
|
|
value,
|
|
type,
|
|
_default_attributes.fetch(name.to_s) { nil },
|
|
)
|
|
end
|
|
_default_attributes[name] = default_attribute
|
|
end
|
|
end
|
|
|
|
def initialize(*)
|
|
@attributes = self.class._default_attributes.deep_dup
|
|
super
|
|
end
|
|
|
|
def initialize_dup(other) # :nodoc:
|
|
@attributes = @attributes.deep_dup
|
|
super
|
|
end
|
|
|
|
# Returns a hash of all the attributes with their names as keys and the values of the attributes as values.
|
|
#
|
|
# class Person
|
|
# include ActiveModel::Attributes
|
|
#
|
|
# attribute :name, :string
|
|
# attribute :age, :integer
|
|
# end
|
|
#
|
|
# person = Person.new(name: 'Francesco', age: 22)
|
|
# person.attributes
|
|
# # => {"name"=>"Francesco", "age"=>22}
|
|
def attributes
|
|
@attributes.to_hash
|
|
end
|
|
|
|
# Returns an array of attribute names as strings
|
|
#
|
|
# class Person
|
|
# include ActiveModel::Attributes
|
|
#
|
|
# attribute :name, :string
|
|
# attribute :age, :integer
|
|
# end
|
|
#
|
|
# person = Person.new
|
|
# person.attribute_names
|
|
# # => ["name", "age"]
|
|
def attribute_names
|
|
@attributes.keys
|
|
end
|
|
|
|
def freeze
|
|
@attributes = @attributes.clone.freeze
|
|
super
|
|
end
|
|
|
|
private
|
|
def write_attribute(attr_name, value)
|
|
name = attr_name.to_s
|
|
name = self.class.attribute_aliases[name] || name
|
|
|
|
@attributes.write_from_user(name, value)
|
|
end
|
|
|
|
def _write_attribute(attr_name, value)
|
|
@attributes.write_from_user(attr_name, value)
|
|
end
|
|
alias :attribute= :_write_attribute
|
|
|
|
def read_attribute(attr_name)
|
|
name = attr_name.to_s
|
|
name = self.class.attribute_aliases[name] || name
|
|
|
|
@attributes.fetch_value(name)
|
|
end
|
|
|
|
def attribute(attr_name)
|
|
@attributes.fetch_value(attr_name)
|
|
end
|
|
end
|
|
end
|