ruby-on-rails – attr_accessor强类型的Ruby on Rails
|
只是想知道有没有人可以在Ruby on Rails上看到有关强类型的getter设置器的基础知识.我在轨道上是非常新的ruby,主要是对.NET有很好的理解.
例如,假设我们有一个名为Person的.net类 class Person
{
public string Firstname{get;set;}
public string Lastname{get;set;}
public Address HomeAddress{get;set;}
}
class Address
{
public string AddressLine1{get;set;}
public string City{get;set;}
public string Country{get;set;}
}
在Ruby中,我会把它写成 class Person attr_accessor :FirstName attr_accessor :LastName attr_accessor :HomeAddress end class Address attr_accessor :AddressLine1 attr_accessor :City attr_accessor :Country end 看看Ruby版本的Person类如何为访问者方法FirstName,LastName和HomeAddress指定类型?如果我要消费这个类,我可以将任何类型的Feed添加到HomeAddress中,但是我希望这个访问器方法只接受TYPE Address. 有什么建议么 ? 谢谢 解决方法
TL; DR:不,这是不可能的…很长的回答,是的,有可能,阅读元编程部分:)
Ruby是一种动态语言,这就是为什么你不会得到编译时类型的警告/错误,就像C#这样的语言. 与您无法为变量指定类型相同,您不能指定attr_accessor的类型. 对于来自.NET的用户来说,这可能听起来很愚蠢,但在Ruby社区,人们期望您能够编写测试.如果这样做,这些类型的问题将基本消失.在Ruby on Rails中,你应该测试你的模型.如果你这样做,你不会真的有任何麻烦,意外地分配错误的地方. 如果您正在Ruby on Rails中讨论ActiveRecord,则将String分配给数据库中定义为Integer的属性将导致抛出异常. 顺便说一句,根据惯例,你不应该使用CamelCase属性,所以正确的类定义应该是 class Person attr_accessor :first_name attr_accessor :last_name attr_accessor :home_address end class Address attr_accessor :address_line1 attr_accessor :city attr_accessor :country end 一个原因是,如果您将第一个字母大写,则Ruby将定义一个常量而不是一个变量. number = 1 # regular variable Pi = 3.14159 # constant ... changing will result in a warning,not an error 元编程黑客 顺便说一句,Ruby还具有非常庞大的元编程能力.您可以使用类型检查编写自己的attr_accessor,可以使用类似的检查 typesafe_accessor :price,Integer 有定义的东西 class Foo
# 'static',or better said 'class' method ...
def self.typesafe_accessor(name,type)
# here we dynamically define accessor methods
define_method(name) do
# unfortunately you have to add the @ here,so string interpolation comes to help
instance_variable_get("@#{name}")
end
define_method("#{name}=") do |value|
# simply check a type and raise an exception if it's not what we want
# since this type of Ruby block is a closure,we don't have to store the
# 'type' variable,it will 'remember' it's value
if value.is_a? type
instance_variable_set("@#{name}",value)
else
raise ArgumentError.new("Invalid Type")
end
end
end
# Yes we're actually calling a method here,because class definitions
# aren't different from a 'running' code. The only difference is that
# the code inside a class definition is executed in the context of the class object,# which means if we were to call 'self' here,it would return Foo
typesafe_accessor :foo,Integer
end
f = Foo.new
f.foo = 1
f.foo = "bar" # KaboOoOoOoM an exception thrown here!
或至少这些线路的东西:)这段代码工作! Ruby允许您快速定义方法,这是attr_accessor的工作原理. 也是块几乎总是关闭,这意味着我可以做如果value.is_a?类型,而不将其作为参数. 这是非常复杂的,在这里解释这是真的,何时不是.总之,有不同类型的块 > Pro,由Proc.new创建 区别之一是调用lambda中的返回值只能从lambda本身返回,但是当您从Proc执行相同的操作时,块周围的整个方法将返回,这在迭代时使用. def find(array,something)
array.each do |item|
# return will return from the whole 'find()' function
# we're also comparing 'item' to 'something',because the block passed
# to the each method is also a closure
return item if item == something
end
return nil # not necessary,but makes it more readable for explanation purposes
end
如果你是这样的东西,我建议你看看PragProg Ruby Metaprogramming screencast. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
