今天在寫(xiě)監(jiān)控腳本的時(shí)候遇到一個(gè)問(wèn)題,就是我執(zhí)行每一個(gè)監(jiān)控模塊(腳本)的時(shí)候,例如CPU、內(nèi)存、磁盤(pán)腳本,都會(huì)返回一個(gè)字典格式的數(shù)據(jù),但是我需要將這三個(gè)字典,組合成一個(gè)大字典,然后通過(guò)requests模塊發(fā)送給api接口,so,我就在網(wǎng)上找了一些方法,然后總結(jié),寫(xiě)成這編博文。

1、首先定義三個(gè)字典(不需要考慮字典的具體內(nèi)容)

            
              >>> cpu_dict = {'cpu_count':8,'cpu_ratio':3.5}
>>> memory_dict = {'memory_count':16,'memory_ration':10}
>>> disk_dict = {'disk_read':200,'disk_write':120,'tps':340}
            
          

2、將上面三個(gè)字典合并成一個(gè)字典

            
              >>> data_dict = dict(**cpu_dict,**memory_dict,**disk_dict)
>>> data_dict
{'cpu_count': 8, 'cpu_ratio': 3.5, 'memory_count': 16, 'memory_ration': 10, 'disk_read': 200, 'disk_write': 120, 'tps': 340}
            
          

注意:需要被合并的字典的key是不能出現(xiàn)重復(fù)的,否則python會(huì)直接報(bào)錯(cuò)

3、從源碼分析dict這個(gè)類(lèi)是如何實(shí)現(xiàn)字典拼接

            
              def __init__(self, seq=None, **kwargs):
    """
    忽略官方注釋..
    """
    pass
            
          

首先我們調(diào)用 dict() 這個(gè)類(lèi)在括號(hào)里傳入值,會(huì)執(zhí)行 init 構(gòu)造方法
第一位形參:self 是對(duì)象即文中的data_dict,
第二位形參:seq 排序作用(當(dāng)前文中我們忽略它)
第三位形參:**kwargs 接收所有以字典形式保存的數(shù)據(jù)

那么我們?cè)趫?zhí)行下面這條命令的時(shí)候發(fā)生了什么?

            
              data_dict = dict(**cpu_dict,**memory_dict,**disk_dict)
            
          

首先cpu_dict、memory_dict、disk_dict 本身就是字典,在它們前面加個(gè)了 * 后就是把這些字典傳給 dict類(lèi)里的__init__構(gòu)造函數(shù)的第三個(gè)位置形參來(lái) * kwargs 來(lái)接收字典,最后在返回結(jié)果給 data_dict 這個(gè)對(duì)象

4、 kwargs實(shí)現(xiàn)方法**

            
              def func(**kwargs):           # 位置形參
    print(kwargs)

func(**{'k1':'v1','k2':'v2'})  # 位置實(shí)參

# 執(zhí)行結(jié)果:
{'k1': 'v1', 'k2': 'v2'}
            
          

現(xiàn)在我們知道了,在位置形參里的 **kwargs 是可以接收所有字典形式的數(shù)據(jù),那如果我把位置實(shí)參里傳一個(gè)字典形式存儲(chǔ)的變量,會(huì)如何?

            
              t1 = {'k1':'v1','k2':'v2'}
def func(**kwargs):
    print(kwargs)

func(**t1)

# 執(zhí)行結(jié)果
{'k1': 'v1', 'k2': 'v2'}
            
          

so,我們可以看到直接執(zhí)行func函數(shù),通過(guò)* t1 可以直接將存儲(chǔ)字典的變量傳入給func函數(shù),當(dāng)然在func函數(shù)里也可以傳入多個(gè)字典,我再次聲明下再位置形參里的* kwargs,是可以接收N個(gè)從位置實(shí)參傳入的字典數(shù)據(jù)
那么,有朋友會(huì)問(wèn),位置形參里的 * kwargs 必須是固定這么寫(xiě)嗎?,不然,* 這個(gè)關(guān)鍵字是必須的,但是* 后面的變量名可以隨意,我們也可以寫(xiě)成 * abc,但是大部分人都是按照開(kāi)發(fā)標(biāo)準(zhǔn)統(tǒng)一寫(xiě)成 **kwargs,因?yàn)閜ython源碼也是這么寫(xiě)的,我們就遵循規(guī)范就行

4、利用神奇的雙星號(hào) 來(lái)自定義類(lèi)似dict的合并字典功能**

            
              cpu_dict = {'cpu_count':8,'cpu_ratio':3.5}
memory_dict = {'memory_count':16,'memory_ration':10}
disk_dict = {'disk_read':200,'disk_write':120,'tps':340}

class task():
    def __init__(self,*args,**kwargs):
        self.kwargs = kwargs

    def __call__(self, *args, **kwargs):
        print(kwargs)

my_dict = task()
my_dict(**cpu_dict,**memory_dict,**disk_dict)
            
          

最終執(zhí)行的結(jié)果:

            
              {'cpu_count': 8, 'cpu_ratio': 3.5, 'memory_count': 16, 'memory_ration': 10, 'disk_read': 200, 'disk_write': 120, 'tps': 340}