面向对象(2)

面向对象(2)

image

1、绑定方法与非绑定方法

# 如何查看名称空间
# 查看全局的名称空间
a = 10
b = 20
print(globals())
# 查看局部的名称空间
def func():
    a = 10
    print(locals())

func()  # {'a': 10}
# 查看类与对象的名称空间
class Main:
    pass

s1 = Main()
print(s1.__dict__)  # 查看对象的名称空间 {}
print(Main.__dict__)  # 查看类的名称空间
# 绑定方法
'''
    绑定方法又分为两种:
    1、绑定给对象的
    2、绑定给类的
'''

# 1、绑定给对象的
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def get_name(self):
        print('这是绑定给对象%s的' % self)  #


s1 = Student('jason',20)
s1.get_name()  # 这是绑定给对象<__main__.Student object at 0x00000161C0AFA438>的
'''
类在定义时,默认是把所有的方法都和对象绑定的,这也是我们为什么方法的第一个参数都是默认的self,即对象本身
,对象在调用方法时,会自动将对象本身当作第一个参数传给方法(函数)
'''

# 2、绑定给类的
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @classmethod
    def get_name(cls):
        print('这是绑定给类%s的' % cls.__name__)  #

Student.get_name()  # 这是绑定给类Student的
'''
在类定义阶段,在方法上面加上'@classmethod'装饰器,声明该方法是绑定给类的,
且第一个参数通常用cls表示,以后类在调用时,会自动将类作为第一个参数传递给该方法
'''


# 举个实例
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @classmethod
    def get_name(cls, name, age):
        obj = Student(name, age)
        return cls



s1 = Student.get_name('jason', 20)
print(s1.__dict__)  # {'name': 'jason', 'age': 20}
print(isinstance(s1, Student))  # True
'''
从上面的实例中可以看出来,调用类时是类名(),且只有实例化对象时才会触发__init__方法,
也就是说,在类调用函数里的方法时,不会触发__init__方法,只有在实例化对象时才会触发,
可以简单的理解为,实例化对象就是:对象名 = 类名()
'''

# 那么绑定给类以后的方法是否还能被对象调用?
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def fun(self):
        pass

    @classmethod
    def get_name(cls, name, age):
        obj = Student(name, age)
        return obj

    @classmethod
    def msg(cls, name):
        print('这是绑定给类的 %s' % name)
#
#
# # 用类调用方法实例化对象
s1 = Student.get_name('jason', 20)
print(isinstance(s1, Student))  # True
# # 用对象调用绑定给类的方法
s1.msg('jason')  # 这是绑定给类的
'''
类中的方法绑定给类以后,对象在调用该方法的时候依旧是可以调用的,
这是因为我们之前说的,对象在产生阶段就指向了类的名称空间,所以对象在调用产生该对象的类的绑定方法时
是不用再填入类参数的,依旧是可以调用的,类里的方法,对于他所产生的对象来说是公用的,且该对象是属于这个类的,
仅仅是类在调用时,会自动将类作为第一个参数传进去
'''
# 非绑定方法
'''
    绑定方法我们都知道分为绑定给对象和绑定给类的,且这两种绑定方法在调用的时候都会把对应的名字当作第一个参数
传入,那么非绑定方法又是怎么一回事?
    非绑定方法:从字面上可以理解为即不绑定给对象,也不绑定给类,也就是说,self或cls参数不是必须的
              非绑定犯法又叫静态方法
'''
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @classmethod
    def get_name(cls, name, age):
        obj = Student(name, age)
        return obj

    @staticmethod  # 静态方法
    def msg():
        print('这是非绑定方法' )


# 用类调用方法实例化对象
s1 = Student.get_name('jason', 20)
print(isinstance(s1, Student))  # True
# 用对象调用
s1.msg()  # 这是非绑定方法


image
image

2、如何隐藏属性

# 如何隐藏属性
# 1. 如何隐藏
# 1. 如何隐藏
'''
    1. 在类定义阶段,发生了语法上的变形_类名__属性名
    2. 隐藏对外不对内
    3. 只有在类定义阶段发生变形,其他情况都不发生变形了

为什么要隐藏: 类里面的隐藏属性, 类外部可以使用,但是目的不是让类外部使用的,类外部要是想用,在类内部开放接口进行访问
            可以达到对外部数据的严格控制
'''


class Student():
    __school = 'SH'  # _Student__school  => _类名__属性名

    def __init__(self, name, age):
        self.__name = name
        self.age = age

    def __tell_info(self):  # _Student__tell_info  => _类名__函数名
        print('name: %s, age: %s' % (self.__name, self.age))

    def get_school(self):
        return self.__school  # self._Student__school

    def set_school(self, v):
        if type(v) is not str:
            print('数据类型不合法')
            # if isinstance(v, str):
            return
        self.__school = v

image

3、property装饰器

# property装饰器
class Student():
    __school = 'SH'  # _Student__school  => _类名__属性名

    def __init__(self, name, age):
        self.__name = name
        self.age = age

    def __tell_info(self):  # _Student__tell_info  => _类名__函数名
        print('name: %s, age: %s' % (self.__name, self.age))

    @property  # 把方法伪装成属性
    def name(self):
        return "name:%s" % self.__name

    @name.setter
    def name(self, v):
        if type(v) is not str:
            print('数据类型不合法')
            # if isinstance(v, str):
            return
        self.__name = v

    @name.deleter
    def name(self):
        print('不让删')


stu = Student('ly', 18)
print(stu.get_name())

stu.name = 123
print(stu.name)

del stu.name


# 练习
class Bmi():
    def __init__(self, height, weight):
        self.height = height
        self.weight = weight

    @property
    def get_bmi(self):
        return self.weight / (self.height ** 2)


bmi = Bmi(1.8, 70)
print(bmi.get_bmi)


# 了解

class Student():
    __school = 'SH'  # _Student__school  => _类名__属性名

    def __init__(self, name, age):
        self.__name = name
        self.age = age

    def __tell_info(self):  # _Student__tell_info  => _类名__函数名
        print('name: %s, age: %s' % (self.__name, self.age))

    def get_name(self):
        return "name:%s" % self.__name

    def set_name(self, v):
        if type(v) is not str:
            print('数据类型不合法')
            # if isinstance(v, str):
            return
        self.__name = v

    def del_name(self):
        print('不让删')

    # 了解
    name = property(get_name, set_name, del_name)


stu = Student('ly', 18)
# print(stu.xxx)

stu.name = 'aaa'
print(stu.name)

image

posted @ 2021-12-03 18:11  PyLy  阅读(52)  评论(0)    收藏  举报

念两句诗

入我相思门,知我相思苦,长相思兮长相忆,短相思兮无穷极。
【唐代】李白