在这种情况下,如何将apply(map,args)转换为map(function,iterable)?

时间:2014-04-07 01:18:35

标签: python python-2.7 python-3.x tkinter

我下载了一个MultiListBox小部件,以及我成功应用的排序扩展。但是,这个小部件在python 3中从未对我有用。这些库是Tkinter(py2)和tkinter(py3)。

所以,我已经越来越接近工作了,但是我不太了解如何过去。不推荐使用apply()函数,看起来,即使在py3中也是如此。就这样吧。

这是我不明白的:

我们已apply(function, args [, keywords])map(function, iterable...)

调用apply()函数,如:

return apply(map, [None] + result)(结果是字符串元组列表)

documentation建议我采用将apply(function, *args, **keywords)转换为function(*args, **keywords)的方法。这一切都很好,但我无法弄清楚要传递什么作为新功能。

return map(what_am_i, result)

以下是一些相关的代码段。希望这一切都清楚,但随时可以要求更多。感谢。

我已添加了更多代码,以便为此处的内容提供背景信息。也许这会产生一些见解,使这段代码更加beautiful

在此示例中,result包含list长度为3(列)的1000-tuples(行)。这些1000-tuples的每个元素都是string,其中包含SubjectSenderDate

我现在使用完整的_sort()函数进行了更新。

现在,我已经更新了另一个重构apply()的实例。我不妨把这个问题看成这个问题所以其他人都可以享受。

from tkinter import *

class MultiListbox(Frame):
    def __init__(self, master, rowslist):
        Frame.__init__(self, master)
        self.rowslist = []
        self.colmapping = {}
        self.origData = None
        for (row, colwidth) in rowslist:
            frame = Frame(self);
            b = Button(frame, ...)
            b.bind(...)
            listbox = Listbox(frame, ...)
            self.rowslist.append(listbox)
            listbox.bind(...)
            ...
        ...
    ...
    def _scroll(self, *args):
        for row in self.rowslist:
            row.yview(None, args)
            # apply(row.yview, args)
    ...
    def get(self, first, last=None):
        result = []
        for row in self.rowslist:
            result.append(row.get(first,last))
        if last:
            return map(None, result)
            #return apply(map, [None] + result)
        return result
    ...
    def _sort(self, e):
        # get the listbox to sort by (mapped by the header button)
        b=e.widget
        col, direction = self.colmapping[b]     

        # get the entire table data into mem
        tableData = self.get(0,END)
        if self.origData == None:
            import copy
            self.origData = copy.deepcopy(tableData)

        rowcount = len(tableData)

        #remove old sort indicators if it exists
        for btn in self.colmapping:
            lab = btn.cget('text')
            if lab[0]=='<': btn.config(text=lab[4:])

        btnLabel = b.cget('text')
        #sort data based on direction
        if direction==0:
            tableData = self.origData
        else:
            if direction==1: b.config(text='<+> ' + btnLabel)
            else: b.config(text='<-> ' + btnLabel)
            # sort by col
            tableData.sort(key=lambda x: x[col], reverse=direction<0)

        #clear widget
        self.delete(0,END)

        # refill widget
        for row in range(rowcount):
            self.insert(END, tableData[row])

        # toggle direction flag
        if direction == 1:
            direction = -1
        else:
            direction = direction + 1

        self.colmapping[b] = (col, direction)
        # =:> End of complete _sort(self, e) <:= #
    ...

if __name__ == '__main__':
    tk = Tk()
    Label(tk, text='SortableMultiListbox').pack()
    mlb = MultiListbox(tk, (('Subject', 40), ('Sender', 20), ('Date', 10)))
    for i in range(1000):
        mlb.insert(END, ('Important Message: %d' % i, 'John Doe', '10/10/%04d' % (1900+i)))
    mlb.pack(expand=YES,fill=BOTH)
    tk.mainloop()

1 个答案:

答案 0 :(得分:4)

apply(map, [None] + result)

相当于

map(*[None] + result)

或者,或许更可读,

map(None, *result)

请注意,如果您想要转置,最好使用zip(*result)

>>> # Usually the same behavior:
... result = [[1, 2, 3], [4, 5, 6]]
>>> map(None, *result)
[(1, 4), (2, 5), (3, 6)]
>>> zip(*result)
[(1, 4), (2, 5), (3, 6)]
>>> # But if result has only one iterable in it:
... result = [[1, 2, 3]]
>>> map(None, *result)
[1, 2, 3]
>>> zip(*result)
[(1,), (2,), (3,)]