簡體   English   中英

在 Python 中的 class 中定義多個列表變量的最佳方法

[英]Best way to define multiple list variables inside class in Python

我創建了一個在系統中存儲和編輯元數據的腳本。 我現在通過定義 class 和方法來清理我的代碼,以前我只使用單獨的函數。

在腳本中,我將某些類型的元數據的新舊值存儲在列表中,在腳本完成運行后打印出來。 我已經定義了多個列表(准確地說是 16 個),當通過方法傳遞它們時,我意識到這些列表非常多。 我想知道解決這個問題的最pythonic方法是什么。

這些是我在開始時定義的以下列表變量。 在函數/方法中,我將 append 值分配給它們。 最后,我將存儲的值作為報告打印出來。

split_name = []
split_name_new = []
name = []
name_new = []
meta = []
meta_new = []
series = []
series_new = []
product = []
owner = []
desc = []
desc_new = []
keywords = []
keywords_new = []
no_edit_page =[]
no_edit_page_name = []

在 class 中,我認為它看起來像(如果我單獨定義所有列表)

class Metadata_editor():
    def __init__(self,url):

        self.split_name = [] 
        self.split_name_new = []
        self.name = []
        self.name_new = []
        self.meta = []
        self.meta_new = []
        self.series = []
        self.series_new = []
        self.product = []
        self.owner = []
        self.desc = []
        self.desc_new = []
        self.keywords = []
        self.keywords_new = []
        self.no_edit_page =[]
        self.no_edit_page_name = []

        #Ugly solution because the method gets crowded by all the variables passed through
        def data_edit(self, split_name, split_name_new, name, name_new,.. etc):
            #Not the whole method, but just to give some idea..
            #Selenium function that locates meta
            md = driver.find_element_by_xpath("//input[@name='metadata-name']") 
            meta_data = md.get_attribute("value")
            #replace_words translate the word using a dictionary object
            meta_data_new = replace_words(meta_data,c) 
            meta.append(meta_data)
            meta_new.append(meta_data_new)

我意識到上面的解決方案並不理想。

我找到了另一種可以使用的方法,即定義一個列表列表。 然后解決方案看起來像這樣(見下文)。 然而,“data_list[10]”不像“所有者”那樣不言自明。 我的問題是,這是解決此問題的“最佳”方法,還是您有任何其他建議? 我真的沒有反對這個解決方案,但想知道是否有更“pythonic”的方法來解決這個問題。

class Metadata_editor():
    def __init__(self,url):
        self.data_list=[[] for _ in range(16)] #Creates a list, that contains 16 lists

    # More eloquent solution, as only one variable is passed through. However finding 
    # the right data from data_list is perhaps not as easy before 
    def data_edit(self, data_list): 

        md = driver.find_element_by_xpath("//input[@name='metadata-name']")
        meta_data = md.get_attribute("value")
        meta_data_new = replace_words(meta_data,c)
        data_list[5].append(meta_data)
        data_list[6].append(meta_data_new)

您可以將其存儲為字典。 這樣做的好處是能夠按名稱引用鍵,而不必記住索引。

class Metadata_editor():
    def __init__(self, url):
       keys = [
            'split_name', 'split_name_new', 'name', 'name_new' 'meta', 'meta_new',
            'series', 'series_new', 'product', 'owner', 'desc', 'desc_new',
            'keywords', 'keywords_new', 'no_edit_page', 'no_edit_page_name',
        ]
        self.data_dict = dict((x, []) for x in keys)

    def data_edit(self):
        md = driver.find_element_by_xpath("//input[@name='metadata-name']")
        meta_data = md.get_attribute("value")
        meta_data_new = replace_words(meta_data,c)
        self.data_dict['meta'].append(meta_data)
        self.data_dict['meta_new'].append(meta_data_new)

需要額外注意的幾點:

  • class 名稱通常遵循 UpperCaseCamelCase 約定。 所以Metadata_editor會更傳統地寫成MetadataEditor
  • 使用self在 class 上設置一個屬性,它可以在 class 中使用self訪問,該屬性不需要傳遞到方法中。 我在上面的例子中展示了這一點,在data_edit方法中訪問self.data_dict
  • 您還可以使用setattr將屬性設置為 class ,如其他一些答案所示。

有多種方法可以達到你想要的:

  • 使用字典

 names  = ["split_name","split_name_new","name","name_new","meta","meta_new","series","series_new","product","owner","desc","desc_new","keywords","keywords_new","no_edit_page","no_edit_page_name"]

data = {}

for name in names : 
     data[name] = []

使用setattr

...
def __init__(self, url):
    names = '''split_name split_name_new name
        name_new meta meta_new series series_new
        product owner desc desc_new keywords
        keywords_new no_edit_page
        no_edit_page_name'''.split()
    for name in names:
        setattr(self, name, [])
...

您可以初始化多個列表,如下所示:

class Metadata_editor():
    def __init__(self,list_names):
        [setattr(self,name,[]) for name in list_names]
me = Metadata_editor(['split_name','split_name_new']) # initialize two lists
me.split_name.append(5) # add value to a list
print(me.split_name, me.split_name_new)

>>[5], [ ]


一旦通過self.list_name設置為 class 的一部分,列表就可以在 class 中全局訪問 - 不再需要“傳入”。 要將列表初始化為特定值,您可以執行以下操作:

 def __init__(self,list_names,list_values): [setattr(self,name,value) for name,value in zip(list_names,list_values)]

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM