我可以在类的方法中使用multiprocessing.Pool吗?

时间:2014-10-08 04:57:34

标签: python python-3.x multiprocessing python-multiprocessing

我想在我的代码中使用multiprocessing以获得更好的性能。

但是,我收到如下错误:

Traceback (most recent call last):
  File "D:\EpubBuilder\TinyEpub.py", line 49, in <module>
    e.epub2txt()
  File "D:\EpubBuilder\TinyEpub.py", line 43, in epub2txt
    tempread = self.get_text()
  File "D:\EpubBuilder\TinyEpub.py", line 29, in get_text
    txtlist = pool.map(self.char2text,charlist)
  File "C:\Python34\lib\multiprocessing\pool.py", line 260, in map
    return self._map_async(func, iterable, mapstar, chunksize).get()
  File "C:\Python34\lib\multiprocessing\pool.py", line 599, in get
    raise self._value
  File "C:\Python34\lib\multiprocessing\pool.py", line 383, in _handle_tasks
    put(task)
  File "C:\Python34\lib\multiprocessing\connection.py", line 206, in send
    self._send_bytes(ForkingPickler.dumps(obj))
  File "C:\Python34\lib\multiprocessing\reduction.py", line 50, in dumps
    cls(buf, protocol).dump(obj)
TypeError: cannot serialize '_io.BufferedReader' object

我已经尝试了另一种方式并且遇到了这个错误:

TypeError: cannot serialize '_io.TextIOWrapper' object

我的代码如下所示:

from multiprocessing import Pool
class Book(object):
    def __init__(self, arg):
        self.namelist = arg
    def format_char(self,char):
        char = char + "a"
        return char
    def format_book(self):
        self.tempread = ""
        charlist = [f.read() for f in self.namelist] #list of char
        with Pool() as pool:
            txtlist = pool.map(self.format_char,charlist)
        self.tempread = "".join(txtlist)
        return self.tempread

if __name__ == '__main__':
    import os
    b = Book([open(f) for f in os.listdir()])
    t = b.format_book()
    print(t)

我认为由于未在main函数中使用Pool而引发错误。

我的推测是对的吗?我如何修改我的代码来修复错误?

1 个答案:

答案 0 :(得分:23)

问题是您在namelist实例中有一个不可解决的实例变量(Book)。因为您在实例方法上调用了pool.map,并且您在Windows上运行,所以需要对整个实例进行选择,以便将其传递给子进程。 Book.namelist是一个打开的文件对象(_io.BufferedReader),无法进行腌制。你可以通过几种方式解决这个问题。根据示例代码,您可以将format_char设为顶级函数:

def format_char(char):
    char = char + "a"
    return char


class Book(object):
    def __init__(self, arg):
        self.namelist = arg

    def format_book(self):
        self.tempread = ""
        charlist = [f.read() for f in self.namelist] #list of char
        with Pool() as pool:
            txtlist = pool.map(format_char,charlist)
        self.tempread = "".join(txtlist)
        return self.tempread

但是,如果实际上您需要format_char作为实例方法,则可以使用__getstate__/__setstate__使Book可选,方法是从namelist中移除class Book(object): def __init__(self, arg): self.namelist = arg def __getstate__(self): """ This is called before pickling. """ state = self.__dict__.copy() del state['namelist'] return state def __setstate__(self, state): """ This is called while unpickling. """ self.__dict__.update(state) def format_char(self,char): char = char + "a" def format_book(self): self.tempread = "" charlist = [f.read() for f in self.namelist] #list of char with Pool() as pool: txtlist = pool.map(self.format_char,charlist) self.tempread = "".join(txtlist) return self.tempread 参数酸洗之前的实例:

namelist

只要您不需要在子进程中访问{{1}},就可以了。