简单整理一下面向对象的类和方法相关的内容。
1 一些概念
类(Class):用来描述具有相同的属性和方法的对象的集合。它定义了该集合中每个对象所共有的属性和方法。对象是类的实例。
对象:通过类定义的数据结构实例。对象包括两个数据成员(类变量和实例变量)和方法。
实例化:创建一个类的实例,类的具体对象。
方法:类中定义的函数。
类变量:类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用。
数据成员:类变量或者实例变量用于处理类及其实例对象的相关的数据。
局部变量:定义在方法中的变量,只作用于当前实例的类。
实例变量:在类的声明中,属性是用变量来表示的,这种变量就称为实例变量,实例变量就是一个用 self 修饰的变量。
继承:即一个派生类(derived class)继承基类(base class)的字段和方法。继承也允许把一个派生类的对象作为一个基类对象对待。例如,有这样一个设计:一个Dog类型的对象派生自Animal类,这是模拟"是一个(is-a)"关系(例图,Dog是一个Animal)。
方法重写:如果从父类继承的方法不能满足子类的需求,可以对其进行改写,这个过程叫方法的覆盖(override),也称为方法的重写。
2 相关定义和使用
2.1 类定义
语法格式如下:
1 classClassName:2
3 .4 .5 .6
2.2 类对象
1 classMyClass:2 """一个简单的类实例"""
3 i = 12345 #定义类属性/类变量
4 deff(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 classComplex:2 def __init__(self, realpart, imagpart):3 self.r =realpart4 self.i =imagpart5 x = Complex(3.0, -4.5)6 print(x.r, x.i) #输出结果:3.0 -4.5
self代表类的实例,而非类。
类的方法与普通的函数只有一个特别的区别——它们必须有一个额外的第一个参数名称, 按照惯例它的名称是 self。
1 classTest:2 defprt(self):3 print(self)4 print(self.__class__)5
6 t =Test()7 t.prt()
2.3 类的方法
在类的内部,使用def 关键字来定义一个方法,与一般函数定义不同,类方法必须包含参数 self, 且为第一个参数,self 代表的是类的实例。
1 #类定义
2 classpeople:3 #定义基本属性
4 name = ''
5 age =06 #定义私有属性,私有属性在类外部无法直接进行访问
7 __weight =08 #定义构造方法
9 def __init__(self,n,a,w):10 self.name =n11 self.age =a12 self.__weight =w13 defspeak(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 classcomputer:3 def __init__(self,cpu='intel'):4
5 #定义一个实例变量(属性)
6 self.cpu =cpu7
8 #定义一个局部变量
9 a = 1
10
11 #定义一个类变量
12 name = 'Dell'
13
14 #这种定义实例变量的方式是错误的
15 #self.gpu = 'nv'
16
17 #定义一个方法
18 defstart(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 classdellcomputer(computer):39 def __init__(self):40 pass
1 开机2 Dell3 Dell4 amd
如果类变量和实例变量同名,则用实例调用的是实例变量,用类名调用的是类变量。
1 classA:2 name = 'zs'
3 def __init__(self):4 self.name = 'ls'
5
6 def __call__(self,x):7 returnself.realMethod(x)8
9 defrealMethod(self,x):10 print(x+'实际的方法')11 returnTrue12
13 a =A()14 print(a.name)#如果类属性和实例属性同名,这种方式获得的是实例属性
15 print(A.name)#这种方式获得的是类属性
1 ls2 zs
2.4 继承
1 classDerivedClassName(BaseClassName1):2
3 .4 .5 .6
BaseClassName(示例中的基类名)必须与派生类定义在一个作用域内。
基类定义在另一个模块中时这一点非常有用,语法如下:
1 class DerivedClassName(modname.BaseClassName):
类computer和类HPcomputer在不同的文件中:
1 importcomputer2
3 classHPcomputer(computer.Comouter):4 pass
例子:
1 #类定义
2 classpeople:3 #定义基本属性
4 name = ''
5 age =06 #定义私有属性,私有属性在类外部无法直接进行访问
7 __weight =08 #定义构造方法
9 def __init__(self,n,a,w):10 self.name =n11 self.age =a12 self.__weight =w13 defspeak(self):14 print("%s 说: 我 %d 岁。" %(self.name,self.age))15
16 #单继承示例
17 classstudent(people):18 grade = ''
19 def __init__(self,n,a,w,g):20 #调用父类的构函
21 people.__init__(self,n,a,w)22 self.grade =g23 #覆写父类的方法
24 defspeak(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 classDerivedClassName(Base1, Base2, Base3):2
3 .4 .5 .6
需要注意圆括号中父类的顺序,若是父类中有相同的方法名,而在子类使用时未指定,python从左至右搜索 即方法在子类中未找到时,从左到右查找父类中是否包含方法。
1 #类定义
2 classpeople:3 #定义基本属性
4 name = ''
5 age =06 #定义私有属性,私有属性在类外部无法直接进行访问
7 __weight =08 #定义构造方法
9 def __init__(self,n,a,w):10 self.name =n11 self.age =a12 self.__weight =w13 defspeak(self):14 print("%s 说: 我 %d 岁。" %(self.name,self.age))15
16 #单继承示例
17 classstudent(people):18 grade = ''
19 def __init__(self,n,a,w,g):20 #调用父类的构函
21 people.__init__(self,n,a,w)22 self.grade =g23 #覆写父类的方法
24 defspeak(self):25 print("%s 说: 我 %d 岁了,我在读 %d 年级"%(self.name,self.age,self.grade))26
27 #另一个类,多重继承之前的准备
28 classspeaker():29 topic = ''
30 name = ''
31 def __init__(self,n,t):32 self.name =n33 self.topic =t34 defspeak(self):35 print("我叫 %s,我是一个演说家,我演讲的主题是 %s"%(self.name,self.topic))36
37 #多重继承
38 classsample(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 defmyMethod(self):3 print ('调用父类方法')4
5 class Child(Parent): #定义子类
6 defmyMethod(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 classJustCounter:2 __secretCount = 0 #私有变量
3 publicCount = 0 #公开变量
4
5 defcount(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
6 print (counter.__secretCount) #报错,实例不能访问私有变量
7 AttributeError: 'JustCounter' object has no attribute '__secretCount'
类的私有方法实例如下:
1 classSite:2 def __init__(self, name, url):3 self.name = name #public
4 self.__url = url #private
5
6 defwho(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.com3 这是公共方法4 这是私有方法5 Traceback (most recent call last):6 File "d:\python_code\testclass.py", line 66, in
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__: 乘方