Jak*_*łos 13 python python-3.x
我的 Python 3 程序有问题。我使用 Mac OS X。此代码运行正常。
# -*- coding: utf-8 -*-
#! python3
# sendDuesReminders.py - Sends emails based on payment status in spreadsheet.
import openpyxl, smtplib, sys
# Open the spreadsheet and get the latest dues status.
wb = openpyxl.load_workbook('duesRecords.xlsx')
sheet = wb.get_sheet_by_name('Sheet1')
lastCol = sheet.max_column
latestMonth = sheet.cell(row=1, column=lastCol).value
# Check each member's payment status.
unpaidMembers = {}
for r in range(2, sheet.max_row + 1):
payment = sheet.cell(row=r, column=lastCol).value
if payment != 'zaplacone':
name = sheet.cell(row=r, column=2).value
lastname = sheet.cell(row=r, column=3).value
email = sheet.cell(row=r, column=4).value
unpaidMembers[name] = email
# Log in to email account.
smtpObj = smtplib.SMTP_SSL('smtp.gmail.com', 465)
smtpObj.ehlo()
smtpObj.login('abc@abc.com', '1234')
# Send out reminder emails.
for name, email in unpaidMembers.items()
body = "Subject: %s - przypomnienie o platnosci raty za treningi GIT Parkour. " \
"\n\nPrzypominamy o uregulowaniu wplaty za uczestnictwo: %s w treningach GIT Parkour w ." \
"\n\nRecords show that you have not paid dues for %s. Please make " \
"this payment as soon as possible."%(latestMonth, name, latestMonth)
print('Sending email to %s...' % email)
sendmailStatus = smtpObj.sendmail('abc@abc.com', email, body)
if sendmailStatus != {}:
print('There was a problem sending email to %s: %s' % (email,
sendmailStatus))
smtpObj.quit()enter code here
Run Code Online (Sandbox Code Playgroud)
当我尝试将下一个值添加到 for 循环时,问题就开始了。
# Send out reminder emails.
for name, lastname, email in unpaidMembers.items()
body = "Subject: %s - przypomnienie o platnosci raty za treningi GIT Parkour. " \
"\n\nPrzypominamy o uregulowaniu wplaty za uczestnictwo: %s %s w treningach GIT Parkour w ." \
"\n\nRecords show that you have not paid dues for %s. Please make " \
"this payment as soon as possible."%(latestMonth, name, lastname, latestMonth)
print('Sending email to %s...' % email)
sendmailStatus = smtpObj.sendmail('abc@abc.com', email, body)
Run Code Online (Sandbox Code Playgroud)
终端显示错误:
Traceback (most recent call last):
File "sendDuesEmailReminder.py", line 44, in <module>
for name, email, lastname in unpaidMembers.items():
ValueError: not enough values to unpack (expected 3, got 2)
Run Code Online (Sandbox Code Playgroud)
1.首先要了解错误含义
错误not enough values to unpack (expected 3, got 2)意思是:
一个由 2 部分组成的元组,但分配给3 个值
我已经编写了演示代码来向您展示:
#!/usr/bin/python
# -*- coding: utf-8 -*-
# Function: Showing how to understand ValueError 'not enough values to unpack (expected 3, got 2)'
# Author: Crifan Li
# Update: 20191212
def notEnoughUnpack():
"""Showing how to understand python error `not enough values to unpack (expected 3, got 2)`"""
# a dict, which single key's value is two part tuple
valueIsTwoPartTupleDict = {
"name1": ("lastname1", "email1"),
"name2": ("lastname2", "email2"),
}
# Test case 1: got value from key
gotLastname, gotEmail = valueIsTwoPartTupleDict["name1"] # OK
print("gotLastname=%s, gotEmail=%s" % (gotLastname, gotEmail))
# gotLastname, gotEmail, gotOtherSomeValue = valueIsTwoPartTupleDict["name1"] # -> ValueError not enough values to unpack (expected 3, got 2)
# Test case 2: got from dict.items()
for eachKey, eachValues in valueIsTwoPartTupleDict.items():
print("eachKey=%s, eachValues=%s" % (eachKey, eachValues))
# same as following:
# Background knowledge: each of dict.items() return (key, values)
# here above eachValues is a tuple of two parts
for eachKey, (eachValuePart1, eachValuePart2) in valueIsTwoPartTupleDict.items():
print("eachKey=%s, eachValuePart1=%s, eachValuePart2=%s" % (eachKey, eachValuePart1, eachValuePart2))
# but following:
for eachKey, (eachValuePart1, eachValuePart2, eachValuePart3) in valueIsTwoPartTupleDict.items(): # will -> ValueError not enough values to unpack (expected 3, got 2)
pass
if __name__ == "__main__":
notEnoughUnpack()
Run Code Online (Sandbox Code Playgroud)
使用VSCode调试效果:

2.对于你的代码
for name, email, lastname in unpaidMembers.items():
Run Code Online (Sandbox Code Playgroud)
但错误
ValueError: not enough values to unpack (expected 3, got 2)
表示 中的每一项(一个元组值)unpaidMembers,只有 1 个部分:email,对应上面的代码
unpaidMembers[name] = email
Run Code Online (Sandbox Code Playgroud)
所以应该将代码更改为:
for name, email in unpaidMembers.items():
Run Code Online (Sandbox Code Playgroud)
以避免错误。
但显然你期望额外的lastname,所以应该将上面的代码更改为
unpaidMembers[name] = (email, lastname)
Run Code Online (Sandbox Code Playgroud)
并更好地更改为更好的语法:
for name, (email, lastname) in unpaidMembers.items():
Run Code Online (Sandbox Code Playgroud)
那么一切就都OK并且清楚了。
您可能想在lastname这里分配您正在阅读的内容
lastname = sheet.cell(row=r, column=3).value
Run Code Online (Sandbox Code Playgroud)
对某事;目前该程序只是忘记了它
你可以在两行之后做,就像这样
unpaidMembers[name] = lastname, email
Run Code Online (Sandbox Code Playgroud)
你的程序仍然会在同一个地方崩溃,因为.items()仍然不会给你 3 元组,而是具有这种结构的东西:(name, (lastname, email))
好消息是,python 可以处理这个
for name, (lastname, email) in unpaidMembers.items():
Run Code Online (Sandbox Code Playgroud)
等等。
在这一行中:
for name, email, lastname in unpaidMembers.items():
Run Code Online (Sandbox Code Playgroud)
unpaidMembers.items()每次迭代只能有两个值。
这是一个小例子来说明这个问题:
这将起作用:
for alpha, beta, delta in [("first", "second", "third")]:
print("alpha:", alpha, "beta:", beta, "delta:", delta)
Run Code Online (Sandbox Code Playgroud)
这将失败,这就是您的代码所做的:
for alpha, beta, delta in [("first", "second")]:
print("alpha:", alpha, "beta:", beta, "delta:", delta)
Run Code Online (Sandbox Code Playgroud)
在最后一个示例中,列表中的什么值被分配给delta?没什么,没有足够的值,这就是问题所在。
| 归档时间: |
|
| 查看次数: |
172877 次 |
| 最近记录: |