python如何通過注冊表動態(tài)管理組件
mmengine.registry 是一個在多個開源項目(如 MMEngine 和 MM系列工具包,例如 MMDetection、MMClassification 等)中使用的模塊化管理機(jī)制,旨在通過注冊表動態(tài)管理不同的組件,例如模型、算法、損失函數(shù)或插件等。這個機(jī)制極大地增強(qiáng)了項目的靈活性和擴(kuò)展性。
1 基本概念
1.1 Registry 類
Registry是一個存儲映射關(guān)系的容器,它將字符串類型的鍵(通常是組件的名字)映射到具體的 Python 對象(如類或函數(shù))。使用注冊表的主要優(yōu)勢是可以在運行時動態(tài)創(chuàng)建對象,從而實現(xiàn)高度可配置和可擴(kuò)展的設(shè)計。
1.2 核心功能
注冊機(jī)制:
提供了一個 register_module 方法,允許開發(fā)者將類或函數(shù)注冊到注冊表中。這通常通過裝飾器的形式實現(xiàn),使得代碼更加簡潔和直觀。注冊時可以指定一個或多個名稱作為鍵,關(guān)聯(lián)到相應(yīng)的 Python 類或函數(shù)。
動態(tài)創(chuàng)建實例:
注冊表可以使用 build 方法根據(jù)配置動態(tài)創(chuàng)建類的實例。這通常需要配置信息包括鍵名 type(對應(yīng)注冊的名稱)和其他用于初始化對象的參數(shù)。通過讀取配置文件(通常是 JSON 或 YAML 格式),可以在不修改代碼的情況下輕松切換使用的組件或調(diào)整參數(shù)。
作用域管理:
在某些復(fù)雜的應(yīng)用場景中,Registry 可能需要處理多個域(scope)下的注冊問題,例如不同的模塊可能需要有各自獨立的注冊表。Registry 類可以通過指定 scope 參數(shù)來實現(xiàn)作用域管理,使得同一個名字在不同的作用域下可以關(guān)聯(lián)到不同的對象。
繼承:
注冊表可以設(shè)置父注冊表,使得查找過程可以在當(dāng)前注冊表未找到對應(yīng)鍵值時回溯到父注冊表中查找。這為模塊間的依賴提供了方便,也允許更靈活的重載和擴(kuò)展。
2 實現(xiàn)示例
2.1 簡化的Registry類實現(xiàn)
展示了如何定義這個類,以及如何注冊和創(chuàng)建對象:
class Registry:
def __init__(self, name, scope=None, parent=None):
self.name = name
self.scope = scope
self.parent = parent
self._module_dict = {}
def register_module(self, name=None):
def _register(cls):
module_name = name or cls.__name__
if module_name in self._module_dict:
raise KeyError(f'{module_name} is already registered in {self.scope}::{self.name}')
self._module_dict[module_name] = cls
return cls
return _register
def get(self, name):
if name in self._module_dict:
return self._module_dict[name]
elif self.parent:
return self.parent.get(name)
else:
raise KeyError(f'{name} is not registered in {self.scope}::{self.name} and no parent registry to fallback.')
def build(self, cfg):
module_name = cfg['type']
module_cls = self.get(module_name)
return module_cls(**{k: v for k, v in cfg.items() if k != 'type'})2.2 這個注冊表類
from torch import nn
# 創(chuàng)建父注冊表
MODELS = Registry('models', scope='mmengine')
# 注冊一個模塊到父注冊表
@MODELS.register_module()
class ResNet(nn.Module):
def __init__(self, layers):
self.layers = layers
def forward(self, x):
return x
# 創(chuàng)建子注冊表,指定 MODELS 為父注冊表
DETECTORS = Registry('detectors', scope='mmengine', parent=MODELS)
# 注冊一個模塊只到子注冊表
@DETECTORS.register_module()
class FasterRCNN(nn.Module):
def __init__(self, num_classes):
self.num_classes = num_classes
def forward(self, x):
return x
# 從子注冊表構(gòu)建 ResNet 實例,盡管它是在父注冊表中注冊的
resnet_instance = DETECTORS.build({'type': 'ResNet', 'layers': 50})
print(resnet_instance)
# 直接從子注冊表構(gòu)建 FasterRCNN 實例
fasterrcnn_instance = DETECTORS.build({'type': 'FasterRCNN', 'num_classes': 80})
print(fasterrcnn_instance)到此這篇關(guān)于python如何通過注冊表動態(tài)管理組件的文章就介紹到這了,更多相關(guān)python動態(tài)管理組件內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python辦公自動化之操控遠(yuǎn)程桌面和文件版本控制
這篇文章主要為大家詳細(xì)介紹了Python辦公自動化中操控遠(yuǎn)程桌面和文件版本控制的相關(guān)知識,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解下2024-01-01
Django實現(xiàn)學(xué)員管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Django實現(xiàn)學(xué)員管理系統(tǒng),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02
Python實現(xiàn)list反轉(zhuǎn)實例匯總
這篇文章主要介紹了Python實現(xiàn)list反轉(zhuǎn)的方法,實例總結(jié)了關(guān)于list的各種較為常見的操作技巧,需要的朋友可以參考下2014-11-11
Python使用Dash開發(fā)網(wǎng)頁應(yīng)用的方法詳解
本文主要是通過Dash的Checklist組件,簡單介紹使用Dash開發(fā)的Web應(yīng)用,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,需要的可以參考一下2022-09-09
python 多線程實現(xiàn)多任務(wù)的方法示例
本文主要介紹了python 多線程實現(xiàn)多任務(wù)的方法示例,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-07-07
Python 代碼中的 yield 關(guān)鍵字到底是什么
yield是Python中一個強(qiáng)大的工具,它可以幫助你以一種高效的方式處理大量數(shù)據(jù),理解yield的工作原理對于掌握Python編程至關(guān)重要,這篇文章主要介紹了Python 代碼中的 yield 到底是什么,需要的朋友可以參考下2024-07-07

