Python基础知识-内存分析实例对象和类对象创建过程(重要)

以下面代码为例,分析整个创建过程:

class Student:
    company = '尚学堂'  #类属性
    count = 0    #类属性

    def __init__(self,name,score):
        self.name = name   #实例属性
        self.score = score
        Student.count = Student.count + 1

    def say_score(self):   #实例方法
        print("我的公司是:", Student.company)
        print(self.name,'的分数是:', self.score)
s1 = Student('高琪',80)  #s1是
实例对象,自动调用__init__()方法
s1.say_score()
print('一共创建{0}个对象'.format(Student.count))

1、调用类是,先创建一个Student类的类对象,类属性和方法会被加载到类对象中;
2、调用构造器 __ init __()把对象构造起来,对象中包括实例属性和方法
3、两个实例属性传入,方法从类对象中获得
4、如果多一个s2则重复2-3的过程,即多一个 __ init __()把对象构造起来的过程,类对象还是只有一个。

__ del __方法(析构函数)和垃圾回收机制

       __ del __方法称为“析构方法”,用于实现对象被销毁时所需的操作。比如:释放对象占用的资源,例如:打开的文件资源、网络连接等。
       Python实现自动的垃圾回收,当对象没有被引用时(引用计数为0),由垃圾回收器调用 __ del __方法。
       我们也可以通过del语句删除对象,从而保证调用 __ del __方法。
       系统会自动提供 __ del __方法,一般不需要自定义析构方法。

__ cal __方法和可调用对象

       定义了__ call __方法的对象,称为“可调用对象”,即该对象可以像函数一样被调用。

class SalaryAccount:
    """工资计算类"""
    def __call__(self,salary):
        yearSalary = salary*12
        daySalary = salary//30
        hourSalary = daySalary//8

        return dict(monthSalary=salary,yearSalary=yearSalary,daySalary=daySalary,hourSalary=hourSalary)
s = SalaryAccount()
print(s(5000))

方法没有重载

       Python中,方法的参数没有声明类型(调用时确定参数的类型),参数的数量也可以由可变参数控制。因此,Python中是没有方法的重载的。定义一个方法即可有多种调用方式,相当于实现了其他语言中的方法的重载。
       如果我们在类体中定义了多个重名的方法,只有最后一个方法有效。
       建议:不要使用重名的方法!Python中方法没有重载。

class Person:
    def say_hi(self):
        print("hello")
    def say_hi(self,name):
        print("{0},hello".format(name))
p1 = Person()
p1.say_hi('高琪')

方法的动态性

       Python是动态语言,我们可以动态的为类添加新的方法,或者动态的修改类的已有的方法。

#测试方法的动态性
class Person:
    def work(self):
        print("努力上班!")
def play_game(self):
    print("{0}玩游戏".format(self))
def work2(s):
    print("好好工作,努力上班!")
Person.play = play_game
Person.work = work2
p = Person()
p.play()
p.work()
执行结果:
<__main__.Person object at 0x0000000002F634A8>玩游戏
好好工作,努力上班!

       我们可以看到,Person动态的新增了play_game方法,以及用work2替换了work方法。

私有属性和私有方法(实现封装)

       Python对于类的成员没有严格的访问控制限制,这与其他面向对象语言有区别。关于私有属性和私有方法,有如下要点:
       1、通常我们约定,两个下划线开头的属性是私有的(private)。其他为公共的(public)。
       2、类内部可以访问私有属性(方法)。
3、类外部不能直接访问私有属性(方法)。
4、类外部可以通过“类名_私有属性(方法)名”访问私有属性(方法)。
【注】方法本质上也是属性!只不过是可以通过()执行而已。所以,此处讲的私有属性和共有属性,也同时讲解了私有方法和公有方法的用法。
【测试】私有属性和公有属性使用测试

class Employee:
    __company = "百战程序员" #私有类属性, 通过dir可以查到

    def __init__(self,name,age):
        self.name = name
        self.__age = age     #私有实例属性

    def say_company(self):
        print("我的公司是:", Employee.__company)   #类内部可以直接访问私有属性
        print(self.name,'的年龄是:',self.__age)
        self.__work()
    def __work(self):  #私有实例方法,通过dir可以查到
        print("工作!好好工作,好好挣钱,去个媳妇!")
p1 = Employee("高琪",32)
print(p1.name)
print(dir(p1))
p1.say_company()
print(p1._Employee__age)  #通过这种方式可以直接访问到私有属性。通过dir可以查到属性:_Employee__age
执行结果:
高琪
['_Employee__age', '_Employee__company', '_Employee__work', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'say_company']
我的公司是: 百战程序员
高琪 的年龄是: 32
工作!好好工作,好好挣钱,去个媳妇!
32

@property装饰器

       @property可以将一个方法的调用方式变成“属性调用”。

#简单测试@property
class Employee:
    @property
    def salary(self):
        return 30000;

emp1 = Employee()
print(emp1.salary)
print(type(emp1.salary))
执行结果:
30000
<class 'int'>
#简单测试@property
class Employee:
    def __init__(self,name,salary):
        self.name = name
        self.__salary = salary
    @property
    def salary(self): #相当于salary属性的getter方法
        print('月薪为{0},年薪为{1}'.format(self.__salary,(12*self.__salary)))
        return self.__salary;
    @salary.setter
    def salary(self,salary): #相当于salary属性的setter方法
        if (0<salary<1000000):
            self.__salary = salary
        else:
            print('薪水录入错误!只能在0~100000之间')

emp1 = Employee('高琪',100)
print(emp1.salary)
print(type(emp1.salary))
emp1.salary = -200
执行结果:
月薪为100,年薪为1200
100
月薪为100,年薪为1200
<class 'int'>
薪水录入错误!只能在0~100000之间
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容