2019-08-12_Note_Day16

一、运算符重载

1. 运算符

python中所有的类型都是类,所以所有的数据都是对象
python中使用任意的运算符都是在调用对应类中的对应方法
每一个运算符对应的方法是固定的
某种数据是否支持某个运算符操作,由这个数据类型中是否实现了对应的方法决定

2. 运算符重载

在不同的类中实现同样的运算符对应的函数
注意:类的对象默认情况下只支持:==, !=
class Student:
    def __init__(self, name, age, score=0):
        self.name = name
        self.age = age
        self.score = score

    # self -> 当前类的对象,也是运算符前的数据
    # other -> 运算符后面的数据,类型根据运算规则的设计可以是任何类型的数据
    # a + b -> a.__add__(b)
    def __add__(self, other):
        # return self.age + other.age
        # return self.score + other.score
        # return self.score + other
        return Student(self.name + other.name, self.age + other.age, self.score + other.score)

    # a * b -> a.__mul__(b)
    def __mul__(self, other: int):
        list1 = []
        for _ in range(other):
            list1.append(copy.copy(self))
        return list1

    # a < b -> a.__lt__(b)
    # 小于和大于符号只需要重载一个,另一个自动支持
    def __lt__(self, other):
        return self.score < other.score

    # return [self for _ in range(other)]
    # return Student(self.name * other, self.age * other, self.score * other)

    def __repr__(self):
        return ''.join(['<', str(self.__dict__)[1:-1], '>'])


stu1 = Student('小明', 19, 90)
stu2 = Student('小花', 20, 78)

print(stu1 + stu2)
print(stu1 * 3)

students = [stu1, stu2, Student('小红', 12, 100)]
students.sort()
# students.sort(key=lambda x: x.score)
print(students)

print(stu1 < stu2, stu1 > stu2)

二、浅拷贝和深拷贝

class Dog:
    def __init__(self, name, color):
        self.name = name
        self.color = color

    def __repr__(self):
        return '<{}>, id = {}'.format(str(self.__dict__)[1:-1], hex(id(self)))


class Person:
    def __init__(self, name, age, dog=None):
        self.name = name
        self.age = age
        self.dog = dog

    def __repr__(self):
        return '<{}>, id = {}'.format(str(self.__dict__)[1:-1], hex(id(self)))

1. 直接赋值

将变量中的地址直接赋值给新的变量,赋值后两个变量的地址相同
p1 = Person('小明', 18, Dog('大黄', '黄色'))
p2 = p1
print(p1)
print(p2)
# <'name': '小明', 'age': 18, 'dog': <'name': '大黄', 'color': '黄色'>, id = 0x101d9f8d0>, id = 0x101d9f908
# <'name': '小明', 'age': 18, 'dog': <'name': '大黄', 'color': '黄色'>, id = 0x101d9f8d0>, id = 0x101d9f908
p1.name = '小花'
print(p1.name, p2.name)  # 小花 小花
p2.dog.color = '绿色'
print(p1.dog.color)  # 绿色

2. 拷贝

无论是浅拷贝还是深拷贝都会对原数据进行复制,产生新的地址
list1 = [1, 2, 3]
list2 = copy(list1)
list3 = deepcopy(list1)
print(id(list1), id(list2), id(list3))  # 4533504520 4533323464 4533504456

list1.append(100)
print(list2, list3)  # [1, 2, 3] [1, 2, 3]

3. 浅拷贝

字符串、列表和元组的切片,对象.copy(),以及copy模块中的copy()都是浅拷贝
浅拷贝只拷贝当前对象,不会对子对象进行拷贝
p3 = copy(p1)

print(id(p1), id(p3))  # 4564163104 4564163384
print(id(p1.dog), id(p3.dog))  # 4564163048 4564163048

p1.name = 'Tom'
print(p1.name, p3.name)  # Tom 小花

p1.dog.color = '红色'
print(p1.dog.color, p3.dog.color)  # 红色 红色

4. 深拷贝

copy模块中的deepcopy方法是深拷贝
p4 = deepcopy(p1)
print(id(p1), id(p4))  # 4364675728 4364730776
print(id(p1.dog), id(p4.dog))  # 4364675672 4364730496

p1.name = 'Jack'
print(p1.name, p4.name)  # Jack Tom

p1.dog.color = '黑色'
print(p1.dog.color, p4.dog.color)  # 黑色 红色

三、枚举

枚举值的特点

1. 可以通过有意义的属性名直接显示数据

2. 每个数据的值不能修改

3. 可以做到不同数据的值唯一

from enum import Enum
from enum import unique

@unique
class PokerNum(Enum):
    J = 11
    Q = 12
    K = 13
    A = 1


print(PokerNum.J.name)  # J
print(PokerNum.K.value, PokerNum.J.value > PokerNum.Q.value)  # 13 False

四、内存管理

1. 内存的开辟

内存区间分为栈区间和堆区间
栈区间的内存自动开辟和释放,堆区间的内存需要程序员手动开辟和释放,但在python中已经将堆区间内存的开辟和释放过程自动化
当给变量赋值时,系统会先在堆区间中开辟空间将数据保存,然后再将数据在堆区的地址存到栈区间的变量中
数字和字符串数据在开辟空间时,会先检查内存中之前是否已经有当前数据,如果有则直接使用之前的数据,没有会开辟新的空间保存当前数据

2. 内存的释放

栈区间:全局栈区间在程序结束后销毁,函数栈区间在函数调用结束后销毁
堆区间:对象是否销毁是由这个对象的引用计数决定的,如果一个对象的引用计数为零,这个对象就会销毁(垃圾回收机制)
注意:python中针对对象的循环引用已经做了处理,程序员不需要针对循环引用问题写额外的代码

你可能感兴趣的:(2019-08-12_Note_Day16)