Python中的类和方法
简单整理一下面向对象的类和方法相关的内容。
1 一些概念
- 类(Class): 用来描述具有相同的属性和方法的对象的集合。它定义了该集合中每个对象所共有的属性和方法。对象是类的实例。
- 对象:通过类定义的数据结构实例。对象包括两个数据成员(类变量和实例变量)和方法。
- 实例化:创建一个类的实例,类的具体对象。
- 方法:类中定义的函数。
- 类变量:类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用。
- 数据成员:类变量或者实例变量用于处理类及其实例对象的相关的数据。
- 局部变量:定义在方法中的变量,只作用于当前实例的类。
- 实例变量:在类的声明中,属性是用变量来表示的,这种变量就称为实例变量,实例变量就是一个用 self 修饰的变量。
- 继承:即一个派生类(derived class)继承基类(base class)的字段和方法。继承也允许把一个派生类的对象作为一个基类对象对待。例如,有这样一个设计:一个Dog类型的对象派生自Animal类,这是模拟"是一个(is-a)"关系(例图,Dog是一个Animal)。
- 方法重写:如果从父类继承的方法不能满足子类的需求,可以对其进行改写,这个过程叫方法的覆盖(override),也称为方法的重写。
2 相关定义和使用
2.1 类定义
语法格式如下:
1 class ClassName: 2 <statement-1> 3 . 4 . 5 . 6 <statement-N>
2.2 类对象
1 class MyClass: 2 """一个简单的类实例""" 3 i = 12345 #定义类属性/类变量 4 def f(self): 5 return 'hello world' 6 7 # 实例化类 8 x = MyClass() 9 10 # 访问类的属性和方法 11 print("MyClass 类的属性 i 为:", x.i) 12 print("MyClass 类的方法 f 输出为:", x.f())
1 MyClass 类的属性 i 为: 12345 2 MyClass 类的方法 f 输出为: hello world
类有一个名为 __init__() 的特殊方法(构造方法),该方法在类实例化时会自动调用
__init__() 方法可以有参数,参数通过 __init__() 传递到类的实例化操作上,如:
1 class Complex: 2 def __init__(self, realpart, imagpart): 3 self.r = realpart 4 self.i = imagpart 5 x = Complex(3.0, -4.5) 6 print(x.r, x.i) # 输出结果:3.0 -4.5
self代表类的实例,而非类。
类的方法与普通的函数只有一个特别的区别——它们必须有一个额外的第一个参数名称, 按照惯例它的名称是 self。
1 class Test: 2 def prt(self): 3 print(self) 4 print(self.__class__) 5 6 t = Test() 7 t.prt()
2.3 类的方法
在类的内部,使用 def 关键字来定义一个方法,与一般函数定义不同,类方法必须包含参数 self, 且为第一个参数,self 代表的是类的实例。
1 #类定义 2 class people: 3 #定义基本属性 4 name = '' 5 age = 0 6 #定义私有属性,私有属性在类外部无法直接进行访问 7 __weight = 0 8 #定义构造方法 9 def __init__(self,n,a,w): 10 self.name = n 11 self.age = a 12 self.__weight = w 13 def speak(self): 14 print("%s 说: 我 %d 岁。" %(self.name,self.age)) 15 16 # 实例化类 17 p = people('runoob',10,30) 18 p.speak()
1 runoob 说: 我 10 岁。
定义一个computer类,方法,类变量,实例变量,局部变量,并进行访问。
1 #类定义computer 2 class computer: 3 def __init__(self,cpu='intel'): 4 5 #定义一个实例变量(属性) 6 self.cpu = cpu 7 8 #定义一个局部变量 9 a = 1 10 11 #定义一个类变量 12 name = 'Dell' 13 14 #这种定义实例变量的方式是错误的 15 # self.gpu = 'nv' 16 17 #定义一个方法 18 def start(self): 19 print('开机') 20 21 #定义一个局部变量 22 b = 2 23 24 #实例化 25 mycomputer = computer('amd') 26 27 #调用实例方法 28 mycomputer.start() 29 30 #访问类属性/类变量 31 print(computer.name) #通过 类名称.类属性 访问,推荐这一种 32 print(mycomputer.name) #通过 实例名称.类属性 访问 33 34 #访问实例属性 35 print(mycomputer.cpu) 36 37 #定义一个子类 38 class dellcomputer(computer): 39 def __init__(self): 40 pass
1 开机 2 Dell 3 Dell 4 amd
如果类变量和实例变量同名,则用实例调用的是实例变量,用类名调用的是类变量。
1 class A: 2 name = 'zs' 3 def __init__(self): 4 self.name = 'ls' 5 6 def __call__(self,x): 7 return self.realMethod(x) 8 9 def realMethod(self,x): 10 print(x+'实际的方法') 11 return True 12 13 a = A() 14 print(a.name)#如果类属性和实例属性同名,这种方式获得的是实例属性 15 print(A.name)#这种方式获得的是类属性
1 ls 2 zs
2.4 继承
1 class DerivedClassName(BaseClassName1): 2 <statement-1> 3 . 4 . 5 . 6 <statement-N>
BaseClassName(示例中的基类名)必须与派生类定义在一个作用域内。
基类定义在另一个模块中时这一点非常有用,语法如下:
1 class DerivedClassName(modname.BaseClassName):
类computer和类HPcomputer在不同的文件中:
1 import computer 2 3 class HPcomputer(computer.Comouter): 4 pass
例子:
1 #类定义 2 class people: 3 #定义基本属性 4 name = '' 5 age = 0 6 #定义私有属性,私有属性在类外部无法直接进行访问 7 __weight = 0 8 #定义构造方法 9 def __init__(self,n,a,w): 10 self.name = n 11 self.age = a 12 self.__weight = w 13 def speak(self): 14 print("%s 说: 我 %d 岁。" %(self.name,self.age)) 15 16 #单继承示例 17 class student(people): 18 grade = '' 19 def __init__(self,n,a,w,g): 20 #调用父类的构函 21 people.__init__(self,n,a,w) 22 self.grade = g 23 #覆写父类的方法 24 def speak(self): 25 print("%s 说: 我 %d 岁了,我在读 %d 年级"%(self.name,self.age,self.grade)) 26 27 28 29 s = student('ken',10,60,3) 30 s.speak()
1 ken 说: 我 10 岁了,我在读 3 年级
2.5 多继承
1 class DerivedClassName(Base1, Base2, Base3): 2 <statement-1> 3 . 4 . 5 . 6 <statement-N>
需要注意圆括号中父类的顺序,若是父类中有相同的方法名,而在子类使用时未指定,python从左至右搜索 即方法在子类中未找到时,从左到右查找父类中是否包含方法。
1 #类定义 2 class people: 3 #定义基本属性 4 name = '' 5 age = 0 6 #定义私有属性,私有属性在类外部无法直接进行访问 7 __weight = 0 8 #定义构造方法 9 def __init__(self,n,a,w): 10 self.name = n 11 self.age = a 12 self.__weight = w 13 def speak(self): 14 print("%s 说: 我 %d 岁。" %(self.name,self.age)) 15 16 #单继承示例 17 class student(people): 18 grade = '' 19 def __init__(self,n,a,w,g): 20 #调用父类的构函 21 people.__init__(self,n,a,w) 22 self.grade = g 23 #覆写父类的方法 24 def speak(self): 25 print("%s 说: 我 %d 岁了,我在读 %d 年级"%(self.name,self.age,self.grade)) 26 27 #另一个类,多重继承之前的准备 28 class speaker(): 29 topic = '' 30 name = '' 31 def __init__(self,n,t): 32 self.name = n 33 self.topic = t 34 def speak(self): 35 print("我叫 %s,我是一个演说家,我演讲的主题是 %s"%(self.name,self.topic)) 36 37 #多重继承 38 class sample(speaker,student): 39 a ='' 40 def __init__(self,n,a,w,g,t): 41 student.__init__(self,n,a,w,g) 42 speaker.__init__(self,n,t) 43 44 test = sample("Tim",25,80,4,"Python") 45 test.speak() #方法名同,默认调用的是在括号中排前地父类的方法
2.6 方法重写
1 class Parent: # 定义父类 2 def myMethod(self): 3 print ('调用父类方法') 4 5 class Child(Parent): # 定义子类 6 def myMethod(self): 7 print ('调用子类方法') 8 9 c = Child() # 子类实例 10 c.myMethod() # 子类调用重写方法 11 super(Child,c).myMethod() #用子类对象调用父类已被覆盖的方法
1 调用子类方法 2 调用父类方法
super() 函数是用于调用父类(超类)的一个方法。
2.7类属性与方法
类的私有属性
__private_attrs:两个下划线开头,声明该属性为私有,不能在类的外部被使用或直接访问。在类内部的方法中使用时 self.__private_attrs。
类的方法
在类的内部,使用 def 关键字来定义一个方法,与一般函数定义不同,类方法必须包含参数 self,且为第一个参数,self 代表的是类的实例。
self 的名字并不是规定死的,也可以使用 this,但是最好还是按照约定是用 self。
类的私有方法
__private_method:两个下划线开头,声明该方法为私有方法,只能在类的内部调用 ,不能在类的外部调用。self.__private_methods。
实例
类的私有属性实例如下:
1 class JustCounter: 2 __secretCount = 0 # 私有变量 3 publicCount = 0 # 公开变量 4 5 def count(self): 6 self.__secretCount += 1 7 self.publicCount += 1 8 print (self.__secretCount) 9 10 counter = JustCounter() 11 counter.count() 12 counter.count() 13 print (counter.publicCount) 14 print (counter.__secretCount) # 报错,实例不能访问私有变量
1 1 2 2 3 2 4 Traceback (most recent call last): 5 File "test.py", line 16, in <module> 6 print (counter.__secretCount) # 报错,实例不能访问私有变量 7 AttributeError: 'JustCounter' object has no attribute '__secretCount'
类的私有方法实例如下:
1 class Site: 2 def __init__(self, name, url): 3 self.name = name # public 4 self.__url = url # private 5 6 def who(self): 7 print('name : ', self.name) 8 print('url : ', self.__url) 9 10 def __foo(self): # 私有方法 11 print('这是私有方法') 12 13 def foo(self): # 公共方法 14 print('这是公共方法') 15 self.__foo() 16 17 x = Site('菜鸟教程', 'www.runoob.com') 18 x.who() # 正常输出 19 x.foo() # 正常输出 20 x.__foo() # 报错
1 name : 菜鸟教程 2 url : www.runoob.com 3 这是公共方法 4 这是私有方法 5 Traceback (most recent call last): 6 File "d:\python_code\testclass.py", line 66, in <module> 7 x.__foo() # 报错 8 AttributeError: 'Site' object has no attribute '__foo'
类的专有方法:
- __init__ : 构造函数,在生成对象时调用
- __del__ : 析构函数,释放对象时使用
- __repr__ : 打印,转换
- __setitem__ : 按照索引赋值
- __getitem__: 按照索引获取值
- __len__: 获得长度
- __cmp__: 比较运算
- __call__: 函数调用
- __add__: 加运算
- __sub__: 减运算
- __mul__: 乘运算
- __truediv__: 除运算
- __mod__: 求余运算
- __pow__: 乘方
注:主要参考:Python3 面向对象 | 菜鸟教程 (runoob.com) ,加了一些个人的总结。