一:私有属性和私有方法(实现封装)
Python对于类的成员没有严格的访问控制限制,这与其他面向对象语言有区别。关于私有 属性和私有方法,有如下要点:
1. 通常我们约定,两个下划线开头的属性是私有的(private)。其他为公共的(public)。
2. 类内部可以访问私有属性(方法)
3. 类外部不能直接访问私有属性(方法)
4. 类外部可以通过“_类名__私有属性(方法)名”访问私有属性(方法)
【注】方法本质上也是属性!只不过是可以通过()执行而已。所以,此处讲的私有属性和公 有属性,也同时是私有方法和公有方法的用法。
【操作】私有属性和公有属性使用测试
#测试私有属性、私有方法 class Employee: __company = 'xxx公司' #私有类属性,通过dir可以查到_Employee__company def __init__(self,name,age): self.name = name self.__age = age #私有实例属性 def __work(self): #私有实例方法,通过dir可以查到_Employee__work print('好好工作,努力赚钱') def say_company(self): print('我的公司是:',Employee.__company) #类内部可以直接访问私有属性,通过类名.属性名 print(self.name,'年龄是:',self.__age) self.__work() p = Employee('jack',18) print(p.name) print(dir(p)) p.say_company() print(p._Employee__age) #通过这种方式可以直接访问到私有属性 p._Employee__work() #通过这种方式可以直接访问到私有方法 #print(p.__age) #直接访问私有属性报错 #p.__work() #直接访问私有方法报错
运行结果:
jack
['_Employee__age', '_Employee__company', '_Employee__work', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'say_company']
我的公司是: xxx公司
jack 年龄是: 18
好好工作,努力赚钱
18
好好工作,努力赚钱
【说明】从打印的 Person 对象所有属性我们可以看出。私有属性“__age”在实际存储时是按照 “_Person__age”这个属性来存储的。这也就是为什么我们不能直接使用“__age”而可以 使用“_Person__age”的根本原因。
二:@property装饰器-get和set方法
@property 可以将一个方法的调用方式变成“属性调用”。
【操作】
#简单测试@property class Employee: @property def salary(self): return 20000 emp = Employee() print(emp.salary) #方法salary()转为了属性调用 print(type(emp.salary)) #<class 'int'> #emp.salary() #报错:TypeError: 'int' object is not callable #emp.salary = 2000 #@property修饰的属性如果没有加setter方法,则为只读属性。报错:AttributeError: can't set attribute
@property 主要用于帮助我们处理属性的读操作、写操作。对于某一个属性,我们可以直 接通过: emp.salary = 30000
如上的操作读操作、写操作。但是,这种做法不安全。比如,我需要限制薪水必须为1-100000 的数字。这时候,我们就需要通过getter、setter方法来处理。
【操作】
#测试@property class Employee: def __init__(self,name,salary): self.name = name self.__salary = salary #私有属性 @property #相当于salary属性的getter方法 def salary(self): print('月薪为:{},年薪为:{}'.format(self.__salary,12*self.__salary)) return self.__salary @salary.setter #相当于salary属性的setter方法 def salary(self,salary): if(0<salary<100000): self.__salary = salary else: print('薪水录入错误,只能在0-100000之间') emp = Employee('jack',10000) print(emp.salary) emp.salary = 0
三:总结
属性和方法命名总结:
1、_xxx:保护成员,不能用“frommodule import * ”导入,只有类对象和子类对象能访 问这些成员。
2、__xxx__:系统定义的特殊成员
3、__xxx: 类中的私有成员,只有类对象自己能访问,子类对象也不能访问。(但,在类外 部可以通过“对象名._类名__xxx”这种特殊方式访问。 Python 不存在严格意义的私有成员)
类编码风格:
1. 类名首字母大写,多个单词之间采用驼峰原则。
2. 实例名、模块名采用小写,多个单词之间采用下划线隔开。
3. 每个类,应紧跟“文档字符串”,说明这个类的作用。
4. 可以用空行组织代码,但不能滥用。在类中,使用一个空行隔开方法;模块中,使用两 个空行隔开多个类。