• Python中if __name__ == ‘__main__‘,__init__和self 的解析


    一、  if __name__ == '__main__'

    if __name__ == '__main__'的意思是:

    当.py文件被直接运行时,if __name__ == '__main__'之下的代码块将被运行;

    当.py文件以模块形式被导入时,if __name__ == '__main__'之下的代码块不被运行。

    1.1、一个  xxx.py 文件被其他 xxx.py 文件引用

    创建一个const.py文件:

    1. PI = 3.14
    2. def main():
    3. print("PI:", PI)
    4. main()
    5. # 运行结果:PI: 3.14

    写一个用于计算圆面积的area.py文件,area.py文件需要用到const.py文件中的PI变量。从const.py中,我们把PI变量导入area.py:

    1. from const import PI
    2. def calc_round_area(radius):
    3. return PI * (radius ** 2)
    4. def main():
    5. print("round area: ", calc_round_area(2))
    6. main()
    7. '''
    8. 运行结果:
    9. PI: 3.14
    10. round area: 12.56
    11. '''

    1.2 修改const.py,添加if __name__ == '__main__'

          我们看到const.py中的main函数也被运行了,实际上我们不希望它被运行,因为const.py提供的main函数只是为了测试常量定义。这时if __name__ ==  '__main__'派上了用场,我们把const.py改一下,添加if __name__ == '__main__'

    1. PI = 3.14
    2. def main():
    3. print("PI:", PI)
    4. if __name__ == "__main__":
    5. main()
    6. 运行结果
    7. PI: 3.14

    导入const.py,运行area.py:

    1. 输出结果:
    2. round area: 12.56
    3. #不显示:PI: 3.14

    二、__init__与self

    2.1 Python中self的含义

          self,英文单词意思很明显,表示自己,本身。python的self,是个对象(Object),是当前类的实例

    2.2 Python中为何要有self

         在定义一个类时,首先需要定义一个构造函数。构造函数是用来初始化对象的属性的。在Python中,构造函数的第一个参数通常是self。self代表当前的对象实例,可以用来引用该对象的属性和方法。

         在Python中,如果要使用一个对象的属性或方法,首先需要实例化对象。实例化对象是指创建一个对象的实例,也就是根据类创建一个具体的对象。在实例化对象时,通常需要传入一些参数来初始化对象的属性。

    Python中就规定好了,函数的第一个参数,就必须是实例对象本身,并且建议,约定俗成,把其名字写为self

    2.3_init__方法的作用:

          __init__ 方法是 Python 中面向对象编程中类的特殊方法,也称为构造方法,当创建一个类的实例时,__init__ 方法会自动调用。

           它的主要作用是初始化实例的属性,实例变量是对象的属性,用于存储数据。在实例被创建后,你可以通过这些属性对实例进行操作。每个类可以定义多个不同的 __init__ 方法,但通常情况下,在类中只有一个,在这种情况下,在创建类的实例时,必须提供所需的参数。

    1. def __init__(self, 参数1, 参数2, ...):
    2. 初始化语句1
    3. 初始化语句2
    4. ...

    实例:在类中使用 init 方法初始化类的属性

    1. class Dog:
    2. def __init__(self, breed, name, age):
    3. self.breed = breed
    4. self.name = name
    5. self.age = age
    6. dog = Dog("Labrador", "Max", 3)
    7. print(dog.breed) # Output: Labrador
    8. print(dog.name) # Output: Max
    9. print(dog.age) # Output: 3

    在这个例子中,我们创建了一个名为 Dog 的类,并且定义了 init 方法。 init 方法接收三个参数:breed, name 和 age,并且通过 self. 前缀来设置类的属性。最后,我们创建了一个名为 dog 的对象,并且初始化了它的 breed, name 和 age 属性。

    2.4__init__()和self对象

    1. class Person(object):
    2. def __init__(self, name, lang, website):
    3. self.name = name
    4. self.lang = lang
    5. self.website = website
    6. print('self: ', self)
    7. print('type of self: ', type(self))
    8. '''
    9. 未实例化时,运行程序,构造方法没有运行
    10. '''
    11. p = Person('Tim', 'English', 'www.universal.com')
    12. '''实例化后运行的结果
    13. self: <__main__.Person object at 0x00000000021EAF98>
    14. type of self:
    15. '''

    可以看出self为实例变量p,是一个Person类型的对象

    1. class Dog(object):
    2. def __init__(self,name,dog_type):
    3. self.name = name
    4. self.type = dog_type
    5. def sayhi(self):
    6. print("hello,I am a dog, my name is ",self.name)
    7. d = Dog('LiChuang',"京巴") # 实例化
    8. d.sayhi()

    以下是d = Dog('LiChuang',"京巴")实例化的示意图:

     2.5 如果没有在__init__中初始化对应的实例变量的话,导致后续引用实例变量会出错

          如下代码,完整的演示了,如果没有在类Class的最初的__init__函数中,正确的初始化实例变量,则会导致后续没有变量可用,因而出现AttributeError的错误:

    1. name = 'whole global name'
    2. '''
    3. 注:此处全局的变量名,写成name,只是为了演示而用
    4. 实际上,好的编程风格,应该写成gName之类的名字,
    5. 以表示该变量是Global的变量
    6. '''
    7. class Person(object):
    8. def __init__(self, newPersonName):
    9. # self.name = newPersonName
    10. '''
    11. 如果此处不写成self.name
    12. 那么此处的name,只是__init__函数中的局部临时变量name而已
    13. 和全局中的name,没有半毛钱关系
    14. '''
    15. name = newPersonName
    16. '''
    17. 此处只是为了代码演示,而使用了局部变量name,
    18. 不过需要注意的是,此处很明显,由于接下来的代码也没有利用到此处的局部变量name
    19. 则就导致了,此处的name变量,实际上被浪费了,根本没有利用到
    20. '''
    21. def sayYourName(self):
    22. '''
    23. 此处由于找不到实例中的name变量,所以会报错:
    24. AttributeError: Person instance has no attribute 'name'
    25. '''
    26. print('My name is %s' %self.name)
    27. def selfAndInitDemo():
    28. personInstance = Person('Tim')
    29. personInstance.sayYourName()
    30. if __name__ == '__main__':
    31. selfAndInitDemo()
    32. ''' 未使用self.name时抛异常
    33. Traceback (most recent call last):
    34. File "E:/python14_workspace/s14/day06/test_1.py", line 18, in
    35. selfAndInitDemo()
    36. File "E:/python14_workspace/s14/day06/test_1.py", line 15, in selfAndInitDemo
    37. personInstance.sayYourName()
    38. File "E:/python14_workspace/s14/day06/test_1.py", line 11, in sayYourName
    39. print('My name is %s' %self.name)
    40. AttributeError: 'Person' object has no attribute 'name'
    41. '''

    从上述代码可见,由于在类的初始化(实例化)的__init__函数中,没有给self.name设置值,使得实例中,根本没有name这个变量,导致后续再去访问self.name,就会出现AttributeError的错误了。

    对应的,如果写成self.name,则意思就正确了,就是初始化的时候,给实例中新增加,并且正常设置了正确的值newPersionName了,所以后续再去通过self.name,就可以访问到,当前实例中正确的变量name了。

    相应的正确写法的代码如下:

    1. name = 'whole global name'
    2. '''
    3. 注:此处全局的变量名,写成name,只是为了演示而用
    4. 实际上,好的编程风格,应该写成gName之类的名字,
    5. 以表示该变量是Global的变量
    6. '''
    7. class Person(object):
    8. def __init__(self, newPersonName):
    9. self.name = newPersonName
    10. '''
    11. 此处正确的,通过访问self.name的形式,实现了:
    12. 1.给实例中,增加了name变量
    13. 2.并且给name赋了初值,为newPersionName
    14. '''
    15. def sayYourName(self):
    16. '''
    17. 此处由于开始正确的初始化了self对象,使得其中有了name变量,
    18. 所以此处可以正确访问了name值了
    19. '''
    20. print('My name is %s' %self.name)
    21. def selfAndInitDemo():
    22. personInstance = Person('Tim')
    23. personInstance.sayYourName()
    24. if __name__ == '__main__':
    25. selfAndInitDemo()
    26. '''My name is Tim'''
  • 相关阅读:
    oppo r11 升级8.1系统 图文教程
    elasticsearch-head连接不上es
    ASTM D4434/D4434M-2021 PVC屋面板检测
    java流知识小结
    浏览器的选择建议,按照这些建议选,总能找到合适的
    Java项目:SSM游戏点评网站
    STL源码阅读笔记-内存空间的分配和回收
    基于Java毕业设计学校图书资源交易平台源码+系统+mysql+lw文档+部署软件
    【Redis 系列】redis 学习十六,redis 字典(map) 及其核心编码结构
    LeetCode 1004.最大连续1的个数
  • 原文地址:https://blog.csdn.net/weixin_74227828/article/details/134066249