当前位置:首页>python>Python依赖注入容器的实现原理与设计模式

Python依赖注入容器的实现原理与设计模式

  • 2026-07-02 23:23:49
Python依赖注入容器的实现原理与设计模式

学习搭子,你好!在前文中咱们学习了依赖注入的基本概念和应用,但你可能好奇:那些强大的依赖注入容器到底是怎么工作的?它们如何管理复杂的依赖关系?今天咱们就深入底层,一起揭开依赖注入容器的神秘面纱!

1. 依赖注入容器的核心职责

在深入实现之前,咱们先明确一个优秀的依赖注入容器应该具备哪些核心能力:

1.1 容器的四大基本功能

  1. 1. 服务注册(Service Registration)
    • • 告诉容器:当需要某个类型时,应该创建什么对象
    • • 支持多种注册方式:类型、实例、工厂函数、单例等
  2. 2. 依赖解析(Dependency Resolution)
    • • 自动分析类型的构造函数参数
    • • 递归解析所有依赖关系
    • • 处理循环依赖等复杂情况
  3. 3. 生命周期管理(Lifecycle Management)
    • • 控制对象的创建和销毁时机
    • • 支持不同的生命周期:瞬时、单例、作用域等
  4. 4. 配置与扩展(Configuration & Extension)
    • • 支持配置文件或代码配置
    • • 提供扩展点允许自定义行为

1.2 Python依赖注入的独特挑战

Python作为一种动态语言,依赖注入容器的实现面临一些独特挑战:

# 挑战1:动态类型系统
# Python没有编译时类型检查,依赖关系在运行时才确定
defsome_function(service):  # service是什么类型?只有运行时才知道
return service.do_something()

# 挑战2:鸭子类型
# 依赖关系基于协议而非类型
classDuck:
defquack(self):
return"Quack!"

defmake_it_quack(duck_like_object):  # 不需要继承特定接口
return duck_like_object.quack()

# 挑战3:装饰器与元编程
# Python支持强大的元编程能力,容器需要与之协同
@inject
defprocess_data(service: IService = Provide[CONTAINER.service]):
# 装饰器修改了函数签名
pass

理解了容器的职责和Python的独特挑战,咱们开始一步步构建自己的依赖注入容器。

2. 从零开始:实现一个简易依赖注入容器

2.1 第一步:基础容器类

咱们从一个最简单的容器开始,只支持最基本的类型注册和解析:

# simple_container.py
import inspect
from typing importDictTypeAnyCallableUnion

classSimpleContainer:
"""简易依赖注入容器"""

def__init__(self):
# 注册表:类型 -> 实现
self._registry: Dict[TypeUnion[TypeAnyCallable]] = {}
# 单例缓存
self._singletons: Dict[TypeAny] = {}

defregister(self, interface: Type, implementation=None):
"""注册服务"""
if implementation isNone:
# 如果没有提供实现,假定接口本身就是实现类
self._registry[interface] = interface
else:
self._registry[interface] = implementation

defresolve(self, interface: Type) -> Any:
"""解析服务"""
if interface notinself._registry:
raise ValueError(f"未注册的服务: {interface}")

        implementation = self._registry[interface]

# 如果已经是实例,直接返回
ifnot inspect.isclass(implementation) andnotcallable(implementation):
return implementation

# 创建实例
returnself._instantiate(implementation)

def_instantiate(self, cls_or_factory) -> Any:
"""实例化类或调用工厂函数"""
ifcallable(cls_or_factory) andnot inspect.isclass(cls_or_factory):
# 是工厂函数
return cls_or_factory(self)

# 是类,需要解析构造函数参数
        signature = inspect.signature(cls_or_factory.__init__)
        parameters = {}

for param_name, param in signature.parameters.items():
if param_name == 'self':
continue

# 获取参数的类型注解
            param_type = param.annotation
if param_type == inspect.Parameter.empty:
raise ValueError(f"构造函数参数'{param_name}'缺少类型注解")

# 递归解析依赖
            parameters[param_name] = self.resolve(param_type)

return cls_or_factory(**parameters)

使用示例:

# 定义接口和实现
classIDatabase:
defquery(self, sql: str) -> list:
pass

classMySQLDatabase(IDatabase):
def__init__(self, host: str, port: int):
self.host = host
self.port = port

defquery(self, sql: str) -> list:
return [{"id"1"name""示例数据"}]

classUserService:
def__init__(self, db: IDatabase):
self.db = db

defget_user(self, user_id: int):
returnself.db.query(f"SELECT * FROM users WHERE id = {user_id}")

# 使用容器
container = SimpleContainer()
container.register(IDatabase, MySQLDatabase)
container.register(UserService)

# 注意:MySQLDatabase需要配置参数,我们的容器还不支持!
# 这里会报错,因为MySQLDatabase需要host和port参数

这个基础版本虽然简单,但已经展示了容器的核心思想:注册-解析-实例化循环。但它有很多不足,咱们继续改进。

2.2 第二步:支持构造函数参数注入

上面的容器无法处理需要配置参数的类,比如MySQLDatabase(host='localhost', port=3306)。咱们来改进:

# container_with_params.py
from typing importDictAnyUnionTypeCallableOptional
import inspect

classContainerWithParams(SimpleContainer):
"""支持构造函数参数注入的容器"""

def__init__(self):
super().__init__()
# 参数配置:类型 -> 参数字典
self._parameters: Dict[TypeDict[strAny]] = {}
# 配置值:类型+参数名 -> 值
self._config_values: Dict[strAny] = {}

defregister_with_params(self, interface: Type, implementation=None, **params):
"""注册服务并指定构造函数参数"""
super().register(interface, implementation)
if params:
self._parameters[interface] = params

defset_config(self, interface: Type, param_name: str, value: Any):
"""设置配置值"""
        key = f"{interface.__name__}.{param_name}"
self._config_values[key] = value

def_instantiate(self, cls_or_factory) -> Any:
"""改进的实例化方法,支持参数注入"""
ifcallable(cls_or_factory) andnot inspect.isclass(cls_or_factory):
return cls_or_factory(self)

# 获取类
        cls = cls_or_factory

# 检查是否有预定义的参数
if cls inself._parameters:
# 直接使用预定义参数
return cls(**self._parameters[cls])

# 解析构造函数参数
        signature = inspect.signature(cls.__init__)
        parameters = {}

for param_name, param in signature.parameters.items():
if param_name == 'self':
continue

# 检查是否有配置值
            config_key = f"{cls.__name__}.{param_name}"
if config_key inself._config_values:
                parameters[param_name] = self._config_values[config_key]
continue

# 获取类型注解
            param_type = param.annotation
if param_type == inspect.Parameter.empty:
# 没有类型注解,检查是否有默认值
if param.default != inspect.Parameter.empty:
                    parameters[param_name] = param.default
else:
raise ValueError(f"参数'{param_name}'缺少类型注解且无默认值")
else:
# 递归解析依赖
                parameters[param_name] = self.resolve(param_type)

return cls(**parameters)

使用示例:

container = ContainerWithParams()

# 注册MySQLDatabase并指定参数
container.register_with_params(
    IDatabase,
    MySQLDatabase,
    host='localhost',
    port=3306
)

# 注册UserService
container.register(UserService)

# 现在可以正确解析了
user_service = container.resolve(UserService)
print(user_service.get_user(1))  # 正常工作

这个版本已经实用多了,但还有几个重要问题:

  1. 1. 不支持单例模式
  2. 2. 不支持工厂函数
  3. 3. 不能处理复杂的依赖图

2.3 第三步:实现完整的依赖注入容器

现在咱们来实现一个功能完整的容器,支持多种注册方式和生命周期管理:

# advanced_container.py
from enum import Enum
from typing importDictAnyUnionTypeCallableOptionalList
import inspect

classLifecycle(Enum):
"""对象生命周期"""
    TRANSIENT = "transient"# 每次解析都创建新实例
    SINGLETON = "singleton"# 全局单例
    SCOPED = "scoped"# 作用域单例

classRegistration:
"""服务注册信息"""

def__init__(
        self,
        interface: Type,
        implementation: Union[TypeAnyCallable],
        lifecycle: Lifecycle = Lifecycle.TRANSIENT,
        factory: Optional[Callable] = None,
        instance: Optional[Any] = None
):
self.interface = interface
self.implementation = implementation
self.lifecycle = lifecycle
self.factory = factory
self.instance = instance  # 对于单例,存储实例

classAdvancedContainer:
"""高级依赖注入容器"""

def__init__(self):
self._registrations: Dict[Type, Registration] = {}
self._scoped_instances: Dict[TypeAny] = {}
self._resolution_stack: List[Type] = []  # 用于检测循环依赖
self._parent: Optional['AdvancedContainer'] = None

# ---------- 注册方法 ----------

defregister(
        self,
        interface: Type,
        implementation=None,
        lifecycle: Lifecycle = Lifecycle.TRANSIENT
) -> 'AdvancedContainer':
"""基本注册"""
if implementation isNone:
            implementation = interface

self._registrations[interface] = Registration(
            interface=interface,
            implementation=implementation,
            lifecycle=lifecycle
        )
returnself

defregister_singleton(
        self,
        interface: Type,
        implementation=None
) -> 'AdvancedContainer':
"""注册单例"""
returnself.register(interface, implementation, Lifecycle.SINGLETON)

defregister_instance(
        self,
        interface: Type,
        instance: Any
) -> 'AdvancedContainer':
"""注册已有实例"""
self._registrations[interface] = Registration(
            interface=interface,
            implementation=type(instance),
            lifecycle=Lifecycle.SINGLETON,
            instance=instance
        )
returnself

defregister_factory(
        self,
        interface: Type,
        factory: Callable
) -> 'AdvancedContainer':
"""注册工厂函数"""
self._registrations[interface] = Registration(
            interface=interface,
            implementation=None,
            lifecycle=Lifecycle.TRANSIENT,
            factory=factory
        )
returnself

# ---------- 解析方法 ----------

defresolve(self, interface: Type) -> Any:
"""解析服务"""
# 检查循环依赖
if interface inself._resolution_stack:
raise RuntimeError(f"检测到循环依赖: {' -> '.join([t.__name__ for t in self._resolution_stack] + [interface.__name__])}")

self._resolution_stack.append(interface)

try:
# 查找注册信息
            registration = self._find_registration(interface)

# 根据生命周期处理
if registration.lifecycle == Lifecycle.SINGLETON:
returnself._resolve_singleton(registration)
elif registration.lifecycle == Lifecycle.SCOPED:
returnself._resolve_scoped(registration)
else:  # TRANSIENT
returnself._resolve_transient(registration)
finally:
self._resolution_stack.pop()

def_find_registration(self, interface: Type) -> Registration:
"""查找注册信息"""
# 先在当前容器查找
if interface inself._registrations:
returnself._registrations[interface]

# 然后在父容器查找
ifself._parent:
returnself._parent._find_registration(interface)

raise ValueError(f"未注册的服务: {interface}")

def_resolve_singleton(self, registration: Registration) -> Any:
"""解析单例"""
if registration.instance isnotNone:
return registration.instance

# 创建实例并缓存
        instance = self._create_instance(registration)
        registration.instance = instance
return instance

def_resolve_scoped(self, registration: Registration) -> Any:
"""解析作用域单例"""
if registration.interface inself._scoped_instances:
returnself._scoped_instances[registration.interface]

        instance = self._create_instance(registration)
self._scoped_instances[registration.interface] = instance
return instance

def_resolve_transient(self, registration: Registration) -> Any:
"""解析瞬时对象"""
returnself._create_instance(registration)

def_create_instance(self, registration: Registration) -> Any:
"""创建实例"""
if registration.factory:
# 使用工厂函数
return registration.factory(self)

# 检查是否已经是实例
ifnot inspect.isclass(registration.implementation):
return registration.implementation

# 解析构造函数
returnself._instantiate_class(registration.implementation)

def_instantiate_class(self, cls: Type) -> Any:
"""实例化类"""
# 获取所有构造函数参数
        signature = inspect.signature(cls.__init__)
        kwargs = {}

for param_name, param in signature.parameters.items():
if param_name == 'self':
continue

# 检查参数类型
            param_type = param.annotation

if param_type != inspect.Parameter.empty:
# 有类型注解,尝试解析
try:
                    kwargs[param_name] = self.resolve(param_type)
except (ValueError, RuntimeError):
# 无法解析,检查是否有默认值
if param.default != inspect.Parameter.empty:
                        kwargs[param_name] = param.default
else:
raise
else:
# 没有类型注解,使用默认值
if param.default != inspect.Parameter.empty:
                    kwargs[param_name] = param.default
else:
# 既没有类型注解也没有默认值,尝试作为值参数
# 在实际应用中,这里可能需要更复杂的逻辑
pass

return cls(**kwargs)

# ---------- 容器组合 ----------

defcreate_child_container(self) -> 'AdvancedContainer':
"""创建子容器"""
        child = AdvancedContainer()
        child._parent = self
return child

defenter_scope(self) -> 'AdvancedContainer':
"""进入新的作用域"""
        scoped_container = self.create_child_container()
# 作用域容器继承父容器的注册
return scoped_container

defexit_scope(self):
"""退出作用域(清理作用域实例)"""
self._scoped_instances.clear()

这个高级容器已经具备了生产环境所需的大部分功能。让咱们看看如何使用它:

# 使用示例
container = AdvancedContainer()

# 1. 注册单例数据库连接
container.register_singleton(IDatabase, MySQLDatabase)

# 2. 注册需要配置的类
classConfigService:
def__init__(self, app_name: str = "默认应用"):
self.app_name = app_name

container.register(ConfigService)

# 3. 注册工厂函数
defcreate_logger(container: AdvancedContainer) -> 'Logger':
    config = container.resolve(ConfigService)
return Logger(name=config.app_name)

container.register_factory(Logger, create_logger)

# 4. 使用作用域
with container.enter_scope() as scoped_container:
# 在作用域内解析的服务会缓存
    db1 = scoped_container.resolve(IDatabase)
    db2 = scoped_container.resolve(IDatabase)
print(db1 is db2)  # True(在作用域内是同一个实例)

# 作用域结束,作用域实例被清理

3. 依赖注入容器的设计模式

在实现容器的过程中,咱们用到了多种设计模式。理解这些模式有助于咱们更好地设计和使用容器。

3.1 组合模式(Composite Pattern)

容器本身就是一个组合模式的应用:

classContainerComposite:
"""容器组合模式实现"""

def__init__(self):
self._children: List['ContainerComposite'] = []

defadd_child(self, child: 'ContainerComposite'):
self._children.append(child)

defresolve(self, interface: Type) -> Any:
# 首先尝试自己解析
if interface inself._registrations:
returnsuper().resolve(interface)

# 然后尝试子容器
for child inself._children:
try:
return child.resolve(interface)
except ValueError:
continue

raise ValueError(f"未找到服务: {interface}")

组合模式让容器可以形成层级结构,支持复杂的应用场景。

3.2 装饰器模式(Decorator Pattern)

依赖注入容器经常与装饰器模式结合使用:

definject(func):
"""依赖注入装饰器"""

    @wraps(func)
defwrapper(*args, **kwargs):
# 获取函数签名
        signature = inspect.signature(func)
        bound_args = signature.bind_partial(*args, **kwargs)

# 填充缺失的参数
for param_name, param in signature.parameters.items():
if param_name in bound_args.arguments:
continue

# 检查类型注解
            param_type = param.annotation
if param_type != inspect.Parameter.empty:
# 从容器解析
                container = get_current_container()
                bound_args.arguments[param_name] = container.resolve(param_type)
elif param.default != inspect.Parameter.empty:
                bound_args.arguments[param_name] = param.default

return func(*bound_args.args, **bound_args.kwargs)

return wrapper

3.3 策略模式(Strategy Pattern)

容器的生命周期管理就是策略模式的典型应用:

classLifecycleStrategy:
"""生命周期策略接口"""

defget_instance(
        self,
        registration: Registration,
        container: 'AdvancedContainer'
) -> Any:
pass

classSingletonStrategy(LifecycleStrategy):
"""单例策略"""

def__init__(self):
self._instance = None

defget_instance(self, registration, container):
ifself._instance isNone:
self._instance = container._create_instance(registration)
returnself._instance

classTransientStrategy(LifecycleStrategy):
"""瞬时策略"""

defget_instance(self, registration, container):
return container._create_instance(registration)

# 在容器中使用策略
classStrategyBasedContainer:
def__init__(self):
self._strategies: Dict[Lifecycle, LifecycleStrategy] = {
            Lifecycle.SINGLETON: SingletonStrategy(),
            Lifecycle.TRANSIENT: TransientStrategy()
        }

4. 循环依赖的检测与处理

循环依赖是依赖注入容器必须面对的挑战。让咱们深入分析这个问题。

4.1 循环依赖的类型

# 直接循环依赖
classA:
def__init__(self, b: 'B'):
self.b = b

classB:
def__init__(self, a: A):
self.a = a

# 间接循环依赖
classX:
def__init__(self, y: 'Y'):
self.y = y

classY:
def__init__(self, z: 'Z'):
self.z = z

classZ:
def__init__(self, x: X):
self.x = x

4.2 检测算法

咱们在高级容器中实现的检测算法基于调用栈跟踪:

defdetect_circular_dependency(container, start_type: Type) -> bool:
"""使用深度优先搜索检测循环依赖"""

    visited = set()
    recursion_stack = set()

defdfs(current_type: Type) -> bool:
if current_type in recursion_stack:
returnTrue# 发现循环

if current_type in visited:
returnFalse

        visited.add(current_type)
        recursion_stack.add(current_type)

# 获取当前类型的所有依赖
        dependencies = get_dependencies(container, current_type)

for dep_type in dependencies:
if dfs(dep_type):
returnTrue

        recursion_stack.remove(current_type)
returnFalse

return dfs(start_type)

defget_dependencies(container, interface: Type) -> List[Type]:
"""获取类型的所有依赖"""
    registration = container._find_registration(interface)
    cls = registration.implementation

ifnot inspect.isclass(cls):
return []

    signature = inspect.signature(cls.__init__)
    dependencies = []

for param in signature.parameters.values():
if param.name == 'self':
continue

        param_type = param.annotation
if param_type != inspect.Parameter.empty:
            dependencies.append(param_type)

return dependencies

4.3 解决方案

方案1:属性注入(Setter Injection)

classA:
def__init__(self):
self.b = None

defset_b(self, b: 'B'):
self.b = b

classB:
def__init__(self, a: A):
self.a = a

# 使用
a = A()
b = B(a)
a.set_b(b)  # 后设置属性

方案2:延迟注入(Lazy Injection)

classLazyProxy:
"""延迟代理"""

def__init__(self, container, interface: Type):
self._container = container
self._interface = interface
self._instance = None

def__getattr__(self, name):
ifself._instance isNone:
self._instance = self._container.resolve(self._interface)
returngetattr(self._instance, name)

classA:
def__init__(self, b: LazyProxy):
self.b = b  # b是代理,使用时才真正解析

classB:
def__init__(self, a: A):
self.a = a

方案3:接口分离(Interface Segregation)

# 将循环依赖拆分为多个接口
classIReader:
defread(self) -> str:
pass

classIWriter:
defwrite(self, data: str):
pass

classA(IReader, IWriter):
def__init__(self, reader: IReader, writer: IWriter):
self.reader = reader
self.writer = writer

defread(self):
returnself.reader.read()

defwrite(self, data):
self.writer.write(data)

5. 性能优化技巧

依赖注入容器如果不加以优化,可能会成为性能瓶颈。让咱们看看如何优化。

5.1 缓存反射结果

反射操作(如inspect.signature)是昂贵的,应该缓存结果:

classReflectionCache:
"""反射结果缓存"""

    _signature_cache: Dict[Type, inspect.Signature] = {}
    _dependency_cache: Dict[TypeList[Type]] = {}

    @classmethod
defget_signature(cls, obj) -> inspect.Signature:
        key = obj if inspect.isclass(obj) elsetype(obj)

if key notin cls._signature_cache:
            cls._signature_cache[key] = inspect.signature(
                obj if inspect.isclass(obj) else obj.__init__
            )

return cls._signature_cache[key]

    @classmethod
defget_dependencies(cls, clazz: Type) -> List[Type]:
if clazz notin cls._dependency_cache:
            signature = cls.get_signature(clazz)
            dependencies = []

for param in signature.parameters.values():
if param.name == 'self':
continue

                param_type = param.annotation
if param_type != inspect.Parameter.empty:
                    dependencies.append(param_type)

            cls._dependency_cache[clazz] = dependencies

return cls._dependency_cache[clazz]

5.2 预编译依赖图

在应用启动时预编译所有服务的依赖图:

classPrecompiledContainer:
"""预编译依赖图的容器"""

def__init__(self):
self._dependency_graph: Dict[TypeList[Type]] = {}
self._resolution_plan: Dict[TypeList[Type]] = {}

defcompile(self):
"""编译所有注册的服务"""
for interface inself._registrations:
self._compile_type(interface)

def_compile_type(self, interface: Type):
"""编译单个类型"""
if interface inself._resolution_plan:
return

        registration = self._find_registration(interface)

# 获取直接依赖
        dependencies = ReflectionCache.get_dependencies(registration.implementation)
self._dependency_graph[interface] = dependencies

# 递归编译依赖
for dep in dependencies:
self._compile_type(dep)

# 生成解析顺序(拓扑排序)
self._resolution_plan[interface] = self._topological_sort(interface)

def_topological_sort(self, start_type: Type) -> List[Type]:
"""拓扑排序生成解析顺序"""
        visited = set()
        result = []

defdfs(node: Type):
if node in visited:
return

            visited.add(node)

for neighbor inself._dependency_graph.get(node, []):
                dfs(neighbor)

            result.append(node)

        dfs(start_type)
return result

5.3 使用__slots__减少内存占用

容器管理的对象如果很多,可以使用__slots__减少内存:

classLightweightService:
"""使用__slots__的轻量级服务"""

    __slots__ = ('data''cache''config')

def__init__(self, data, cache, config):
self.data = data
self.cache = cache
self.config = config

# 相比普通类,每个实例节省约50%内存

6. 实际框架分析:dependency-injector源码解析

让咱们看看实际生产环境中使用的依赖注入框架是如何实现的。以dependency-injector为例:

6.1 核心类结构

# dependency_injector的核心类(简化版)
classProvider:
"""所有提供者的基类"""

def__init__(self):
self._overrides = []

def__call__(self):
"""获取提供的对象"""
raise NotImplementedError

defoverride(self, provider):
"""重写提供者"""
self._overrides.append(provider)
returnself

classFactory(Provider):
"""工厂提供者"""

def__init__(self, provides, *args, **kwargs):
super().__init__()
self._provides = provides
self._args = args
self._kwargs = kwargs

def__call__(self):
ifself._overrides:
returnself._overrides[-1]()

# 解析依赖
        resolved_args = [resolve_arg(arg) for arg inself._args]
        resolved_kwargs = {k: resolve_arg(v) for k, v inself._kwargs.items()}

returnself._provides(*resolved_args, **resolved_kwargs)

classSingleton(Factory):
"""单例提供者"""

def__init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self._instance = None

def__call__(self):
ifself._instance isNone:
self._instance = super().__call__()
returnself._instance

6.2 容器实现原理

dependency_injector的容器实际上是提供者的集合:

classContainerMeta(type):
"""容器的元类,用于自动装配"""

def__new__(mcs, name, bases, namespace):
# 收集所有Provider实例
        providers = {}
for key, value in namespace.items():
ifisinstance(value, Provider):
                providers[key] = value

# 创建容器类
        cls = super().__new__(mcs, name, bases, namespace)
        cls._providers = providers

return cls

classContainer(metaclass=ContainerMeta):
"""依赖注入容器"""

    @classmethod
defprovider(cls, name):
"""获取提供者"""
return cls._providers[name]

defwire(self, modules=None):
"""自动装配"""
# 遍历模块,注入依赖
pass

6.3 依赖解析机制

dependency-injector使用装饰器和描述符实现依赖解析:

classProvide:
"""依赖描述符"""

def__init__(self, provider):
self._provider = provider

def__get__(self, instance, owner):
if instance isNone:
returnself

# 从容器获取实例
        container = getattr(instance, '_container'None)
if container isNone:
raise RuntimeError("容器未设置")

return container.provider(self._provider)()

definject(func):
"""依赖注入装饰器"""

    @wraps(func)
defwrapper(self, *args, **kwargs):
# 解析参数依赖
        signature = inspect.signature(func)
        bound = signature.bind_partial(self, *args, **kwargs)

for param_name, param in signature.parameters.items():
if param_name in bound.arguments:
continue

# 检查是否有Provide描述符
ifisinstance(param.default, Provide):
                bound.arguments[param_name] = param.default.__get__(selftype(self))

return func(*bound.args, **bound.kwargs)

return wrapper

7. 测试策略:如何测试依赖注入容器

依赖注入容器本身也需要测试。让咱们看看如何为容器编写测试:

7.1 单元测试容器功能

# test_container.py
import unittest
from typing import Protocol

classIService(Protocol):
defdo_something(self) -> str:
pass

classServiceImpl(IService):
defdo_something(self) -> str:
return"完成"

classTestContainer(unittest.TestCase):

defsetUp(self):
self.container = AdvancedContainer()

deftest_register_and_resolve(self):
"""测试基本注册和解析"""
self.container.register(IService, ServiceImpl)

        instance = self.container.resolve(IService)

self.assertIsInstance(instance, ServiceImpl)
self.assertEqual(instance.do_something(), "完成")

deftest_singleton_lifecycle(self):
"""测试单例生命周期"""
self.container.register_singleton(IService, ServiceImpl)

        instance1 = self.container.resolve(IService)
        instance2 = self.container.resolve(IService)

self.assertIs(instance1, instance2, "单例应该返回同一个实例")

deftest_transient_lifecycle(self):
"""测试瞬时生命周期"""
self.container.register(IService, ServiceImpl)

        instance1 = self.container.resolve(IService)
        instance2 = self.container.resolve(IService)

self.assertIsNot(instance1, instance2, "瞬时应该返回不同实例")

deftest_circular_dependency_detection(self):
"""测试循环依赖检测"""
classA:
def__init__(self, b: 'B'):
self.b = b

classB:
def__init__(self, a: A):
self.a = a

self.container.register(A)
self.container.register(B)

withself.assertRaises(RuntimeError) as context:
self.container.resolve(A)

self.assertIn("循环依赖"str(context.exception))

deftest_factory_registration(self):
"""测试工厂函数注册"""
defcreate_service(container) -> IService:
return ServiceImpl()

self.container.register_factory(IService, create_service)

        instance = self.container.resolve(IService)
self.assertIsInstance(instance, ServiceImpl)

7.2 集成测试容器应用

# test_integration.py
classTestIntegration(unittest.TestCase):

deftest_complete_application(self):
"""测试完整应用"""
        container = AdvancedContainer()

# 配置容器
        container.register_singleton(IDatabase, MySQLDatabase)
        container.register(IUserRepository, UserRepository)
        container.register(IOrderRepository, OrderRepository)
        container.register(UserService)
        container.register(OrderService)
        container.register(OrderController)

# 模拟Web请求
        controller = container.resolve(OrderController)
        result = controller.create_order({
"user_id"123,
"items": [{"product_id"1"quantity"2}]
        })

self.assertTrue(result["success"])
self.assertIn("order_id", result["data"])

deftest_container_hierarchy(self):
"""测试容器层级"""
        root = AdvancedContainer()
        child = root.create_child_container()

# 在根容器注册
        root.register(IService, ServiceImpl)

# 子容器应该能解析父容器的服务
        instance = child.resolve(IService)
self.assertIsInstance(instance, ServiceImpl)

# 子容器注册的服务不应该影响父容器
classAnotherService(IService):
defdo_something(self):
return"另一个"

        child.register(IService, AnotherService)

# 父容器还是原来的实现
        root_instance = root.resolve(IService)
self.assertIsInstance(root_instance, ServiceImpl)

# 子容器是自己的实现
        child_instance = child.resolve(IService)
self.assertIsInstance(child_instance, AnotherService)

7.3 性能测试

# test_performance.py
import time
import statistics

classTestPerformance(unittest.TestCase):

deftest_resolution_speed(self):
"""测试解析速度"""
        container = AdvancedContainer()

# 注册100个服务
for i inrange(100):
classService:
def__init__(self, num: int = i):
self.num = num

            container.register(Service)

# 测量解析时间
        times = []
for _ inrange(1000):
            start = time.perf_counter()
            container.resolve(Service)
            end = time.perf_counter()
            times.append((end - start) * 1000)  # 转换为毫秒

        avg_time = statistics.mean(times)
        max_time = max(times)

print(f"平均解析时间: {avg_time:.3f}ms")
print(f"最大解析时间: {max_time:.3f}ms")

# 断言性能要求
self.assertLess(avg_time, 1.0"平均解析时间应小于1ms")
self.assertLess(max_time, 5.0"最大解析时间应小于5ms")

8. 总结与进阶

通过今天的学习,咱们深入理解了依赖注入容器的实现原理。从最简单的容器开始,逐步添加功能,最终实现了一个功能完整的容器。在这个过程中,咱们学习了:

  1. 1. 容器的核心职责:注册、解析、生命周期管理
  2. 2. 实现关键技术:反射、递归解析、循环依赖检测
  3. 3. 设计模式应用:组合、装饰器、策略模式
  4. 4. 性能优化技巧:缓存、预编译、内存优化
  5. 5. 测试策略:单元测试、集成测试、性能测试

8.1 关键收获

  • • 理解胜于记忆:知道容器如何工作,比记住API更重要
  • • 逐步构建:复杂系统从简单开始,逐步添加功能
  • • 模式识别:识别并应用设计模式解决通用问题
  • • 性能意识:在实现功能的同时考虑性能影响

8.2 进阶学习方向

如果你对依赖注入容器的实现还想深入了解,可以:

  1. 1. 研究其他语言实现:学习Java Spring、C# Autofac等容器的设计
  2. 2. 深入元编程:研究Python描述符、元类在DI中的应用
  3. 3. 探索AOP集成:了解如何将面向切面编程与依赖注入结合
  4. 4. 研究容器标准:了解JSR-330等依赖注入标准

8.3 实践建议

  1. 1. 自己实现一次:按照今天的步骤,自己动手实现一个容器
  2. 2. 阅读开源代码:深入研究dependency-injector等框架的源码
  3. 3. 应用到实际项目:在项目中实践分层架构和依赖注入
  4. 4. 参与开源贡献:为现有的依赖注入框架贡献代码或文档

学习搭子,依赖注入容器是现代软件工程中的重要工具。理解它的原理不仅能让咱们更好地使用现有框架,还能在需要时定制自己的解决方案。希望今天的学习对你有所帮助!

如果在实践中遇到问题,或者想深入讨论某个技术细节,随时告诉我。咱们一起成长,一起进步!🚀

最新文章

随机文章

基本 文件 流程 错误 SQL 调试
  1. 请求信息 : 2026-07-04 12:29:15 HTTP/2.0 GET : https://f.mffb.com.cn/a/490165.html
  2. 运行时间 : 0.096494s [ 吞吐率:10.36req/s ] 内存消耗:5,003.78kb 文件加载:140
  3. 缓存信息 : 0 reads,0 writes
  4. 会话信息 : SESSION_ID=10b267b35bae32d18b4f45cf10bfb064
  1. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/public/index.php ( 0.79 KB )
  2. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/autoload.php ( 0.17 KB )
  3. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/autoload_real.php ( 2.49 KB )
  4. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/platform_check.php ( 0.90 KB )
  5. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/ClassLoader.php ( 14.03 KB )
  6. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/autoload_static.php ( 4.90 KB )
  7. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper.php ( 8.34 KB )
  8. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-validate/src/helper.php ( 2.19 KB )
  9. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/helper.php ( 1.47 KB )
  10. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/stubs/load_stubs.php ( 0.16 KB )
  11. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Exception.php ( 1.69 KB )
  12. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-container/src/Facade.php ( 2.71 KB )
  13. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/deprecation-contracts/function.php ( 0.99 KB )
  14. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/polyfill-mbstring/bootstrap.php ( 8.26 KB )
  15. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/polyfill-mbstring/bootstrap80.php ( 9.78 KB )
  16. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/var-dumper/Resources/functions/dump.php ( 1.49 KB )
  17. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-dumper/src/helper.php ( 0.18 KB )
  18. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/var-dumper/VarDumper.php ( 4.30 KB )
  19. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/App.php ( 15.30 KB )
  20. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-container/src/Container.php ( 15.76 KB )
  21. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/container/src/ContainerInterface.php ( 1.02 KB )
  22. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/provider.php ( 0.19 KB )
  23. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Http.php ( 6.04 KB )
  24. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper/Str.php ( 7.29 KB )
  25. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Env.php ( 4.68 KB )
  26. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/common.php ( 0.03 KB )
  27. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/helper.php ( 18.78 KB )
  28. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Config.php ( 5.54 KB )
  29. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/app.php ( 0.95 KB )
  30. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/cache.php ( 0.78 KB )
  31. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/console.php ( 0.23 KB )
  32. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/cookie.php ( 0.56 KB )
  33. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/database.php ( 2.48 KB )
  34. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/Env.php ( 1.67 KB )
  35. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/filesystem.php ( 0.61 KB )
  36. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/lang.php ( 0.91 KB )
  37. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/log.php ( 1.35 KB )
  38. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/middleware.php ( 0.19 KB )
  39. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/route.php ( 1.89 KB )
  40. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/session.php ( 0.57 KB )
  41. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/trace.php ( 0.34 KB )
  42. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/view.php ( 0.82 KB )
  43. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/event.php ( 0.25 KB )
  44. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Event.php ( 7.67 KB )
  45. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/service.php ( 0.13 KB )
  46. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/AppService.php ( 0.26 KB )
  47. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Service.php ( 1.64 KB )
  48. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Lang.php ( 7.35 KB )
  49. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/lang/zh-cn.php ( 13.70 KB )
  50. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/Error.php ( 3.31 KB )
  51. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/RegisterService.php ( 1.33 KB )
  52. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/services.php ( 0.14 KB )
  53. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/PaginatorService.php ( 1.52 KB )
  54. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/ValidateService.php ( 0.99 KB )
  55. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/ModelService.php ( 2.04 KB )
  56. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/Service.php ( 0.77 KB )
  57. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Middleware.php ( 6.72 KB )
  58. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/BootService.php ( 0.77 KB )
  59. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/Paginator.php ( 11.86 KB )
  60. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-validate/src/Validate.php ( 63.20 KB )
  61. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/Model.php ( 23.55 KB )
  62. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/Attribute.php ( 21.05 KB )
  63. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/AutoWriteData.php ( 4.21 KB )
  64. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/Conversion.php ( 6.44 KB )
  65. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/DbConnect.php ( 5.16 KB )
  66. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/ModelEvent.php ( 2.33 KB )
  67. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/RelationShip.php ( 28.29 KB )
  68. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/contract/Arrayable.php ( 0.09 KB )
  69. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/contract/Jsonable.php ( 0.13 KB )
  70. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/contract/Modelable.php ( 0.09 KB )
  71. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Db.php ( 2.88 KB )
  72. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/DbManager.php ( 8.52 KB )
  73. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Log.php ( 6.28 KB )
  74. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Manager.php ( 3.92 KB )
  75. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/log/src/LoggerTrait.php ( 2.69 KB )
  76. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/log/src/LoggerInterface.php ( 2.71 KB )
  77. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Cache.php ( 4.92 KB )
  78. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/simple-cache/src/CacheInterface.php ( 4.71 KB )
  79. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper/Arr.php ( 16.63 KB )
  80. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/cache/driver/File.php ( 7.84 KB )
  81. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/cache/Driver.php ( 9.03 KB )
  82. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/CacheHandlerInterface.php ( 1.99 KB )
  83. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/Request.php ( 0.09 KB )
  84. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Request.php ( 55.78 KB )
  85. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/middleware.php ( 0.25 KB )
  86. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Pipeline.php ( 2.61 KB )
  87. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/TraceDebug.php ( 3.40 KB )
  88. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/middleware/SessionInit.php ( 1.94 KB )
  89. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Session.php ( 1.80 KB )
  90. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/session/driver/File.php ( 6.27 KB )
  91. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/SessionHandlerInterface.php ( 0.87 KB )
  92. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/session/Store.php ( 7.12 KB )
  93. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Route.php ( 23.73 KB )
  94. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleName.php ( 5.75 KB )
  95. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Domain.php ( 2.53 KB )
  96. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleGroup.php ( 22.43 KB )
  97. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Rule.php ( 26.95 KB )
  98. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleItem.php ( 9.78 KB )
  99. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/route/app.php ( 1.72 KB )
  100. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/Route.php ( 4.70 KB )
  101. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/dispatch/Controller.php ( 4.74 KB )
  102. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Dispatch.php ( 10.44 KB )
  103. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/controller/Index.php ( 4.81 KB )
  104. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/BaseController.php ( 2.05 KB )
  105. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/facade/Db.php ( 0.93 KB )
  106. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/connector/Mysql.php ( 5.44 KB )
  107. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/PDOConnection.php ( 52.47 KB )
  108. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Connection.php ( 8.39 KB )
  109. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/ConnectionInterface.php ( 4.57 KB )
  110. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/builder/Mysql.php ( 16.58 KB )
  111. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Builder.php ( 24.06 KB )
  112. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/BaseBuilder.php ( 27.50 KB )
  113. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Query.php ( 15.71 KB )
  114. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/BaseQuery.php ( 45.13 KB )
  115. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/TimeFieldQuery.php ( 7.43 KB )
  116. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/AggregateQuery.php ( 3.26 KB )
  117. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ModelRelationQuery.php ( 20.07 KB )
  118. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ParamsBind.php ( 3.66 KB )
  119. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ResultOperation.php ( 7.01 KB )
  120. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/WhereQuery.php ( 19.37 KB )
  121. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/JoinAndViewQuery.php ( 7.11 KB )
  122. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/TableFieldInfo.php ( 2.63 KB )
  123. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/Transaction.php ( 2.77 KB )
  124. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/log/driver/File.php ( 5.96 KB )
  125. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/LogHandlerInterface.php ( 0.86 KB )
  126. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/log/Channel.php ( 3.89 KB )
  127. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/event/LogRecord.php ( 1.02 KB )
  128. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/Collection.php ( 16.47 KB )
  129. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/View.php ( 1.70 KB )
  130. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/View.php ( 4.39 KB )
  131. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Response.php ( 8.81 KB )
  132. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/response/View.php ( 3.29 KB )
  133. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Cookie.php ( 6.06 KB )
  134. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-view/src/Think.php ( 8.38 KB )
  135. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/TemplateHandlerInterface.php ( 1.60 KB )
  136. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/Template.php ( 46.61 KB )
  137. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/template/driver/File.php ( 2.41 KB )
  138. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/template/contract/DriverInterface.php ( 0.86 KB )
  139. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/runtime/temp/067d451b9a0c665040f3f1bdd3293d68.php ( 11.98 KB )
  140. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/Html.php ( 4.42 KB )
  1. CONNECT:[ UseTime:0.000578s ] mysql:host=127.0.0.1;port=3306;dbname=f_mffb;charset=utf8mb4
  2. SHOW FULL COLUMNS FROM `fenlei` [ RunTime:0.000821s ]
  3. SELECT * FROM `fenlei` WHERE `fid` = 0 [ RunTime:0.000312s ]
  4. SELECT * FROM `fenlei` WHERE `fid` = 63 [ RunTime:0.000298s ]
  5. SHOW FULL COLUMNS FROM `set` [ RunTime:0.000518s ]
  6. SELECT * FROM `set` [ RunTime:0.000195s ]
  7. SHOW FULL COLUMNS FROM `article` [ RunTime:0.000521s ]
  8. SELECT * FROM `article` WHERE `id` = 490165 LIMIT 1 [ RunTime:0.000558s ]
  9. UPDATE `article` SET `lasttime` = 1783139355 WHERE `id` = 490165 [ RunTime:0.010390s ]
  10. SELECT * FROM `fenlei` WHERE `id` = 66 LIMIT 1 [ RunTime:0.000256s ]
  11. SELECT * FROM `article` WHERE `id` < 490165 ORDER BY `id` DESC LIMIT 1 [ RunTime:0.004475s ]
  12. SELECT * FROM `article` WHERE `id` > 490165 ORDER BY `id` ASC LIMIT 1 [ RunTime:0.000804s ]
  13. SELECT * FROM `article` WHERE `id` < 490165 ORDER BY `id` DESC LIMIT 10 [ RunTime:0.004300s ]
  14. SELECT * FROM `article` WHERE `id` < 490165 ORDER BY `id` DESC LIMIT 10,10 [ RunTime:0.001064s ]
  15. SELECT * FROM `article` WHERE `id` < 490165 ORDER BY `id` DESC LIMIT 20,10 [ RunTime:0.000781s ]
0.098273s