python基础 python类的成员和装饰器

python基础 python类的成员和装饰器[python基础教程]

Python和c++一样,可以定义类,可以继承,类中又包含了类变量、实例变量(私有变量和公有变量)、方法(包括静态方法staticmethod、类方法classmethod和实例方法instancemethod)。这里只着重介绍类的成员。

通过语言描述会比较费事,下面通过例子来说明

# coding: utf-8
class MyClass:
    """I simple example class"""
    val1 = "Value 1"            #类变量
    val4 = 1
    def __init__(self):
        self.val2 = "Value 2"   #公有实例变量
        self.__val3 = "Value 3" #私有实例变量
 
    def __func():
        print "val1 : ", MyClass.val1
        print "static method cannot access val2"
        print "static method cannot access __val3"
        print "val4 : ", MyClass.val4
        MyClass.val4 = ((MyClass.val4 + 1))
 
    smd = staticmethod(__func)
 
    def __func2(cls):
        print "val1 : ", cls.val1
        print "class method cannot access val2"
        print "class method cannot access __val3"
        print "val4 : ", cls.val4
        cls.val4 = ((cls.val4 + 1))
 
    cmd = classmethod(__func2)
 
    def func3(self):
        print "val1 : ", self.val1
        print "val2 : ", self.val2
        print "instance method cannot access __val3"
        print "val4 : ", self.val4
        self.val4 = ((self.val4 + 1))

这个类中已经基本包含了上面提到的类中的各种成员,然后通过调用看下这些成员有什么不同

print "--------------------MyClass.smd()-------------------"
MyClass.smd()        #类调用静态方法
print "--------------------MyClass.cmd()-------------------"
MyClass.cmd()        #类调用类方法
#MyClass.func3()        #类无法直接调用实例方法
 
x = MyClass()
print "--------------------x.smd()-------------------"
x.smd()            #实例调用静态方法
print "--------------------x.cmd()-------------------"
x.cmd()            #实例调用类方法
print "--------------------x.func3()-------------------"
x.func3()        #实例调用实例方法
#结果
--------------------MyClass.smd()-------------------
val1 :  Value 1
static method cannot access val2
static method cannot access __val3
val4 :  1
--------------------MyClass.cmd()-------------------
val1 :  Value 1
class method cannot access val2
class method cannot access __val3
val4 :  2
--------------------x.smd()-------------------
val1 :  Value 1
static method cannot access val2
static method cannot access __val3
val4 :  3
--------------------x.cmd()-------------------
val1 :  Value 1
class method cannot access val2
class method cannot access __val3
val4 :  4
--------------------x.func3()-------------------
val1 :  Value 1
val2 :  Value 2
instance method cannot access __val3
val4 :  5

结合上面的输出结果来解释下不同变量和方法的功能。

静态方法,可以认为是一种全局方法,因为它不需要类实例化就能访问,和模块内的方法没什么区别,可以通过类和实例进行调用,它不能访问实例变量。当然,但能够通过类名访问类变量,如MyClass.val1。

类方法,类似是个全局方法,它也能如静态方法那样被类调用,也能被实例调用,不同的是它通过实例来访问类变量,有类变量cls传入,并且有子类继承时,调用该类方法时,传入的类变量cls是子类,而非父类,如x.val1。

实例方法,实例方法只能通过实例访问,它能够访问实例变量(公有)和类变量。

私有方法,无法被类和实例调用。

类变量,能够被类、类方法、实例和实例方法等访问。且在类和实例中进行传递(不停累加),如val4。

实例变量(公有),能被实例和实例方法访问,但不能被类和类方法访问。

实例变量(私有),不能被任何实例访问,但我们可以通过装饰器对其增加get/set方法来进行操作,具体在下面介绍。

私有属性,通过在变量和方法前增加__(两个下划线)来定义。

装饰器

简单的说,装饰器就是对函数的一种装饰,可以在不修改被装饰函数定义和调用的情况下,增加对被调用函数的操作或指定其属性。

语法糖

通过装饰器简化方法装饰的代码

# coding: utf-8 
def deco(func):
    def _deco():
        print("before myfunc() called.")
        func()
        print("  after myfunc() called.")
    return _deco
  
@deco
def myfunc():
    print(" myfunc() called.")
    return "ok"
  
myfunc()
myfunc()
#输出
before myfunc() called.
 myfunc() called.
  after myfunc() called.
before myfunc() called.
 myfunc() called.
  after myfunc() called.

多层装饰器

@A
@B
@C
def func():
...

可以看成是A(B(C(func))),依次执行A、B、C和func。

内置装饰器(将上面的MyClass类重新用装饰器定义一次,并增加对私有变量的操作方法)

# coding: utf-8
class MyClass:
    """I simple example class"""
    val1 = "Value 1"
    val4 = 1
    def __init__(self):
        self.val2 = "Value 2"
        self.__val3 = "Value 3"
 
    def func3(self):        #定义实例方法
        print "val1 : ", self.val1
        print "val2 : ", self.val2
        print "instance method cannot access __val3"
        print "val4 : ", self.val4
        self.val4 = ((self.val4 + 1))
 
    @classmethod            #定义类方法
    def func2(cls):
        print "val1 : ", cls.val1
        print "class method cannot access val2"
        print "class method cannot access __val3"
        print "val4 : ", cls.val4
        cls.val4 = ((cls.val4 + 1))
 
    @staticmethod            #定义静态方法
    def func():
        print "val1 : ", MyClass.val1
        print "static cannot access val2"
        print "static method cannot access __val3"
        print "val4 : ", MyClass.val4
        MyClass.val4 = ((MyClass.val4 + 1))
 
    @property            #私有实例变量get属性
    def val3(self):
        return self.__val3
 
    @val3.setter            #私有实例变量set属性
    def val3(self, value):
        self.__val3 = value
 
    @val3.deleter            #私有实例变量del属性
    def val3(self):
        del self.__val3

@classmethod和@staticmethod分别代替了之前的类方法和静态方法的声明方式,除了简洁外没有其它特殊的意义。

在2中定义的类中,有一个私有实例变量,它不能被类和实例所访问,我们通过增加@property和@setter来使实例能够对私有变量进行访问和赋值,并可以通过@deleter来将该变量删除。注意,类也能通过@property进行私有变量的访问,但无法通过@setter来给私有变量赋值,且@deleter不支持类访问。

调用如下:

print "-------------------MyClass.func()------------------"
MyClass.func()
 
 
x = MyClass()
print "-------------------x.func()------------------"
x.func()
print "-------------------x.func2()------------------"
x.func2()
print "-------------------x.func3()------------------"
x.func3()
 
print ""
print "MyClass().val3 : ",MyClass().val3        #类调用property
MyClass().val3 = "New Value"            #类调用setter
print "after "MyClass().val3 = New Value" val3 :", MyClass().val3
 
print""
print "val3 : ",x.val3                #实例调用property
x.val3 = "New Value"                #实例调用setter
print "after "x.val3 = New Value" val3 :", x.val3
del x.val3                    #实例调用deleter
print "after "del x.val3"  val3 : ", x.val3
#结果
-------------------MyClass.func()------------------
val1 :  Value 1
static cannot access val2
static method cannot access __val3
val4 :  1
-------------------x.func()------------------
val1 :  Value 1
static cannot access val2
static method cannot access __val3
val4 :  2
-------------------x.func2()------------------
val1 :  Value 1
class method cannot access val2
class method cannot access __val3
val4 :  3
-------------------x.func3()------------------
val1 :  Value 1
val2 :  Value 2
instance method cannot access __val3
val4 :  4
 
MyClass().val3 :  Value 3                #类调用property
after "MyClass().val3 = New Value" val3 : Value 3    #类调用setter没有生效
 
val3 :  Value 3                    #实例调用property
after "x.val3 = New Value" val3 : New Value    #实例调用setter
after "del x.val3"  val3 :  Value 3        #实例调用deleter

来源:PY学习网:原文地址:https://www.py.cn/article.html

hmoban主题是根据ripro二开的主题,极致后台体验,无插件,集成会员系统
自学咖网 » python基础 python类的成员和装饰器