python学习笔记:第16天 面向对象02--对象中的成员
目录
⼀、类的成员介绍:
⾸先, 什么是类的成员. 很简单. 我么能在类中写什么? 写的内容就是成员. 到⽬前为⽌. 我们
已经学过了⼀些成员了.
class foo: def __init__(self, a, b): self.a = a # 这里的self.a和self.b都是成员 self.b = b def method(self): # 方法也是类的成员 pass
- 在上⾯代码中__init__和method都属于类的成员⽅法,⼜称为实例⽅法。总之这样的东⻄
需要⽤对象来访问。- 上⽅的
self.a = a
这个代码的含义是给对象设置属性信息,含义是这个对象的xxx属性是xxxx. 这种东⻄⼜被称之为成员变量或者实例变量, 再或者被称之为字段。
二、类的成员-变量
在类中变量分成两⼤类:
- 实例变量(字段)
- 类变量(静态变量)
1. 实例变量
其实之前写的都是实例变量,所谓实例变量就是实例(也就是对象)的变量,实例也是就是指self
,也是就是说,前面写的self.xxx
都是实例变量:
class foo: def __init__(self, name, age): self.name = name self.age = age f1 = foo('jerry', 18) f2 = foo('james', 29) print(f1.name, f1.age) # 每个对象都拥有自己的实例变量 print(f2.name, f2.age) # 实例变量都必须要通过实例来调用 # 结果: # jerry 18 # james 29
2. 类变量
类变量是一个类所拥有的,每个实例也可以直接访问类变量。类变量就可以看作是一个类公有的一种变量,如果对象自己没有设置这个值,那么默认就用统一的类变量。
class person: country = '中国' # 类变量 def __init__(self, name, age): # 实例变量中并没有设置country属性 self.name = name self.age = age person.country = '大清' print(person.country) # 打印结果是:大清 p1 = person('zzc', 26) print(p1.country) # 打印结果也是大清
从上面的例子可以看出如果对象中没有country属性,那么会使用类变量country,下面再看一个例子:
class person: country = '中国' def __init__(self, name, age): self.name = name self.age = age person.country = '大清' # 类变量修改后变成了‘大清’ p1 = person('zzc', 26) print(p1.country) # 打印结果是:大清 p2 = person('milkgood', 27) print(p2.country) # 打印结果也是大清
上面的例子说明,变量p1
和p2
使用的都是同一个变量country
,在手动把类变量的值修改过后,变量p1
和p2
所对应的country
也跟着改变了,所以他们是指向的同一块物理地址。下面我们把上面的程序改一下,来看⼀个和类变量息息相关的坑:
class person: country = '中国' def __init__(self, name, age): self.name = name self.age = age # person.country = '大清' # 这里先把修改类变量的语句注释了 print(person.country) # 首次打印的是类变量最开始时候的值 p1 = person('zzc', 26) # 然后实例化了一个对象p1 p1.country = '大清' # 然后修改了对象中的country的值为大清 print(p1.country) # 此时打印结果是:大清 p2 = person('milkgood', 27) # 再次实例化了一个对象p2 print(p2.country) # 打印结果还是中国 # 输出结果: # 中国 # 大清 # 中国
我们来看下面的图来分析:
好了. 来对类成员中的变量做个简单的总结:
- 实例变量:给对象⽤的.
- 类变量:多个对象共享的. 最好是⽤类名来访问. 这样更加规范
三、类的成员-方法
类中的方法分为三种:
- 成员⽅法(实例⽅法)
- 静态⽅法
- 类⽅法
1. 成员方法
成员方法也是之前写的最多的方法,可以说之前写的方法都是成员方法,也就是说对象直接访问的方法就是成员方法
class car: def run(self): print('老司机永不翻车') def jump(self): # 这中带self参数的都是实例方法,之前写的都是实方法 print('you jump i jump') c = car() c.run() c.jump() # 输出结果: # 老司机永不翻车 # you jump i jump
实例方法也能像实例变量一样赋值
def fly(): print('我是一辆会飞的车') class car: def run(self): print('老司机永不翻车') def jump(self): # 这中带self参数的都是实例方法,之前写的都是实方法 print('you jump i jump') c = car() c.fly = fly # 像实例变量一样把函数名赋值给实例的一个属性 c.fly() # 打印结果: 我是一辆会飞的车
2. 类方法
跟类变量一样,是属于类的命名空间的,可以直接通过类名来访问(需要传入一个参数cls),但是要注意,类方法需要在函数上加上一个名为classmethod的装饰器
class person: def eat(self): print('i\'m eating') @classmethod # 加上classmethod装饰器 def drink(cls): # 这个cls参数在使用类调用时,python解释器会自动传入 print(cls) print('i\'m drinking') print(person) person.drink() # 使用类名直接访问 p = person() p.drink() # 也可以实例化对象后,通过对象访问,但是不建议这么做 # 打印结果: # <class '__main__.person'> # <class '__main__.person'> # i'm drinking # <class '__main__.person'> # i'm drinking
3. 静态方法
静态方法也跟类方法一样,跟对象无关,直接使用类直接访问且无需参数,但是要加上staticmethod装饰器
class foo: def __init__(self, user): self.user = user @staticmethod # 加上staticmethod装饰器 def staticmthod(): # 这里不用传入参数 print('这是一个静态方法') foo.staticmthod() # 直接使用类访问 f = foo('s') f.staticmthod() # 也可以通过对象访问,不过不建议 # 输出结果 # 这是一个静态方法 # 这是一个静态方法
既然静态方法和类方法都是跟对象无关的,都可以通过类来访问,那么他们有什么区别呢:
- 类方法在传参的时候,需要传入一个类,而静态方法在使用时不需要传入任何参数,可以直接通过类调用
- 类方法在传参的时候接收了一个参数,
cls
可以通过cls
访问类中的变量,而静态方法则不行,静态方法需要使用类名来访问(即使在类的内部也是如此)
四、类的成员-属性
属性其实就是通过⽅法改造过来的⼀种变量的写法, 在⽅法上添加⼀个@property就可以了
class person: def __init__(self): pass @property def age(self): return 1 p = person() age = p.age print(age)
应⽤场景: 我们⼀般保存数据的时候, 不会保存⼀个⼈的年龄. 因为随着时间的推移. 每个⼈的年龄都时刻在改变着. 那如何保存更加完美呢? 很简单. 保存出⽣年⽉⽇. 然后⽤程序来计算,你当前的年龄. 实时的. 那这个时候就需要进⾏相应的计算了. ⽽计算属于⼀个功能. 当然要写⽅法⾥了. 但是对于年龄这个属性⽽⾔. 他应该是⼀个数值. ⽽不是动作. 所以python就提供了这样⼀种机制. 通过⽅法来描述⼀个属性.
注意:
- ⽅法参数只能有⼀个self
- ⽅法上⽅要写@property
- 调⽤的时候, 我们不需要写括号. 直接当成属性变量来⽤就可以了.
- 这种套路只能取值. 不能设置值
五、私有属性
在python的类中使用__xx
(以双下划线开头,不能以双下划线结尾)来定义一个私有属性,定义的私有属性一般(除去使用特殊的方法)是从外面获取不到的。
class foo: __c = 30 # 私有类变量 def __init__(self): self.a = 10 self.__b = 20 # 私有的实例变量 def __func(self): # 私有方法也跟私有变量一样 print('私有方法') f = foo() print(f.a) # 下面的都会报错 print(f.__b) # attributeerror: 'foo' object has no attribute '__b' print(foo.__c) # attributeerror: type object 'foo' has no attribute '__c' f.__func() # attributeerror: 'foo' object has no attribute '__func'
下一篇: 腾讯趟自媒体的浑水,邀约码是柄双刃剑