面向对象编程2(装饰器、静态方法,访问控制)

本文介绍了Python面向对象编程中的装饰器、静态方法和访问控制。装饰器用于在不修改函数代码的情况下为其添加额外功能。类方法和静态方法是两种特殊类型的方法,类方法通过@classmethod装饰,可被类或实例调用,而静态方法通过@staticmethod装饰,不绑定实例。访问控制包括私有属性和方法,通过__双下划线__标记,外部不能直接访问,但可通过_类名__方式间接访问,而单下划线_表示不推荐外部直接访问。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

实例方法

实例方法指的是只有实例对象才可以调用的方法。
在类对象中定义实例方法是,第一个形参表示调用该方法的实例对象,其对应的实参由系统自动输入。第一个形参的名称通常为self,当然,也可以有其他名称,但self是大家默认的,用它作为形参名称更容易让人读懂你的程序。
实例方法只能被实例对象所调用,有两种调用方法:
1、在类对象内部(其他实例方法中):

➢ 语法格式为:self.方法名([实参])

2、在类对象外部

➢ 语法格式:实例对象.方法名([实参])

类对象的所有实例对象都有一个指向类对象的指针,所以,类对象的所有实例对象都可以调用类对象中定义的实例方法。

调用实例方法时,系统自动将调用该实例方法的实例对象作为实参传递给第一个形参。第一个实参传递給第二个形参,第二个实参会传递給第三个形参,依次类推。

class My_Class(object):
    #### 在类对象中定义实例方法
    def eg1(self, m, n):
        print(m, n)
    #### 在类对象中定义实例方法
    def eg2(self):
        ### 在类对象的内部(其他实例方法中) 调用实例方法
        self.eg1(3,2)

my= My_Class()
print(my.eg1(1,2))
print(my.eg2())
##############            输出              #################
1 2
3 2
My_Class.eg2()
##############            输出              #################
TypeError: eg2() missing 1 required positional argument: 'self'

我们知道,Python是动态语言,所以,在实例对象或类对象创建之后,可以对其动态地绑定实例方法。

但是同一类对象的不同实例对象所绑定的实例方法是相互独立的,也就是说,给一个实例对象绑定的实例方法,对于另一个实例对象是不起作用的。

### 定义一个函数作为实例方法
def do_one(self):
    print("do_one这个方法被调用了")

#### 导入标准库 types 中的类 MethodType
from types import MethodType
#### 給实例对象 my 动态地绑定实例方法
my.do_one = MethodType(do_one, my)

print(my.do_one())

###############   输出   ###############
do_one这个方法被调用了
my2 = My_Class()
my2.do_one()

#########        输出       ########
AttributeError: 'My_Class' object has no attribute 'do_one'

为了能让一个类对象的所有实例对象都能调用某个实例方法,可以给类对象绑定该实例方法。

def eg(self):
    print("eg()这个方法被调用了!")

### 給实例对象动态地绑定实例方法,这里不特指某一个实例对象
My_Class.eg = eg

### 类对象的所用实例对象都能调用给类对象动态绑定的实例方法

print(my.eg())
print(my2.eg())

###################     输出     ###################
eg()这个方法被调用了!
eg()这个方法被调用了!

函数装饰器

装饰器是用来装饰函数的,对于某个函数,如果我们希望在不改变该函数代码的前提下,为该函数增加额外的功能,那么就可以使用装饰器来装饰该函数。

装饰器是一个函数,它接收一个函数作为参数(传入旳实参是被装饰的函数)

装饰器的内部嵌套定义另一个函数,内函数中会引用装饰器的参数

并且装饰器的返回值是内函数
在这里插入图片描述
为了让内函数接收任意类型的参数,将内函数的形参定义为(args,* kwargs)

在函数中,首先完成为被装饰函数添加的新功能,然后调用被装饰的函数

把装饰器应用到被装饰函数的语法为:在被装饰的函数的前面添加“@装饰器的函数名”

def notice(func):
    def wrapper(*args, **kwargs):
        print("公告:请注意身体,加强锻炼!")
        print("锻炼身体可以提高免疫力")
        return func(*args, **kwargs)
    return wrapper
@notice   ### 表示 add = notice(add)
def add(a,b):
    return a + b
print(2,7,add(2,3)) ###在函数中,首先完成为被装饰函数添加的新功能,然后调用被装饰的函数

###############              输出              ##################
公告:请注意身体,加强锻炼!
锻炼身体可以提高免疫力
2 7 5

类方法

类方法指的是类对象中使用@classmethod进行装饰的方法.

在类对象中定义类方法时,必须使用装饰器@classmethod进行装饰,此外,第一次形参表示类对象,其对应的实参有系统自动传入。第一个形参的名称通常是 cls ,也可以是其他名称,同self。

类方法可以被类对象所调用,语法格式为:类对象.方法名([实参]) 或 cls.方法名([实参])。

类方法也可以被实例对象所调用,语法格式为: 实例对象.方法名([实参]) 或 self.方法名([实参])。
类对象的所有实例对象都有一个指向类对象的指针,所以,类对象的所有实例对象都可以调用类对象中定义的类方法。

调用类方法时,系统自动将类对象作为实参传递給第一个形参。第一个是参会传递給第二个形参,第二个是参会传递給第三个形参,依次类推。

class My_Class1(object):
    ### 在类对象中定义类方法
    @classmethod
    def class_func(cls, a, b):
        print(a, b)
        
##### 通过类对象调用类方法
My_Class1.class_func("Hi","friends!")  ##### 有输出

#### 通过实例对象调用类方法
my = My_Class1()
my.class_func(2,5)   #####  有输出

##############            输出             ###############
Hi friends!
2 5
class My_Class2(object):
    ### 在类对象中定义类方法
    @classmethod
    def class_func_1(cls, a, b):
        print(a, b)
    ### 在类对象内部调用类方法
    @classmethod
    def class_func_2(cls):
        My_Class2.class_func_1(5,6)  ### 等价于 cls.class_func_1(5,6)
        
    ### 通过实例对象调用类方法
    def fac_func(self):
        self.class_func_1(8,9)
my = My_Class2()
my.class_func_2()
my.fac_func()
#############     输出     ############
5 6
8 9
My_Class2.class_func_2()
 ###########           输出           ############
 5 6
My_Class2.fac_func()

#########         输出        #########
TypeError: fac_func() missing 1 required positional argument: 'self'

静态方法

类对象的静态方法只是一个普通函数。把某个普通函数归属于类对象,可能只是为了易于代码管理。

在类对象中定义静态方法时,必须使用装饰器@staticmethod进行装饰。

静态方法只是一个普通函数,因此,第一个形参没有特殊含义和要求。

静态方法可以被类对象所调用,语法格式为:类对象,方法名([实参]) 或: cls.方法名([实参])。
静态方法也可以被实例对象所调用,语法格式为:实例对象,方法名([实参]) 或: self.方法名([实参])。
调用静态方法时的参数传递与调用普通函数是一样的。

class My_Class3(object):
    @staticmethod
    def some(s1, s2):
        print(s1, s2)

My_Class3.some(2,3)    

#######                 输出                  #########
2 3
my = My_Class3()
my.some(3,5)
########                  输出                  ############
3 5

访问控制

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

class My_Class(object):
    def __init__(self):
        self.__eg1 = 23
    def __egs(self):
        print("__egs()这个方法已经被调用了!")
my = My_Class()
print(my.__eg1)
###########                   输出                  ##############
AttributeError: 'My_Class' object has no attribute '__eg1'
my.__egs()
###########                   输出                  ############## 

AttributeError: 'My_Class' object has no attribute '__egs'
class My_Class(object):
    def __init__(self):
        self.__eg1 = 23
    def __egs(self):
        print("__egs()这个方法已经被调用了!")
    def do_some(self):
        print(self.__eg1)
        self.__egs()

my = My_Class()
print(my.do_some())
###########                   输出                  ############## 
23
__egs()这个方法已经被调用了!
None

之所以不能在类对象的外部直接访问以双下划线开头的属性或方法,是因为: Python解释器对外把属性或方法__xxx改成了另外一个名字:_类名__xxx。所以,在类对象的外部仍然可以通过_类名__xxx访问属性或方法_xxx。但是,强烈建议不要这样访问,因为不同版本的Python解释器可能会把属性或方法__xxx改成不同的名字。

print(my._My_Class__eg1)
23
print(my._My_Class__egs())
__egs()这个方法已经被调用了!
None

调用内置函数dir()可获得指定对象所有可以访问的属性和方法

但是可以在类对象的外部动态绑定名为__xxx的属性或方法,这与类对象内部名为__xxx的属性或方法是不同的。

my.__eg1 = "I can"
print(my.__eg1)
print(My_Class.__eg1)
############                      输出                    #############
I can
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-24-ad99973d3898> in <module>()
      2 my1 = My_Class()
      3 print(my.__eg1)
----> 4 print(my1.__eg1)

AttributeError: 'My_Class' object has no attribute '__eg1'

除了在类对象的属性或方法前添加两个下划线__,还可以在类对象的属性或方法前添加单下划线_,这表示:虽然可以在类对象的外部访问该属性或方法,但是最好不要访问。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值