我正在寻找的不仅仅是简单的类型列表 that is found on this page:
:primary_key, :string, :text, :integer, :float, :decimal, :datetime, :timestamp, :time, :date, :binary, :boolean
但是是否有任何文档实际定义了这些字段?
具体来说:
:string 和 :text 有什么区别?
在 :float 和 :decimal 之间?
:time、:timestamp 和:datetime 的区别是什么?
这些类型的细微差别是否记录在任何地方?
编辑:DB 平台实现的要点与我要问的问题无关。如果说,:datetime
在 Rails 文档中没有定义的预期含义,那么 db-适配器作者在选择相应的列类型时会经过吗?
:string
和 :text
之外的其他 事物,但我找不到除此之外的任何东西。所以,我只是想知道以供将来参考。
根据个人经验制定的指南:
细绳:
限制为 255 个字符(取决于 DBMS)
用于短文本字段(姓名、电子邮件等)
文本:
无限长度(取决于 DBMS)
用于评论、博客文章等。一般经验法则:如果它是通过 textarea 捕获的,请使用 Text。对于使用文本字段的输入,请使用字符串。
整数:
整数
漂浮:
以浮点精度存储的十进制数
精度是固定的,这对于某些计算可能会产生问题;由于舍入不准确,通常对数学运算没有好处。
十进制:
十进制数存储的精度根据您的计算需要而变化;将这些用于需要准确的数学
有关浮点数和小数之间差异的示例和深入解释,请参见这篇文章。
布尔值:
用于存储真/假属性(即只有两种状态的事物,如开/关)
二进制:
用于将图像、电影和其他文件以其原始原始格式存储在称为 blob 的数据块中
:首要的关键
这个数据类型是一个占位符,Rails 将其转换为您选择的数据库所需的任何主键数据类型(即 postgreSQL 中的串行主键)。它的使用有些复杂,不推荐使用。
使用模型和迁移约束(例如带有 :unique => true 选项的 validates_uniqueness_of 和 add_index )来模拟您自己的字段之一的主键功能。
日期:
仅存储日期(年、月、日)
时间:
仅存储时间(小时、分钟、秒)
约会时间:
存储日期和时间
时间戳
存储日期和时间
注意:对于 Rails 而言,Timestamp 和 DateTime 的含义相同(使用任一类型来存储日期和时间)。对于 TL;DR 说明为什么两者都存在,请阅读底部段落。
这些是经常存在混淆的类型;我希望这有帮助。我真的不知道为什么没有关于这些的官方文档。另外,我想您提到的这些数据库适配器是由编写 Rails 的同一个人编写的,因此他们在编写适配器时可能不需要任何文档。希望这可以帮助!
注意:据我所知,Rails 包含 :DateTime
和 :Timestamp
,主要是为了与数据库系统兼容。例如,MySQL 的 TIMESTAMP
数据类型存储为 unix 时间戳。它的有效范围从 1970 年到 2038 年,时间存储为自上次 epoch 以来经过的秒数,这应该是标准的,但实际上可能因系统而异。 MySQL 后来引入了 DATETIME
数据类型,它以秒为单位存储(自 5.6.4 起可选小数秒),因为“1000-01-01 00:00:00”,以 size increase 为代价。保留 TIMESTAMP
数据类型是为了向后兼容。其他数据库系统也经历了类似的演变。 Rails 认识到存在多种标准,并为这两种标准提供了接口。但是,Rails ActiveRecord 将 :Timestamp
和 :DateTime
都默认为存储在 MySql 的 DATETIME
中的 UTC 日期,因此它对 Rails 程序员没有功能上的差异。这些存在以便希望区分两者的用户可以这样做。 (有关更深入的解释,请参阅this SO 答案)。
从 Rails 主分支源代码中我发现:
#activerecord/lib/active_record/connection_adapters/abstract_mysql_adapter.rb
NATIVE_DATABASE_TYPES = {
primary_key: "bigint auto_increment PRIMARY KEY",
string: { name: "varchar", limit: 255 },
text: { name: "text", limit: 65535 },
integer: { name: "int", limit: 4 },
float: { name: "float" },
decimal: { name: "decimal" },
datetime: { name: "datetime" },
timestamp: { name: "timestamp" },
time: { name: "time" },
date: { name: "date" },
binary: { name: "blob", limit: 65535 },
boolean: { name: "tinyint", limit: 1 },
json: { name: "json" },
}
# Maps logical Rails types to MySQL-specific data types.
def type_to_sql(type, limit = nil, precision = nil, scale = nil, unsigned = nil)
sql = case type.to_s
when 'integer'
integer_to_sql(limit)
when 'text'
text_to_sql(limit)
when 'blob'
binary_to_sql(limit)
when 'binary'
if (0..0xfff) === limit
"varbinary(#{limit})"
else
binary_to_sql(limit)
end
else
super(type, limit, precision, scale)
end
sql << ' unsigned' if unsigned && type != :primary_key
sql
end
# and integer ...
def integer_to_sql(limit) # :nodoc:
case limit
when 1; 'tinyint'
when 2; 'smallint'
when 3; 'mediumint'
when nil, 4; 'int'
when 5..8; 'bigint'
else raise(ActiveRecordError, "No integer type has byte size #{limit}")
end
end
# and text ..
def text_to_sql(limit) # :nodoc:
case limit
when 0..0xff; 'tinytext'
when nil, 0x100..0xffff; 'text'
when 0x10000..0xffffff; 'mediumtext'
when 0x1000000..0xffffffff; 'longtext'
else raise(ActiveRecordError, "No text type has byte length #{limit}")
end
end
# and binary ...
def binary_to_sql(limit) # :nodoc:
case limit
when 0..0xff; "tinyblob"
when nil, 0x100..0xffff; "blob"
when 0x10000..0xffffff; "mediumblob"
when 0x1000000..0xffffffff; "longblob"
else raise(ActiveRecordError, "No binary type has byte length #{limit}")
end
end
type_to_sql
中的 super
方法
#activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb
def type_to_sql(type, limit = nil, precision = nil, scale = nil) #:nodoc:
type = type.to_sym if type
if native = native_database_types[type]
column_type_sql = (native.is_a?(Hash) ? native[:name] : native).dup
if type == :decimal # ignore limit, use precision and scale
scale ||= native[:scale]
if precision ||= native[:precision]
if scale
column_type_sql << "(#{precision},#{scale})"
else
column_type_sql << "(#{precision})"
end
elsif scale
raise ArgumentError, "Error adding decimal column: precision cannot be empty if scale is specified"
end
elsif [:datetime, :time].include?(type) && precision ||= native[:precision]
if (0..6) === precision
column_type_sql << "(#{precision})"
else
raise(ActiveRecordError, "No #{native[:name]} type has precision of #{precision}. The allowed range of precision is from 0 to 6")
end
elsif (type != :primary_key) && (limit ||= native.is_a?(Hash) && native[:limit])
column_type_sql << "(#{limit})"
end
column_type_sql
else
type.to_s
end
end
ActiveRecord::Base.connection.native_database_types