python-查漏补缺笔记-更新中

本文详细解释了Python中包导入时__init__.py文件的执行顺序,以及sys.modules的作用。特别关注了父包和子包的导入情况,指出仅导入子包只会记录父包信息,而不会执行其__init__.py中的所有代码。此外,还介绍了函数参数、元组、类方法、静态方法和抽象类的概念及其用法。
摘要由CSDN通过智能技术生成

包导入时__init__.py中命令的执行顺序和sys.modules变化

ref: https://edu.csdn.net/skill/practice/python-3-6/164

  1. 在有父包和子包的情况下,父包中的“ __ init__.py”语句会在子包的“ __ init__.py”语句之前执行,然后按下列顺序执行
  2. 导入子包和模块
  3. 执行“ __ init__.py”中其他代码

需要注意的是,如果只导入子包,Python解释器会将父包的信息添加到sys.modules中,这样导入子包的时候能知道父包的存在,但是除了父包的__init__.py中的语句,并不会执行其他代码,只有需要访问父包时才会进行真正的导入。

sys.modules是一个字典,记录的是已经导入的模块信息。首次导入一个模块时,python会检查这个模块是否存在于sys.modules中,如果已经存在,会直接返回模块对象;否则导入。

实例
文件结构如下图所示,father包含one、two和three三个包,one包又包含一个one包。每个 __ init__.py中包含的内容只有一条print语句,打印诸如 this is father、this is 1、this is 1.1这样的语句。
在这里插入图片描述
leetcode.py中的内容如下所示:

import sys

if __name__ == '__main__':
    for i in range(2):
        import father
        import father.one.one
        import father.two
        #print(sys.modules.keys())

        del sys.modules['father']
        del sys.modules['father.one.one']
        del sys.modules['father.two']
        print('-----------\n')

''' 输出结果
this is father
this is 1
this is 1.1
this is 2
-----------

this is father
this is 1.1
this is 2
-----------
'''

这是因为father.one在第一次循环时被加入了sys.modules,后续并没有被删除,所以再次导入时不会执行father.one的__init__.py语句,也就不会输出this is 1。

可变形参:匿名和带关键字的

Python函数的参数可以有4种:

  1. 按位置顺序指定的参数,如下面的index
  2. 带默认值的参数,如下面的default
  3. 没有名字的可选参数,如*args
  4. 有名字的可选参数,如**kw

实例

# -*- coding: UTF-8 -*-
def dump(index, default=0, *args, **kw):
    print('打印函数参数')
    print('---')
    print('index:', index)
    print('default:', default)
    for i, arg in enumerate(args):
        print(f'arg[{i}]:', arg)
    # 注意这里要用kw.items()
    for key,value in kw.items():
        print(f'keyword_argument {key}:{value}')
    print('')

if __name__=='__main__':
    dump(0)
    dump(0,2)
    dump(0,2,"Hello","World")
    dump(0,2,"Hello","World", install='Python', run='Python Program')

'''输出
打印函数参数
---
index: 0
default: 0

打印函数参数
---
index: 0
default: 2

打印函数参数
---
index: 0
default: 2
arg[0]: Hello
arg[1]: World

打印函数参数
---
index: 0
default: 2
arg[0]: Hello
arg[1]: World
keyword_argument install:Python
keyword_argument run:Python Program
'''

元组tuple

tuple1 = ('红色')
for element in tuple1:
    print(element)
'''输出
红
色
'''

这个输出是因为tuple1实际不是元组,是str,用type()可知。如果要声明为元组,应该用 tuple1 = (‘红色’,)

类方法classmethod与静态方法staticmethod

类方法可以通过类名或者类的实例调用,类方法可以访问类级别的变量,如class_variable。

使用@classmethod装饰器,第一个参数通常被命名为cls,表示类本身,不是实例。

class MyClass:
    class_variable = 10

    def __init__(self, value):
        self.instance_variable = value

    @classmethod
    def class_method(cls):
        print("This is a class method.")
        print("Class variable:", cls.class_variable)

    def instance_method(self):
        print("This is an instance method.")
        print("Instance variable:", self.instance_variable)

# 调用类方法
MyClass.class_method()

# 创建实例
obj = MyClass(20)

# 通过实例调用类方法
obj.class_method()

# 调用实例方法
obj.instance_method()

''' 输出
This is a class method.
Class variable: 10
This is a class method.
Class variable: 10
This is an instance method.
Instance variable: 20

'''

静态方法与实例和类本身无关,因此不能访问实例变量或者类变量,(类方法是可以访问类变量的),因此也不需要传递self或者cls参数。它一般用来写某些与类相关、但不依赖于类或者实例的操作

使用@staticmethod装饰器。

class MathOperations:
    @staticmethod
    def add(x, y):
        return x + y

# 调用静态方法,不需要创建类的实例
result_sum = MathOperations.add(3, 5)
print("Sum:", result_sum) # Sum: 8

抽象类

Python中使用abc模块(Abstract Base Classes)定义抽象类。实例:

from abc import ABC, abstractmethod

# 定义抽象类
class Shape(ABC):
    # 抽象方法,子类需要实现
    @abstractmethod
    def area(self):
        pass


# 具体的子类实现抽象类
class Circle(Shape):
    def __init__(self, radius):
        self.radius = radius

    def area(self):
        return 3.14 * self.radius**2

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值