面向对象编程2

实例方法

实例方法指的是只有实例对象才可以调用的方法
在类对象中定义实例方法时,第一个形参表示调用该方法的实例对象,其对应的实参由系统自动传入。第一个形参的名称通常是self,也可以是其他名称。

实例方法只能被实例对象所调用,有两种调用方式:
1.在类对象的内部(其它实例方法中)
		语法格式:self.方法名([实参])
2.在类对象的外部
		语法格式:实例对象.方法名([实参])
		类对象的所有实例对象都有一个指向类对象的指针,所以,类对象的所有实例对象都可以调用类对象中定义的实例方法  

调用实例方法时,系统自动将调用该实例方法的实例对象作为实参传递给第一个形参。第一个形参会传递给第二个形参,第二个会传给第三个,依次类推
class MyClass(object):
    #在类对象中定义实例方法
    def im1(self, p1, p2):
        print(p1,p2)
        
    #在类对象中定义实例方法
    def im2(self):
        #在类对象内部(其它实例方法中)调用实例方法
        self.im1(1, 2)
mc = MyClass()

#在类对象的外部调用实例方法
mc.im1(1, 2)   #1  2 
mc.im2()       #1  2
Python是动态语言,所以,在实例对象或类对象创建之后,可以对其动态地绑定实例方法。  
同一个类对象的不同实例对象所绑定的实例方法是相互独立的。也就是说,给一个实例对象绑定的实例方法,对于另一个实例对象是不起作用的。  
为了能让一个类对象的所有实例对象都能调用某个实例方法,可以给类对象绑定该实例方法。
#定义一个函数作为实例方法
def do_sth(self):
    print("do_sth()被调用了")
    
#导入标准库types中的类MethodType
from types import MethodType
#给实例对象mc动态地绑定实例方法
mc.do_sth = MethodType(do_sth, mc)
#调用绑定的实例方法
mc.do_sth()   #do_sth()被调用了
mc2 = MyClass()
#mc2.do_sth()  #Error

#再定义一个函数作为实例方法
def im3(self):
    print("im3()被调用了")
    
#给类对象动态地绑定实例方法    
MyClass.im3 = im3

#类对象的所有实例对象都能调用给类对象动态绑定的实例方法
mc.im3()  #im3()被调用了
mc2.im3()  #im3被调用了

函数装饰器

对于某个函数,如果我们希望在不改变该函数代码的前提下,为该函数增加额外的功能,那么就可以使用装饰器来装饰函数
装饰器是一个函数,装饰器接收一个函数作为参数(传入的实参是被装饰的函数)
装饰器的内部嵌套定义另一个函数,内函数中会引用装饰器的参数,并且装饰器的返回值是内函数
为了让内函数接收任意类型的参数,将内函数的形参定义为(*args, **kwargs)
在函数中,首先完成为被装饰函数添加的新功能,然后调用被装饰的函数。
把装饰器应用到被装饰函数的语法为:在被装饰函数的前面添加“@装饰器的函数名”

def notice(func):
    def wrapper(*args, **kwargs):
        print("公告:")
        print(".........")
        return func(*args, **kwargs)
    return wrapper

@notice
def add(a,b):
    return a+b

print(add(3,4))

>>>公告:
   .........
   7

类方法

类方法指的是类对象中使用装饰器@classmethod进行装饰的方法
在类对象中定义类方法时,必须使用装饰器@classmethod进行装饰,此外,第一个形参表示类对象,其对应的实参由系统自动传入。第一个形参的名称通常是cls,也可以是其它名称。
类方法可以被类对象所调用,语法格式为:类对象.方法名([实参]) 或:cls.方法名([实参])。
类方法也可以被实例对象所调用,语法格式为:实例对象.方法名([实参]) 或:self.方法名([实参])。
类对象的所有实例对象都有一个指向类对象的指针,所以,类对象的所有实例对象都可以调用类对象中定义的类方法。
调用类方法时,系统自动将类对象作为实参传递给第一个形参。第一个实参会传递给第二个,依次类推。

class MyClass(object):
    #在类对象中定义类方法
    @classmethod
    def class_func(cls, a, b):
        print(a,b)
        
#通过类对象调用类方法
MyClass.class_func('Hi','everybody!!')  #Hi everybody!!

#通过实例对象调用类方法
mc = MyClass()
mc.class_func(1,2)  #1 2
class MyClass(object):
    #在类对象中定义类方法
    @classmethod
    def class_func_1(cls, a, b):
        print(a, b)
        
    #在类对象内部调用类方法
    @classmethod
    def class_func_2(cls):
        MyClass.class_func_1(1,2)
        #or
        cls.class_func_1(1,2)
        
    #通过实例对象调用类方法
    def ins_func(self):
        self.class_func_1(1,2)
        
mc = MyClass()
mc.class_func_2()
>>1 2
  1 2
  
mc.ins_func()
>>1 2

MyClass.class_func_2()
>>1 2
  1 2

MyClass.ins_func()
>>TypeError: ins_func() missing 1 required positional argument: 'self'

静态方法

类对象的静态方法只是一个普通函数。把某个普通函数归属于类对象,可能只是为了易于代码管理。
在类对象中定义静态方法时,必须使用装饰器@staticmethod进行装饰
静态方法只是一个普通函数,因此,第一个形参没有特殊含义和要求。
静态方法可以被类对象所调用,语法格式为:类对象.方法名([实参]) 或: cls.方法名([实参])。
静态方法也可以被实例对象所调用,语法格式为:实例对象.方法名([实参]) 或: self.方法名([实参])。
调用静态方法时的参数传递与调用普通函数是一样的。

class MyClass(object):
    @staticmethod
    def sm(p1, p2):
        print(p1, p2)
    
MyClass.sm(1, 2)  #1 2

mc = MyClass()
mc.sm(1,2)  #1 2

访问控制

访问控制指的是:控制类对象的属性和方法在类对象的外部是否可以直接访问
如果在类对象的某个属性或方法前添加两个下划线__,那么在类对象的外部就不能直接访问该属性或方法了。

class MyClass(object):
    def __init__(self):
        self.__pia = 18
        
    def __pim(self):
        print("__pim()被调用了")
        
mc = MyClass()
print(mc.__pia)
>>AttributeError: 'MyClass' object has no attribute '__pia'

mc.__pim()
>>AttributeError: 'MyClass' object has no attribute '__pim'
class MyClass(object):
    def __init__(self):
        self.__pia = 18
        
    def __pim(self):
        print("__pim()被调用了")
        
    def do_sth(self):
        print(self.__pia)
        self.__pim()
        
mc = MyClass()
mc.do_sth()
>>18
  __pim()被调用了
之所以不能在类对象的外部直接访问以双下划开头的属性或方法,是因为:Python解释器对外把属性或方法__xxx改成了另外一个名字:_类名__xxx.所以,在类对象的外部仍然可以通过_类名__xxx访问属性或方法__xxx.但是,强烈建议不要这样访问,因为不同版本的Python解释器可能会把属性或方法__xxx改成不同的名字。
print(mc._MyClass__pia)
>>18

mc._MyClass__pim()
>>__pim()被调用了

#调用内置函数dir()获得指定对象所有可以访问的属性和方法
print(dir(mc))
>>['_MyClass__pia', '_MyClass__pim', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'do_sth']
仍然可以在类对象的外部动态绑定名为__xxx的属性或方法,这与类对象内部名为__xxx的属性或方法是不同的。
mc.__pia = "Hi"
print(mc.__pia)
>>Hi

print(dir(mc))
>>['_MyClass__pia', '_MyClass__pim', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__pia', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'do_sth']
除了在类对象的属性或方法前添加两个下划线__,还可以在类对象的属性或方法前添加单下划线_,这表示:虽然可以在类对象的外部访问该属性或方法,但最好不要访问。
class SomeClass(object):
    def __init__(self):
        self._pia = 18
        
    def _pim(self):
        print("__pim()被调用了")
        
sc = SomeClass()
print(sc._pia)
>>18

sc._pim()
>>__pim()被调用了
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值