ผ่านพจนานุกรมไปยังฟังก์ชั่นเป็นพารามิเตอร์คำหลัก


346

ฉันต้องการเรียกใช้ฟังก์ชันในไพ ธ อนโดยใช้พจนานุกรม

นี่คือรหัสบางส่วน:

d = dict(param='test')

def f(param):
    print(param)

f(d)

พิมพ์นี้แต่ฉันต้องการเพียงแค่พิมพ์{'param': 'test'}test

ฉันต้องการให้พารามิเตอร์คล้ายกันทำงานมากกว่านี้:

d = dict(p1=1, p2=2)
def f2(p1, p2):
    print(p1, p2)
f2(d)

เป็นไปได้ไหม

คำตอบ:


528

คิดออกเพื่อตัวเองในที่สุด มันง่ายมากฉันเพิ่งหายตัวไปของผู้ดำเนินการ ** เพื่อแกะพจนานุกรมออก

ดังนั้นตัวอย่างของฉันกลายเป็น:

d = dict(p1=1, p2=2)
def f2(p1,p2):
    print p1, p2
f2(**d)

57
ถ้าคุณต้องการที่จะช่วยให้คนอื่น ๆ ที่คุณควรใช้ถ้อยคำคำถามของคุณ: ปัญหาที่ไม่ได้ผ่านพจนานุกรมสิ่งที่คุณต้องการก็คือการเปิด Dict ลงพารามิเตอร์คำหลัก
ฮาเวียร์

11
เป็นที่น่าสังเกตว่าคุณสามารถแยกรายการไปยังอาร์กิวเมนต์ตำแหน่ง: f2 (* [1,2])
Matthew Trevor

10
"dereference": คำศัพท์ปกติในบริบท Python นี้คือ "unpack" :)
mipadi

2
นี่ยอดเยี่ยมเพียงใช้กับ argparse / __ dict__ เพื่อให้ง่ายต่อการทำอาร์กิวเมนต์บรรทัดคำสั่งโดยแยกวิเคราะห์โดยตรงลงในตัวเลือกสำหรับคลาสวัตถุ
Horus

1
อะไรคือเหตุผลที่เราต้องการแยกไฟล์พจนานุกรมเมื่อผ่านมันเป็นอาร์กิวเมนต์ของฟังก์ชั่น?
Mona Jalal

128
In[1]: def myfunc(a=1, b=2):
In[2]:    print(a, b)

In[3]: mydict = {'a': 100, 'b': 200}

In[4]: myfunc(**mydict)
100 200

รายละเอียดเพิ่มเติมเล็กน้อยที่อาจเป็นประโยชน์ในการรู้ (คำถามที่ฉันได้หลังจากอ่านและผ่านการทดสอบแล้ว):

  1. ฟังก์ชั่นสามารถมีพารามิเตอร์ที่ไม่รวมอยู่ในพจนานุกรม
  2. คุณไม่สามารถแทนที่พารามิเตอร์ที่มีอยู่แล้วในพจนานุกรม
  3. พจนานุกรมไม่สามารถมีพารามิเตอร์ที่ไม่ได้อยู่ในฟังก์ชัน

ตัวอย่าง:

หมายเลข 1:ฟังก์ชันสามารถมีพารามิเตอร์ที่ไม่รวมอยู่ในพจนานุกรม

In[5]: mydict = {'a': 100}
In[6]: myfunc(**mydict)
100 2

หมายเลข 2:คุณไม่สามารถแทนที่พารามิเตอร์ที่มีอยู่แล้วในพจนานุกรม

In[7]: mydict = {'a': 100, 'b': 200}
In[8]: myfunc(a=3, **mydict)

TypeError: myfunc() got multiple values for keyword argument 'a'

หมายเลข 3:พจนานุกรมไม่สามารถมีพารามิเตอร์ที่ไม่ได้อยู่ในฟังก์ชันได้

In[9]:  mydict = {'a': 100, 'b': 200, 'c': 300}
In[10]: myfunc(**mydict)

TypeError: myfunc() got an unexpected keyword argument 'c'

ตามที่ร้องขอในความคิดเห็นโซลูชันสำหรับหมายเลข 3คือการกรองพจนานุกรมตามอาร์กิวเมนต์ของคำสำคัญที่มีอยู่ในฟังก์ชัน:

In[11]: import inspect
In[12]: mydict = {'a': 100, 'b': 200, 'c': 300}
In[13]: filtered_mydict = {k: v for k, v in mydict.items() if k in [p.name for p in inspect.signature(myfunc).parameters.values()]}
In[14]: myfunc(**filtered_mydict)
100 200

ตัวเลือกอื่นคือการยอมรับ (และเพิกเฉย) kwargs เพิ่มเติมในฟังก์ชั่นของคุณ:

In[15]: def myfunc2(a=None, **kwargs):
In[16]:    print(a)

In[17]: mydict = {'a': 100, 'b': 200, 'c': 300}

In[18]: myfunc2(**mydict)
100

แจ้งให้ทราบล่วงหน้ามากกว่าที่คุณสามารถใช้อาร์กิวเมนต์ตำแหน่งและรายการหรือสิ่งอันดับอย่างมีประสิทธิภาพเช่นเดียวกับ kwargs นี่เป็นตัวอย่างขั้นสูงเพิ่มเติมที่รวมทั้ง args ตำแหน่งและคำหลัก:

In[19]: def myfunc3(a, *posargs, b=2, **kwargs):
In[20]:    print(a, b)
In[21]:    print(posargs)
In[22]:    print(kwargs)

In[23]: mylist = [10, 20, 30]
In[24]: mydict = {'b': 200, 'c': 300}

In[25]: myfunc3(*mylist, **mydict)
10 200
(20, 30)
{'c': 300}

4
การใช้การเปิดกล่องบรรจุด้วย print.format มีประโยชน์อย่างยิ่ง เช่น:'hello {greeting} {name}'.format( **{'name': 'Andrew', 'greeting': 'Mr'})
Martlark

คำถามเก่า แต่ยังมีความเกี่ยวข้องมาก ขอบคุณสำหรับการตอบสนองอย่างละเอียด คุณรู้วิธีแก้ไขกรณีที่ 3 บ้างไหม? แปลความหมายของแผนที่ในรายการพจนานุกรมกับพารามิเตอร์ฟังก์ชันเมื่อมีรายการในพจนานุกรมมากกว่าพารามิเตอร์?
spencer

2
@spencer มีการเพิ่มโซลูชันในคำตอบ
David Parks

33

ในหลามนี้เรียกว่า "เอาออก" และคุณสามารถหาบิตเกี่ยวกับเรื่องนี้ในการกวดวิชา ฉันเห็นด้วยกับเอกสารของมันโดยเฉพาะอย่างยิ่งเพราะมันมีประโยชน์อย่างมาก


20
เป็นการดีกว่าที่จะคัดลอกเนื้อหาที่เกี่ยวข้องของลิงก์ไปยังคำตอบของคุณแทนที่จะใช้ลิงก์ที่เหลืออยู่จนกว่าจะหมดเวลา
Richard

3
@ Richard นั่นเป็นความเห็นเชิงลึกเกี่ยวกับเว็บซึ่งฉันไม่เห็นด้วยอย่างยิ่ง! อนิจจาฉันขาดพื้นที่ในที่นี่ขอบนี้เพื่อแบ่งปันหลักฐานที่ยอดเยี่ยมของฉัน ...
llimllib

@llimllib ฉันจะต้องถามดร. ไวล์สแล้ว!
Richard

6

นี่ยาไป - ทำงานซ้ำ ๆ อื่น ๆ :

d = {'param' : 'test'}

def f(dictionary):
    for key in dictionary:
        print key

f(d)

ดูเหมือนว่าผู้คนกำลัง downvoting สิ่งนี้เพราะมันตอบคำถามเดิมไม่ใช่คำถาม rephrased ฉันแนะนำให้ลบโพสต์นี้ทันที
dotancohen

@dotancohen ไม่เคยไม่ถูกต้องมันล้มเหลวในการบล็อกที่สองของรหัสที่มักจะมีคำถาม มันใช้อย่างแท้จริงเกินไปพิมพ์เป็นตัวอย่าง
Dave Hillier

มันจะตอบคำถาม แต่มันไม่ได้ทำผ่านการเปิดพจนานุกรม วิธีการของเขาใช้ได้อย่างสมบูรณ์ตามคำถามที่โพสต์
Natecat
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.