zoukankan      html  css  js  c++  java
  • day22

    类中定义的函数分成两大类

    一:绑定方法(绑定给谁,谁来调用就自动将它本身当作第一个参数传入):

    1. 绑定到类的方法:用classmethod装饰器装饰的方法。
    #为类量身定制
    
    #类.boud_method(),自动将类当作第一个参数传入
    
    #(其实对象也可调用,但仍将类当作第一个参数传入)
    
    1. 绑定到对象的方法:没有被任何装饰器装饰的方法。
    #为对象量身定制
    
    #对象.boud_method(),自动将对象当作第一个参数传入
    
    #(属于类的函数,类可以调用,但是必须按照函数的规则来,没有自动传值那么一说)
    

    二:非绑定方法:用staticmethod装饰器装饰的方法

    不与类或对象绑定,类和对象都可以调用,但是没有自动传值那么一说。就是一个普通工具而已

    注意:与绑定到对象方法区分开,在类中直接定义的函数,没有被任何装饰器装饰的,都是绑定到对象的方法,可不是普通函数,对象调用该方法会自动传值,而staticmethod装饰的方法,不管谁来调用,都没有自动传值一说

    绑定方法

    绑定给对象的方法(略)

    绑定给类的方法(classmethod)

    classmehtod是给类用的,即绑定到类,类在使用时会将类本身当做参数传给类方法的第一个参数(即便是对象来调用也会将类当作第一个参数传入),python为我们内置了函数classmethod来把类中的函数定义成类方法

    #settings.py
    HOST='127.0.0.1'
    PORT=3306
    DB_PATH=r'C:UsersAdministratorPycharmProjects	est面向对象编程	est1db'
    
    #test.py
    import settings
    class MySQL:
        def __init__(self,host,port):
            self.host=host
            self.port=port
    
        @classmethod
        def from_conf(cls):
            print(cls)
            return cls(settings.HOST,settings.PORT)
    
    print(MySQL.from_conf) #<bound method MySQL.from_conf of <class '__main__.MySQL'>>
    conn=MySQL.from_conf()
    
    conn.from_conf() #对象也可以调用,但是默认传的第一个参数仍然是类
    

    非绑定方法

    在类内部用staticmethod装饰的函数即非绑定方法,就是普通函数

    statimethod不与类或对象绑定,谁都可以调用,没有自动传值效果

    import hashlib
    import time
    class MySQL:
        def __init__(self,host,port):
            self.id=self.create_id()
            self.host=host
            self.port=port
        @staticmethod
        def create_id(): #就是一个普通工具
            m=hashlib.md5(str(time.time()).encode('utf-8'))
            return m.hexdigest()
    
    
    print(MySQL.create_id) #<function MySQL.create_id at 0x0000000001E6B9D8> #查看结果为普通函数
    conn=MySQL('127.0.0.1',3306)
    print(conn.create_id) #<function MySQL.create_id at 0x00000000026FB9D8> #查看结果为普通函数
    

    classmethod与staticmethod的对比

    import settings
    class MySQL:
        def __init__(self,host,port):
            self.host=host
            self.port=port
    
        @staticmethod
        def from_conf():
            return MySQL(settings.HOST,settings.PORT)
    
        # @classmethod #哪个类来调用,就将哪个类当做第一个参数传入
        # def from_conf(cls):
        #     return cls(settings.HOST,settings.PORT)
    
        def __str__(self):
            return '就不告诉你'
    
    class Mariadb(MySQL):
        def __str__(self):
            return '<%s:%s>' %(self.host,self.port)
    
    
    m=Mariadb.from_conf()
    print(m) #我们的意图是想触发Mariadb.__str__,但是结果触发了MySQL.__str__的执行,打印就不告诉你:
    

    一、isinstance和issubclass

    class Foo:
        pass
    
    class Son(Foo):
        pass
    
    s = Son()
    #判断一个对象是不是这个类的对象,传两个参数(对象,类)
    print(isinstance(s,Son))
    print(isinstance(s,Foo))
    #type更精准
    print(type(s) is Son)
    print(type(s) is Foo)
    
    #判断一个类是不是另一类的子类,传两个参数(子类,父类)
    print(issubclass(Son,Foo))
    print(issubclass(Son,object))
    print(issubclass(Foo,object))
    print(issubclass(int,object))
    

    二、反射

    反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。

    python面向对象中的反射:通过字符串的形式操作对象相关的属性。python中的一切事物都是对象(都可以使用反射)

    四个可以实现反射的函数:hasattr,getattr,setattr,delattr

    下列方法适用于类和对象(一切皆对象,类本身也是一个对象)

    三、类的内置函数

    1、str__和__repr

    class Foo:
        def __init__(self,name):
            self.name = name
        def __str__(self):
            return '%s obj info in str'%self.name
        def __repr__(self):
            return 'obj info in repr'
    
    f = Foo('egon')
    # print(f)
    print('%s'%f)
    print('%r'%f)
    print(repr(f))  # f.__repr__()
    print(str(f))
    #当打印一个对象的时候,如果实现了str,打印中的返回值
    #当str没有被实现的时候,就会调用repr方法
    #但是当你用字符串格式化的时候 %s和%r会分别去调用__str__和__repr__
    #不管是在字符串格式化的时候还是在打印对象的时候,repr方法都可以作为str方法的替补
    #但反之不行
    #用于友好的表示对象。如果str和repr方法你只能实现一个:先实现repr
    

    2、del

    class Foo:
        def __del__(self):
            print('执行我啦')
    
    f = Foo()
    print(123)
    print(123)
    print(123)
    #析构方法,当对象在内存中被释放时,自动触发执行。
    #注:此方法一般无须定义,因为Python是一门高级语言,程序员在使用时无需关心内存的分配和释放,因为此工作都是交给Python解释器来执行,所以,析构函数的调用是由解释器在进行垃圾回收时自动触发执行的。
    

    3、item系列

    getitem_setitem__delitem_

    class Foo:
        def __init__(self):
            self.name = 'egon'
            self.age = 73
            
        def __getitem__(self, item):
            return self.__dict__[item]
    
        def __setitem__(self, key, value):
            # print(key,value)
            self.__dict__[key] = value
    
        def __delitem__(self, key):
            del self.__dict__[key]
    f = Foo()
    print(f['name'])
    print(f['age'])
    f['name'] = 'alex'
    # del f['name']
    print(f.name)
    f1 = Foo()
    print(f == f1)
    

    4、new

    # class A:
    #     def __init__(self):  #有一个方法在帮你创造self
    #         print('in init function')
    #         self.x = 1
    #
    #     def __new__(cls, *args, **kwargs):
    #         print('in new function')
    #         return object.__new__(A, *args, **kwargs)
    # a = A()
    # b = A()
    # c = A()
    # d = A()
    # print(a,b,c,d)
    
    #单例模式
    class Singleton:
        def __new__(cls, *args, **kw):
            if not hasattr(cls, '_instance'):
                cls._instance = object.__new__(cls, *args, **kw)
            return cls._instance
    
    one = Singleton()
    two = Singleton()
    three = Singleton()
    go = Singleton()
    print(one,two)
    
    one.name = 'alex'
    print(two.name)
    

    5、call

    class Foo:
        def __init__(self):
            pass
        def __call__(self, *args, **kwargs):
            print('__call__')
    
    obj = Foo()  # 执行 __init__
    obj()  # 执行 __call__
    Foo()()  # 执行 __init__和执行 __call__
    #构造方法的执行是由创建对象触发的,即:对象 = 类名() ;而对于 __call__ 方法的执行是由对象后加括号触发的,即:对象() 或者 类()()
    

    6、lenhash

    class Foo:
        def __len__(self):
            return len(self.__dict__)
        def __hash__(self):
            print('my hash func')
            return hash(self.name)
    f = Foo()
    print(len(f))
    f.name = 'egon'
    print(len(f))
    print(hash(f))
    

    7、eq

    class A:
        def __init__(self):
            self.a = 1
            self.b = 2
    
        def __eq__(self,obj):
            if  self.a == obj.a and self.b == obj.b:
                return True
    a = A()
    b = A()
    print(a == b)
    
    #__eq__控制着==的结果
    
  • 相关阅读:
    334 Increasing Triplet Subsequence 递增的三元子序列
    332 Reconstruct Itinerary 重建行程单
    331 Verify Preorder Serialization of a Binary Tree 验证二叉树的前序序列化
    330 Patching Array
    329 Longest Increasing Path in a Matrix 矩阵中的最长递增路径
    328 Odd Even Linked List 奇偶链表
    327 Count of Range Sum 区间和计数
    326 Power of Three 3的幂
    Java中的Class.forName
    巧用Java中Calendar工具类
  • 原文地址:https://www.cnblogs.com/xwjhyy/p/11661532.html
Copyright © 2011-2022 走看看