面向对象封装特点之一就是通过实现好的方法来访问,限制对数据的不合理访问,把对象状态私有化,仅供类的内部进行操作
下方示例,Test方法的number属性类实例的时候传递1,number是一个公开属性,可以在外部任意修改、访问,没有对属性进行进行约束
class Test:
def __init__(self,number)
self.number = number
if __name__ == '__main__':
test = Test(1)
print(test.number)
test.number = -100
print(test.number)
@property的作用是声明一个只读属性,在函数上装饰,像调用属性一样的去调用这个函数,对象.属性的方式操作,内部可以做一些逻辑判断,需要注意的两点
class Test:
def __init__(self, number):
self.__number = number
@property # 声明number属性,只读
def number(self):
if self.__number >= 100:
raise ValueError('范围超出100')
return self.__number
if __name__ == '__main__':
test = Test(3)
print(test.number) # 3
setter就是给@property声明的属性设置值,func是当前函数的名字,但是当前函数setter,又需要指向property,所以property声明的函数名 和 setter的函数名需要一致,根据赋值操作还是获取操作决定走哪个方法
class Test:
def __init__(self,number):
self.__number = None
@property # 声明属性
def number(self):
if self.__number >= 100:
raise ValueError('范围超出100')
return self.__number
@number.setter # 设置属性
def number(self,number)
self.__number = number
'''
声明和设置属性的函数名字可以完全自定义,但是要与初始化变量名做区分,如果都一样的话会出现同名递归报错
'''
test = Test()
test.number = 30 # setter 设置值
print(test.number) #property声明的number获取值
deleter装饰器见名知意,就是属性的删除操作,实际开发过程中,具体实现的逻辑由我们自己实现,当del属性的时候,会触发该装饰器函数内的流程,但是是否是删除操作,根据业务场景来实现
class Test:
def __init__(self):
self.__number = None
@property
def number(self):
return self.__number
@number.setter
def number(self, number):
self.__number = number
@number.deleter
def number(self):
self.__number = 100
if __name__ == '__main__':
test = Test()
test.number = 10
print(test.number) # 10
del test.number
print(test.number) # 100
'''
该示例中,deleter的方法中将number设置为了100,
del test.number后触发该流程,所以在deleter做什么事情,由自己决定,
可以是 del self.__number 也可有是self.__number=None等等,del只是触发条件
'''
除了上面使用装饰器声明,还可有在类中设置属性指向property()对象,设置set和get的方法以及del方法
class Test:
def __init__(self):
self.__number = None
def get_number(self):
return self.__number
def set_number(self, number):
self.__number = number
def del_number(self):
del self.__number
# 设置number属性指向property对象
# fget 获取的时候触发的方法
# fset 赋值的时候触发的方法
# fdel 删除的时候触发的方法
number = property(fget=get_number, fset=set_number, fdel=del_number)
if __name__ == '__main__':
test = Test()
test.number = 100
print(test.number)
del test.number
我正在使用Sinatra(1.2)和RSpec(2.5),并希望创建一个具有属性TDD样式的新对象。最终结果应该是这样的:classUserdefinitialize(name)@name=nameendend我知道我必须在实现之前编写示例,但我想在这里解释我的问题。:)这是我目前无法使用的规范:describeUserit"createsanewuserobject"doname=mock("Aname")user=mock(User)#shouldn'tdothis,seethereply'suser.should_receive(:name=).with(name)User.ne
我们将我们的应用程序从Rails4.1.14升级到4.2.5.1并遇到了以下问题:string="SomeString"ar_model=SomeArModel.newar_model.some_attribute=string#nextlineistruefor4.1,butfailsfor4.2ar_model.some_attribute.object_id==string.object_id显然,对象setter会复制每个对象(如果我有一个数组,里面的每个对象也会被复制),我想知道,这是不是有意为之并且是某些新安全功能的一部分?更新我将ruby-2.2.2p95用于两个ra
我为你们准备了一个简单的。我想要一个特色内容部分,其中排除了当前文章所以这可以通过delete_if使用MiddlemanBlog:但是我使用的是中间人代理,所以我无法访问current_article方法...我有一个YAML结构,其中包含以下模拟数据(以及其他数据),文件夹设置如下:data>site>caseStudy>RANDOM-ID423536.yaml(由CMS生成)在每个yaml文件中,您会发现如下内容::id:2k1YccJrQsKE2siSO6o6ac:title:Heyplace我的config.rb看起来像这样data.site.caseStudy.eachdo
这个问题在这里已经有了答案:Isitpossibletohaveclass.property=xreturnsomethingotherthanx?(3个答案)关闭8年前。我想迭代一个字符串数组,并将它们中的每一个分配给类User的一个新实例,我希望我会得到一个User对象数组:classUserdefname=(name)@name=nameselfendendoriginal_array=["aaa","bbb","bbb"]result=original_array.collect{|str|User.new.name=str}但结果是一个字符串数组!putsresult.ins
我已经通过这种方式在数据库中创建了外键:classCreateUser但忘记添加on_delete::nullify。迁移已被推送并用于生产。我想添加新的迁移,这将为此PK约束添加级联删除。如何实现? 最佳答案 您可以在下次迁移时删除和添加外键:classChangeForgeinKeyOnUsersTable 关于ruby-on-rails-添加:on_deletetoalreadyexistingforeign_keyinrailsmigration,我们在StackOverflow
这个问题在这里已经有了答案:WhydoRubysettersneed"self."qualificationwithintheclass?(3个答案)关闭7年前。我一直在研究PragmaticProgrammers的“ProgrammingRuby”一书,想知道是否可以在类中调用setter方法,而不是直接分配给实例变量。classBookInStockattr_reader:isbn,:pricedefinitialize(isbn,price)@isbn=isbn@price=Float(price)enddefprice_in_centsInteger(price*100+0.5
我有一个嵌套模型设置,其中用户有多个制造商,制造商有多个生产线。当我尝试从行索引中删除该行的实例时,我收到一条错误的路由消息。我不确定我做错了什么,但这可能很简单,希望有人能在这里拯救我一些心痛。我已经阅读了hartl的资料,但我对嵌套资源和路由仍然非常陌生。===模型===classUser======classManufacturer======classLine======lines_controller.rbclassLinesController=====PrefixVerbURIPatternController#Actionmanufacturer_linesGET/ma
我正在尝试使用YouTubeAPIv3来更新和删除视频与OAuth2forauthentication通过google-api-client(0.6.4)Rubygem。但是,当我尝试执行这两个操作中的任何一个时,我看到以下错误消息:Google::APIClient::ClientError:InsufficientPermission奇怪的是:使用与update和delete完全相同的身份验证过程,我可以insert(上传)成功,没问题!所以,我不认为这是我的身份验证设置的问题,而是我代码中的其他地方。我的读写scope在所有这些操作中始终相同:https://www.google
我正在尝试添加一个名为“已启用?”的属性吗?到同时具有getter和setter的模型。但是,当我执行以下操作时:defenabled?=value#..logicgoeshere..end我收到语法错误,意外的“?”,需要“\n”或“;”我应该怎么做? 最佳答案 是的,ruby语法只允许?在方法名称中,如果它是最后一个字符,所以foo?=是无效的。您可以做的一件事是定义enabled?、enable和disable(或enable!和disable!如果你想强调它们是变异方法)。如果这不符合您的需求,您可以只将方法命名为enabl
我有一个对象,它有一个名为value的属性,它是bigdecimal类型。在类定义中我有validates_numericality_of。但是如果我:a.value='fire''fire'最终在验证触发之前进行类型转换为正确的类型,因此:a.valid?=>true如何在类型转换之前触发验证?谢谢丹 最佳答案 来自ActiveRecord::Basedocs:Sometimesyouwanttobeabletoreadtherawattributedatawithouthavingthecolumn-determinedtype