python – 当list为空时,List comprehension抛出异常

我有一个函数可以过滤列表项目,如果它们的日期是过去的(小于当前日期).

meetings = []
def clean_old():
    meetings = [meeting for meeting in meetings if time.mktime(meeting) >= time.localtime()]

当列表为空时,此代码崩溃.

为什么会崩溃?它说要在会议上开会,如果会议是空的,那么一切都应该没问题.

我该如何修复它以及该事件的解释是什么?

最佳答案 我假设你看到了这个例外:

UnboundLocalError: local variable 'meetings' referenced before assignment

您在这里遇到的内容与列表推导实际上没有任何关系.发生此错误的原因是您最初在函数外部定义了会议,但是您尝试在函数内部为其分配新值.

当Python看到变量在函数内部被赋值时,它会将其视为特定于该函数的新变量.这可以防止函数访问具有相同名称的函数外部的任何变量.

在内部,Python正在做类似这样的事情:

meetings_outside = []

def clean_old():
    meetings_inside = [meeting for meeting in meetings_inside if time.mktime(meeting) >= time.localtime()]

您可以理解为什么会失败:meetings_inside是根据自身定义的.当Python尝试查找meetings_inside的值以开始迭代其内容时,它会失败,因为尚未为其分配值.

如何处理这取决于您使用的是哪个版本的Python,并且定义了会议的初始值.

在Python 3中,您只需将非本地会议添加到功能的顶部即可.这将告诉它您指的是名为meetings的现有变量,而不是创建另一个变量.

但是,您可能正在使用Python 2,它没有nonlocal关键字.它确实有global关键字,它执行相同的操作,但仅限于在模块的顶层定义会议:在任何其他函数或类之外.

例如,如果您的文件中没有其他内容,则可以在Python 2中使用:

进口时间

meetings = []

def clean_old():
    global meetings
    meetings = [meeting for meeting in meetings if time.mktime(meeting) >= time.localtime()]

clean_old()
print meetings
[]

但是这不会,因为会议是在函数内部定义的:

import time

def main():
    meetings = []

    def clean_old():
        global meetings
        meetings = [meeting for meeting in meetings if time.mktime(meeting) >= time.localtime()]

    clean_old()
    print meetings

main()
NameError: global name 'meetings' is not defined

你需要解决这个问题.最简单的方法是将赋值修改为:

 meetings[:] = [meeting for meeting in meetings if time.mktime(meeting) >= time.localtime()]

这告诉Python您正在替换会议中的所有值,但不会创建新的列表对象. (:语法称为“切片”,部分描述为in the Python tutorial.)

点赞