【7.0】Python面向对象之绑定方法与非绑定方法

发布时间 2023-11-30 14:25:19作者: Chimengmeng

【一】绑定方法与非绑定方法介绍

【1】绑定方法

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

(1)绑定到类的方法

  • 用classmethod装饰器装饰的方法。

  • 为类量身定制

  • 类.boud_method(),自动将类当作第一个参数传入

  • (其实对象也可调用,但仍将类当作第一个参数传入)

(2)绑定到对象的方法

  • 没有被任何装饰器装饰的方法

  • 为对象量身定制

  • 对象.boud_method(),自动将对象当作第一个参数传入

  • (属于类的函数,类可以调用,但是必须按照函数的规则来,没有自动传值那么一说)

【2】非绑定方法

  • 用staticmethod装饰器装饰的方法

  • 不与类或对象绑定,类和对象都可以调用,但是没有自动传值那么一说。

  • 就是一个普通工具而已

【3】小结

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

【二】绑定方法

【1】绑定给对象的方法(略)

【2】绑定给类的方法(classmethod)

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


class MySQL:
    def __init__(self, host, port):
        self.host = host
        self.port = port

    @classmethod
    def from_conf(cls):
        print(cls)
        return cls(HOST, PORT)


# <bound method MySQL.from_conf of <class '__main__.MySQL'>>
print(MySQL.from_conf)
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()


# <function MySQL.create_id at 0x0000000001E6B9D8> 
# #查看结果为普通函数
print(MySQL.create_id)
conn = MySQL('127.0.0.1', 3306)

# <function MySQL.create_id at 0x00000000026FB9D8> 
# #查看结果为普通函数
print(conn.create_id)

【四】classmethod与staticmethod的区别

HOST = '127.0.0.1'
PORT = 3306


class MySQL:
    def __init__(self, host, port):
        self.host = host
        self.port = port

    @staticmethod
    def from_conf_static():
        return MySQL(HOST, PORT)

    # 哪个类来调用,就将哪个类当做第一个参数传入
    @classmethod
    def from_conf_class(cls):
        return cls(HOST, PORT)

    def __str__(self):
        return '就不告诉你'


class Mariadb(MySQL):
    def __str__(self):
        return '<%s:%s>' % (self.host, self.port)


m = Mariadb.from_conf_static()

# 我们的意图是想触发Mariadb.__str__,但是结果触发了MySQL.__str__的执行
print(m)
# 就不告诉你

【五】总结

  • 若类中需要一个功能,该功能的实现代码中需要引用对象则将其定义成对象方法
  • 需要引用类则将其定义成类方法、无需引用类或对象则将其定义成静态方法。