Python Multiprocessing Pool Map: AttributeError: Can't pickle local object

前端 未结 2 1976
无人共我
无人共我 2020-12-01 11:42

I have a method inside a class that needs to do a lot of work in a loop, and I would like to spread the work over all of my cores.

I wrote the following code, which

相关标签:
2条回答
  • 2020-12-01 12:14

    I accidentally discovered a very nasty solution. It works, as long as you use a def statement. If you declare the function, that you want to use in Pool.map with the global keyword at the beginning of the function that solves it. But I would not rely on this in serious applications

    0 讨论(0)
  • 2020-12-01 12:30

    Error 1:

    AttributeError: Can't pickle local object 'SomeClass.some_method..single'

    You solved this error yourself by moving the nested target-function single() out to the top-level.

    Background:

    Pool needs to pickle (serialize) everything it sends to its worker-processes (IPC). Pickling actually only saves the name of a function and unpickling requires re-importing the function by name. For that to work, the function needs to be defined at the top-level, nested functions won't be importable by the child and already trying to pickle them raises an exception (more).


    Error 2:

    AttributeError: Can't get attribute 'single' on module 'main' from '.../test.py'

    You are starting the pool before you define your function and classes, that way the child processes cannot inherit any code. Move your pool start up to the bottom and protect (why?) it with if __name__ == '__main__':

    import multiprocessing
    
    class OtherClass:
      def run(self, sentence, graph):
        return False
    
    
    def single(params):
        other = OtherClass()
        sentences, graph = params
        return [other.run(sentence, graph) for sentence in sentences]
    
    class SomeClass:
       def __init__(self):
           self.sentences = [["Some string"]]
           self.graphs = ["string"]
    
       def some_method(self):
          return list(pool.map(single, zip(self.sentences, self.graphs)))
    
    if __name__ == '__main__':  # <- prevent RuntimeError for 'spawn'
        # and 'forkserver' start_methods
        with multiprocessing.Pool(multiprocessing.cpu_count() - 1) as pool:
            print(SomeClass().some_method())
    

    Appendix

    ...I would like to spread the work over all of my cores.

    Potentially helpful background on how multiprocessing.Pool is chunking work:

    Python multiprocessing: understanding logic behind chunksize

    0 讨论(0)
提交回复
热议问题