面向对象是个抽象的东西,概念比较多,下面会一一介绍。
一、类和实例
类(Class)和实例(Instance)是面向对象最重要的概念。
类是指抽象出的模板。实例则是根据类创建出来的具体的“对象”,每个对象都拥有从类中继承的相同的方法,但各自的数据可能不同。
classStudent(object):passkate= Student()
关键字class后面跟着类名,类名通常是大写字母开头的单词,紧接着是(object),表示该类是从哪个类继承下来的。通常,如果没有合适的继承类,就使用object类,这是所有类最终都会继承下来的类。
Student 就是类名,kate 就是Student()的实例,类只有实例化以后才能使用。
二、构造函数,析构函数,类变量,实例变量
构造函数:__init__(self,name,age,sex), 这个方法就是构造函数,在实例化的时候自动调用。所有如果这个函数内有打印的方法,当kate实例出来的时候会打印里面的信息。
__init__方法的第一个参数永远都是self,表示创建实例本身,在__init__方法内部,可以把各种属性绑定到self,因为self指向创建的实例本身。
有了__init__方法,在创建实例的时候,就不能传入空的参数了,必须传入与__init__方法匹配的参数,但self不需要传,Python解释器自己会把实例变量传进去。
析构函数:__del__(self), 这个方法就是析构函数,是在实例被销毁时自动调用的。
类变量:country = "China" , 类变量不需要实例,可以直接使用, 如line 14
实例变量:self.name = name, self.name 这种形式就是实例变量,需要实例化后才能使用, 如 line15就会报错,需要实例化line16的kate, 才能使用name, age, sex
可以自由地给一个实例变量绑定属性,比如,给实例kate绑定一个language属性。
kate.language = "中文"
print(kate.language) #中文
和普通函数相比,在类中定义的函数只有一点不同,就是第一个参数永远是实例变量self,并且,调用时,不用传递该参数。除此之外,类的方法和普通函数没有什么区别。比如school函数只需要传city就可以了。
kate.school("北京") #凯特在北京上学
下面是具体的代码段例子。
1 classPerson(object):2 country = "China" #类变量,不需要实例化,可以直接用
3 def __init__(self,name,age,sex):#构造函数,实例化的使用自动调用
4 self.name = name #实例变量,必须实例化之后才能用,也叫成员变量
5 self.age =age6 self.sex =sex7 defsay_my_country(self):8 print(self.country)#类变量可以在类里作为属性使用
9 defschool(self,city):10 self.city =city11 print("%s在%s上学" %(self.name,self.city))12 def __del__(self):#析构函数,实例销毁的时候自动调用
13 print("实例销毁的时候自动调用")14 print(Person.country)#类变量,不需要实例化,可以直接用
15 #print(Person.name)#实例变量,必须实例化之后才能用,AttributeError: type object "Person" has no attribute "name"
16 kate = Person("凯特",18,"女")17 print(kate.name)18 print(kate.age)19 print(kate.sex)20 print(kate.country) #China
21 kate.say_my_country() #China
22 kate.language = "中文"
23 print(kate.language) #中文
24 kate.school("北京") #凯特在北京上学
三、访问限制
在Class内部,可以有属性和方法,而外部代码可以通过直接调用实例变量的方法来操作数据,这样就隐藏了内部的复杂逻辑。
1 kate = Person("凯特",18,"女")2 kate.name = "王菲"
3 print(kate.name)
如果加这句代码line2,kate.name = "王菲", 这样执行line3就会打印王菲,而不是凯特,name会修改了。
四、私有变量,私有函数,静态方法,类方法
私有变量:self.__host = host ,只能在类里使用,如果执行line31会报错 AttributeError: "MyRedis" object has no attribute "__host"
私有函数:def __conn_redis(self),只能在类里使用,执行line32会报错 AttributeError: "MyRedis" object has no attribute "__conn_redis"
静态方法:需要加 @staticmethod,不需要实例化就能直接用,其实和类没有什么关系,就是一个普通的函数,写在了类里面而已,也用不了self的那些东西,也调用不了类里的其他函数
类方法:需要加@classmethod,不需要实例化就能直接用,它比静态方法高级一点,它可以使用类变量和类方法, 如函数 class_fun(cls)
1 importredis2 classMyRedis():3 hi= "哈哈"
4 def __init__(self,host,db,password="",port=6379):5 self.__host = host #私有变量,只能在类里使用
6 self.passwd =password7 self.port =port8 self.db =db9 #self.conn_redis()
10 self.__conn_redis()11 #def conn_redis(self):
12 #self.conn = redis.Redis(host=self.__host,db=self.db,password=self.passwd,port=self.port)
13 def __conn_redis(self): #私有函数,只能在类里使用
14 self.conn = redis.Redis(host=self.__host,db=self.db,password=self.passwd,port=self.port)15 defget(self,k):16 print("__host...",self.__host)17 returnself.conn.get(k).decode()18
19 @staticmethod #静态方法
20 defother():21 print("我是other")22 @classmethod #类方法,也不需要实例化,直接就能用,它比静态方法高级一点,它可以使用类变量和类方法
23 defclass_fun(cls):24 print(cls.hi) #可以调用类变量
25 cls.class_fun2()#可以调用类方法
26 @classmethod27 defclass_fun2(cls):28 print("类方法2")29
30 r = MyRedis("localhost",3)31 #print(r.__host) #AttributeError: "MyRedis" object has no attribute "__host"
32 #r.__conn_redis() #AttributeError: "MyRedis" object has no attribute "__conn_redis"
33 print(r.get("kate11"))34 MyRedis.other() #静态方法不需要实例化
35 MyRedis.class_fun()
五、继承
在OOP程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass),而被继承的class称为基类、父类或超类(Base class、Super class)。
比如,我们已经编写了一个名为Animal的class,有一个run()方法可以直接打印一句话,然后新建一个叫Dog的类,继承了Animal类。
所以Dog可以用Animal类里的函数run(), eat(),同时自己也可以定义自己的函数。
@property ,这是个装饰器,把函数变成一个属性方法,如果这个方法没有入参的话,那就可以变成一个属性方法。
在使用的时候可以如下面代码所示的b.protect, 而不是像其他的方法一样加(), 如 b.bite()
1 classAnimal(object):2 defrun(self):3 print("running...")4 defeat(self):5 print("eating...")6
7 classDog(Animal):8 defrun(self):9 print("dog is running")10 defbite(self):11 print("dog will bite people.")12 @property13 defprotect(self):14 print("dog will protect people.")15 a =Animal()16 a.run()17 a.eat()18 b =Dog()19 b.run()20 b.eat()21 b.bite()22 b.protect #属性方法, dog will protect people.