面向对象 之 类
一 类的成员
首先什么是类的成员,很简单,你能在类中写什么?写的内容就是成员.
class Foo: # 类名 # 方法 def __init__(self,参数1,参数2): # 属性变量 self.属性1 = 参数1 self.属性2 = 参数2 # 方法 def methond(self): pass
在上面的 代码中 __init__ 和 methond 都属于类的成员方法,又称为实例方法,总之这样的东西需要
用对象来访问,而上面的 self.属性1 = 参数1 这个代码的含义是给对象设置属性信息,含义
是这个对象的 xxx 属性是 xxx,这种东西又被称为 成员变量 或者 实例变量,再或者称为
字段,都是一个意思
也就是说,类中,是存在 实例变量 和实例方法 的,还有什么呢??我们来一个个的看
二 类的成员 ------- 变量
在类中 变量分为两大类:
1.实例变量(字段)
2.类变量(静态变量)
实例变量,就是每个实例都应该拥有的变量,比如,人的名字,人的爱好,每个人的个人信息,
都属于实例变量.
类变量: 就是这一类事物统一拥有的变量,比如,在座的各位都是中国人,那大家都拥有同一个国家
class Person: # 类变量,表示所有的该类的对象都共享这个变量 country = "中国" def __init__(self,name,age,birthday): # 实例变量(字段)表示你创建的每一个人都有这三个变量 self.name = name self.num = age self.birthday = birthday p1 = Person("哈哈哥",18,"2000年01月01日") print(p1.name) # 哈哈哥 print(p1.country) # 中国 p2 = Person("嘻嘻姐",16,"2001年05月05日") print(p2.name) # 嘻嘻姐 print(p2.country) # 中国
我们发现对象 p1 和 p2 的 name 都是对象自己 的,但是 country 是类的,大家共有同一个变量
来我们验证一下啊......
class Person: # 类变量,表示所有的该类的对象都共享这个变量 country = "中国" def __init__(self,name,age,birthday): # 实例变量(字段)表示你创建的每一个人都有这三个变量 self.name = name self.num = age self.birthday = birthday Person.country = "中华人民共和国" p1 = Person("哈哈哥",18,"2000年01月01日") print(p1.name) # 哈哈哥 print(p1.country) # 中华人民共和国 p2 = Person("嘻嘻姐",16,"2001年05月05日") print(p2.name) # 嘻嘻姐 print(p2.country) # 中华人民共和国
发现了吧,我把类 变量中的值改了,大家看到的都跟着变了,为什么呢?我们要从内存来分析
通过图 我们发现,实例变量中都 隐含着一个创建这个对象的类,通过这个类 能找到我们类中定义的
全部内容,包括 方法 和 属性的 信息等
接下来,我们看一个 和类变量 息息相关的坑
class Person: # 类变量,表示所有的该类的对象都共享这个变量 country = "中国" def __init__(self,name,age,birthday): # 实例变量(字段)表示你创建的每一个人都有这三个变量 self.name = name self.num = age self.birthday = birthday p1 = Person("哈哈哥",18,"2000年01月01日") p1.country = "中华人民共和国" print(p1.name) # 哈哈哥 print(p1.country) # 中华人民共和国 p2 = Person("嘻嘻姐",16,"2001年05月05日") print(p2.name) # 嘻嘻姐 print(p2.country) # 中国
上面是 中华人民共和国,下面是 中国,为什么会这样呢? 注意: 在执行 p1.country= "中华人民共和国" 的 时候,其实并没有 去改变类中的 country ,而是给对象 添加了一个 实例变量,并且这个 实例 变量,只有当前 p1是 存在的 ,在 p2 中是不存在的
在图中,我们能清晰的看到,我们其实并没有改变类变量,只是 p1 中添加了一个实例变量,仅此而已
通过这里,我们应该能发现,类变量,最好是用类名来访问,当然,我们通过对象名也可以访问,但只能看,
不能改变,象牙哦改变,需要用类名来改变
class Foo: count = 0 def __init__(self): Foo.count += 1 print(Foo.count) # 0 Foo() Foo() Foo() print(Foo.count) # 3 # 通过程序来记录当前类被创建了多少个对象
总结:
实例变量:给对象用的
类变量:多个对象共享的,最好是用类名来访问,这样更加规范
二 类的成员 方法
1.成员方法(实例方法)
2.静态方法
3.类方法
1 .先说第一个 成员方法,我们用的最多的就是这个,对象直接访问的方法叫成员方法
class Computer: # 实例方法(成员方法) def play(self): print("我的电脑可以玩") c = Computer() c.play() # 对象直接去调 实例方法 # 我的电脑可以玩
2.静态方法,静态方法不需要我们给方法船度 self,也就是说,当出现一个方法不需要使用到成员
变量的时候,就可以选择使用静态方法,静态方法需要我们再上面添加一个
@staticmethond
class Computer: # 实例方法(成员方法) def play(self): print("我的电脑可以玩") @staticmethod def fare(): print("我的电脑非常牛,会游泳...哈哈哈") c = Computer() c.play() # 对象直接去调 实例方法 # 我的电脑可以玩 # 静态方法 和 静态变量一样,一般都是用类名直接访问 和 调用的 Computer.fare() # 类名可以直接访问的 c.fare() # 对象也可以访问,但最好别这么干,以区分静态方法 和 实例 方法 # 我的电脑非常牛,会游泳...哈哈哈 # 我的电脑非常牛,会游泳...哈哈哈
类方法,类方法 和 静态方法差不多,只不过类方法需要在 参数列表中的第一个位置预留一个位置
通常我们给第一个位置参数起名字叫 cls,类方法在被调用的时候,也不需要传递实例对象
但是,系统会自动的 把类传递给第一个参数,类方法在编写的时候,需要在类方法上面
添加 @classmethod
class Computer: def play(self): print("我的电脑可以玩") @staticmethod def fare(): print("我的电脑非常牛,会跑....") @classmethod def cal(cls,a,b): print(cls) # <class '__main__.Computer'> return a+b print(Computer.cal(1,2)) # 3
想一下啊, 类方法/静态方法 和 实例方法 有什么区别???
三 类的成员 ---属性
属性其实就是通过方法改造过来的一种变量的写法,在方法上添加一个 @property 就可以了
class Person: def __init__(self): pass @property def age(self): return 1 p = Person() age = p.age print(age)
应用场景:我们一般保存数据的时候,不会保存一个人的年龄,因为随着时间的推移,每个人的年龄都时刻在改变着,那如何保存更加完美呢?很简单,保存出生年月日,然后用程序来计算,你当前的年龄,实时的,那这个时候就需要进行 相应的计算了,而计算属于一个功能,当然要写方法里,但是对于年龄这个属性而言,他应该是一个数值,而不是一个动作,所以 python 就提供了这样一种机制,通过方法来描述一个属性
注意:
1.方法参数只能有一个 self
2.方法上方要写 @property
3.调用的时候,我们不需要写括号,直接当成属性变量来用就可以了
4.这种套路只能取值,不能设置值
四 私有
在python 中 一般是不提倡设置和 使用私有信息的,但有些场景,我们不得不这么做,比如在一个公司,
每个人的收入情况,这种内容绝对是不能公开的
在 python 中,使用 __ 作为方法或者变量的前缀,那么这个方法或者变量就是一盒私有的
1.私有变量
class Person: def __init__(self,laopo,mimi): self.__laopo = laopo # 私有 self.__mimi = mimi p = Person("哈哈","呵呵") # 报错,对象没有__mimi 这个属性 ,为什呢?? print(p.__mimi) # AttributeError: 'Person' object has no attribute '__mimi'
程序报错,私有内容是访问不到的,但是,这个访问不到试油局限性的,比如
class Person: def __init__(self,laopo,mimi): self.__laopo = laopo # 私有 self.__mimi = mimi def gaosu(self): print("大喇叭广播了,大家都知道了") return self.__mimi # 私有的内容,在他 自己哪里,可以任意的进行使用 p = Person("哈哈","我是个惊天大秘密") mimi = p.gaosu() # 通过一个私有的方法,访问到了他的秘密 print(mimi) # 大喇叭广播了,大家都知道了 # 我是个惊天大秘密
记住,私有的内容不能直接访问,但是如果对方开辟了外界访问的通道(公共方法)那可以通过这个公共
的方法来获取到私有的内容,这样的好处是,外界,只能看,但是改不了,不单单实例变量有私有的,类变量
,静态变量 一样拥有这样的属性
class Person: def __init__(self,laopo,mimi): self.__laopo = laopo # 私有 self.__mimi = mimi def gaosu(self): print("大喇叭广播了,大家都知道了") return self.__mimi # 私有的内容,在他 自己哪里,可以任意的进行使用 p = Person("哈哈","我是个惊天大秘密") mimi = p.gaosu() # 通过一个私有的方法,访问到了他的秘密 print(mimi) print(Person.__zisi) # 还是报错,没有这个 变量 # AttributeError: type object 'Person' has no attribute '__zisi'
2.私有方法
私有方法,顾名思义,只能自己访问的方法,别人都不能随便调用的
class Person: def __init__(self): pass def __yue(self): print("我要约会!!!") def job(self): print("我要工作!!!") p = Person() # p.__yue() # 会报错的 类中没有这个方法,为什么呢??明明有的嘛 p.job()
# 因为 __yue 是一个 私有方法,只能在类中自己调用,类外面不能访问
# 然而 job 是yield成员方法,并且是一个开放的方法,在类外界可以被访问到
同样的,类中的私有方法也是相对而言的 ,我们可以通过其他方法来访问到这样的方法
class Person: def __init__(self): pass def __yue(self): print("我要约会!!!") def job(self): print("我要工作!!!") self.__yue() #在自己类中访问自己的其他方法,哪怕私有的,也是自己在调用 p = Person() p.job()
关于类方法和静态方法 和 成员方法一样,就不再赘述了
需要注意的是,对于私有的内容而言,子类是无法继承的
class Fu: __qingfu = "情妇" class Zi(Fu): pass print(Zi.__qinfu) # 报错