- 实例方法—— 方法内部需要访问实例属性
实例方法 内部可以使用 类名. 访问类属性 - 类方法 —— 方法内部 只 需要访问 类属性
- 静态方法 —— 方法内部,不需要访问 实例属性 和 类属性
提问
如果方法内部 即需要访问 实例属性,又需要访问 类属性,应该定义成什么方法?细分类的组成成员
类主要有两部分组成
- 静态字段(变量)部分
方法部分
class A:# 第一部分:静态字段(静态变量)部分name = '陈松'# 第二部分:方法部分def __init__(self):passdef func(self):pass
每个区域的详细划分
class A:name = 'zhangsan' # 静态变量__telphone = '132xxxxxxx' # 私有静态变量def __init__(self, where, status): # 特殊方法self.where = where # 对象属性self.__status = status # 私有对象属性def func(self): # 普通方法passdef __func(self): # 私有方法pass@classmethoddef class_func(cls): # 类方法# 定义类方法,至少要有一个cls参数print('类方法')pass@staticmethoddef static_func(): # 静态方法# 静态方法没有默认参数print('静态方法')pass@propertydef prop(self): # 方法当属性用pass
类的私有成员
对每一个类的成员有两种形式:
公有成员:在任何地方都可以访问
-
私有成员和公有成员的访问限制不同
静态字段(静态属性)
公有静态字段:类可以访问;类的内部可以访问;派生类中可以访问
- 私有静态字段:仅类的内部可以访问
公有静态字段访问实例
class C:name = "公有静态字段"def func(self):print(C.name)class D(C):def show(self):print(C.name)# 类访问print(C.name)# 类内部访问obj = C()obj.func()# 派生类中访问obj_son = D()obj_son.show()# 运行结果公有静态字段公有静态字段公有静态字段
私有静态字段访问实例(只能在类内部访问)
class C:__name = "私有静态字段"def func(self):print(C.__name)class D(C):def show(self):print(C.__name)obj = C()obj.func()# 运行结果私有静态字段
普通字段(对象属性)
- 公有普通字段:对象可以访问;类内部可以访问;派生类中可以访问
- 私有普通字段:仅类内部可以访问;
公有普通字段实例
class C:def __init__(self):self.foo = "公有字段"def func(self):print(self.foo) # 类内部访问class D(C):def show(self):print(self.foo) # 派生类中访问obj = C()print(obj.foo) # 通过对象访问obj.func() # 类内部访问obj_son = D();obj_son.show() # 派生类中访问# 运行结果公有字段公有字段公有字段
私有普通字段实例(只能类内部访问)
class C:def __init__(self):self.__foo = "私有字段"def func(self):print self.foo # 类内部访问class D(C):def show(self):print self.foo # 派生类中访问obj = C()print(obj.__foo) # 通过对象访问 ==> 错误obj.func() # 类内部访问 ==> 正确obj_son = D();obj_son.show() # 派生类中访问 ==> 错误
方法
- 公有方法:对象可以访问;类内部可以访问;派生类中可以访问
- 私有方法:仅类内部可以访问;
公有方法实例
class C:def __init__(self):passdef add(self):print('in C')class D(C):def show(self):print('in D')def func(self):self.show()obj = D()obj.show() # 通过对象访问obj.func() # 类内部访问obj.add() # 派生类中访问
私有对象实例
class C:def __init__(self):passdef __add(self):print('in C')class D(C):def __show(self):print('in D')def func(self):self.__show()obj = D()obj.__show() # 通过不能对象访问obj.func() # 类内部可以访问obj.__add() # 派生类中不能访问
总结
私有成员的访问只能在类内部(通过使用类内的公有方法如前几例中的func),不能在类外部或派生类中访问。
ps:非要访问私有成员的话,可以通过 对象.类_属性名(如obj._C__name),但是绝对不允许!!!
这是因为私有成员不能被直接访问的本质就是保存在内存时在前面自动加上 _类名,这样直接调用原来的私有成员名自然访问不到东西了。
class C:def __init__(self):self.__foo = "私有字段"def func(self):self.foo # 类内部访问class D(C):def show(self):self.foo # 派生类中访问obj = C()print(obj.__dir__())print(obj._C__foo)# 运行结果['_C__foo', '__module__', '__init__', 'func', '__dict__', '__weakref__', '__doc__', '__repr__', '__hash__', '__str__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__']私有字段
类的其他成员(方法)
方法包括:普通方法、静态方法和类方法,三种方法在内存中都归属于类,区别在于调用方式不同。
实例方法
定义:第一个参数必须是实例对象,该参数名一般约定为“self”,通过它来传递实例的属性和方法(也可以传
类的属性和方法);
调用:只能由实例对象调用。
类方法
定义:使用装饰器@classmethod。第一个参数必须是当前类对象,该参数名一般约定为“cls”,通过它来传
递类的属性和方法(不能传实例的属性和方法);
调用:实例对象和类对象都可以调用。
静态方法
定义:使用装饰器@staticmethod。参数随意,没有“self”和“cls”参数,但是方法体中不能使用类或实例
的任何属性和方法;
调用:实例对象和类对象都可以调用。
双下方法
定义:双下方法是特殊方法,他是解释器提供的 由双下划线加方法名加双下划线 方法名的具有特殊意
义的方法,双下方法主要是python源码程序员使用的,
我们在开发中尽量不要使用双下方法,但是深入研究双下方法,更有益于我们阅读源码。
调用:不同的双下方法有不同的触发方式,就好比盗墓时触发的机关一样,不知不觉就触发了双下方
法,例如:init
类方法
使用装饰器@classmethod。
原则上,类方法是将类本身作为对象进行操作的方法。假设有个方法,且这个方法在逻辑上采用类本身
作为对象来调用更合理,那么这个方法就可以定义为类方法。另外,如果需要继承,也可以定义为类方
法。
如下场景:
- 假设我有一个学生类和一个班级类,想要实现的功能为:
- 执行班级人数增加的操作、获得班级的总人数;
- 学生类继承自班级类,每实例化一个学生,班级人数都能增加;
- 最后,我想定义一些学生,获得班级中的总人数。
思考:这个场景使用类方法比较合适,为什么?因为我要实例化学生类,班级类的需要的功能是增加人数和获得班级总人数,实例化一个班级类是不必要的,然而从学生这一个实例获取班级总人数逻辑上也不合理。
class Student:__num = 0def __init__(self, name, age):self.name = nameself.age = ageStudent.addNum()@classmethoddef addNum(cls):cls.__num += 1@classmethoddef getNum(cls):return cls.__numStudent('陈松', 18)Student('阿松', 36)Student('松松', 73)print(Student.getNum())# 运行结果3
静态方法
使用装饰器@staticmethod。
静态方法是类中的函数,不需要实例。静态方法主要是用来存放逻辑性的代码,逻辑上属于类,但是和
类本身没有关系,也就是说在静态方法中,不会涉及到类中的属性和方法的操作。可以理解为,静态方
法是个独立的、单纯的函数,它仅仅托管于某个类的名称空间中,便于使用和维护。
譬如,我想定义一个关于时间操作的类,其中有一个获取当前时间的函数。
import timeclass TimeTest(object):def __init__(self, hour, minute, second):self.hour = hourself.minute = minuteself.second = second@staticmethoddef showTime():return time.strftime("%H:%M:%S", time.localtime())print(TimeTest.showTime())t = TimeTest(2, 10, 10)nowTime = t.showTime()print(nowTime)# 运行结果22:45:5622:45:56
方法综合案例
需求
- 设计一个 Game 类
- 属性:
定义一个 类属性 top_score 记录游戏的 历史最高分
定义一个 实例属性 player_name 记录 当前游戏的玩家姓名
- 方法:
静态方法 show_help 显示游戏帮助信息
类方法 show_top_score 显示历史最高分
实例方法 start_game 开始当前玩家的游戏
- 主程序步骤
1) 查看帮助信息
2) 查看历史最高分
3) 创建游戏对象,开始游戏
