Concurrent programming¶
Using concurrent.futures¶
concurrent.futures is a high-level interface for asynchronously executing callables (i.e., functions, methods, or any object that can be called). It’s part of the standard library and provides two main components: ThreadPoolExecutor and ProcessPoolExecutor.
ThreadPoolExecutor and ProcessPoolExecutor are implementations of an interface defined by the abstract base class Executor. They allow you to manage and control the execution of tasks in different threads or processes, respectively.
Here’s a simple example of how you can use concurrent.futures:
from concurrent.futures import ThreadPoolExecutor
import time
def task(n):
    time.sleep(n)
    return n
with ThreadPoolExecutor(max_workers=4) as executor:
    future = executor.submit(task, 5)
    print(future.result())  # this will print '5' after waiting for 5 seconds
In this example, we’re using ThreadPoolExecutor to create a pool of worker threads. The submit() method schedules a callable to be executed as task(5) and returns a Future object. Future objects represent the execution of the operation and allow you to check on the operation’s status or result.
The result() method of a Future object returns the result of the operation once it’s completed. If the operation hasn’t completed yet, it will wait until it does. If the operation completed successfully, result() will return its result. If the operation raised an exception, result() will raise the same exception.
For multiple tasks, you can use as_completed function which yields futures as they complete:
from concurrent.futures import ThreadPoolExecutor, as_completed
def task(n):
    time.sleep(n)
    return n
with ThreadPoolExecutor(max_workers=4) as executor:
    futures = {executor.submit(task, n) for n in range(5)}
    for future in as_completed(futures):
        print(future.result())  # this will print the numbers 0 to 4 as they complete
In this example, we’re submitting multiple tasks to the executor and getting an iterable of Future objects. Then, we’re using as_completed() to iterate over these futures as they complete.
The main benefits of concurrent.futures is that it provides a high-level, Pythonic way to do multithreading or multiprocessing. However, if you need more control over your threads or processes, you might need to use the lower-level threading or multiprocessing modules instead.
References¶
Using actors¶
The actor concurrent programming model is supported in Python vias 3rd-party libraries, like Pykka and Thespian.
The actor model is a design pattern for concurrent computation where “actors” are the universal primitives. They encapsulate state and behavior, communicate exclusively by sending messages, and each actor processes messages sequentially in the order they were received. This model helps to manage and reason about concurrency and distributed computation.
Using Pykka¶
Here’s a brief description of how you might use an actor system in Python using Pykka:
- 
Define Actor Classes: In Pykka, an actor is an instance of any Python class that subclasses pykka.ThreadingActororpykka.FutureActor. Here’s an example:```python 
 import pykkaclass MyActor(pykka.ThreadingActor): 
 def init(self, my_value):
 super().init()
 self.my_value = my_valuedef get_value(self): return self.my_value``` In this example, MyActoris an actor class with a single methodget_value(). It also has a constructor that accepts an argumentmy_value.
- 
Create Actors: To create an actor, just instantiate your class. The actor will start running in its own thread or process immediately. python actor_ref = MyActor.start(my_value=42)In this example, MyActor.start(my_value=42)creates a newMyActoractor withmy_valueset to42. It returns anActorRefthat you can use to interact with the actor.
- 
Send Messages to Actors: You can ask an actor to execute a method by sending it a message. In Pykka, you do this using the tell()method for sending a message without waiting for a reply, or theask()method for sending a message and waiting for a reply.python future = actor_ref.ask({'method': 'get_value'}) print(future.result()) # prints '42'In this example, actor_ref.ask({'method': 'get_value'})sends a message to the actor asking it to execute theget_value()method. This returns aFuturethat will be completed with the result of the method.
- 
Stop Actors: When you’re done with an actor, you should stop it to free up its resources. python actor_ref.stop()In this example, actor_ref.stop()stops the actor.
Using Thespian¶
The following example shows how you can create an actor system using Thespian:
- 
Define Actor Classes: In Thespian, an actor is an instance of any Python class that subclasses thespian.actors.Actor. Here’s an example:```python 
 from thespian.actors import Actorclass MyActor(Actor): 
 def init(self, my_value):
 self.my_value = my_valuedef receiveMessage(self, message, sender): if message == 'get_value': self.send(sender, self.my_value)``` In this example, MyActoris an actor class with a single methodreceiveMessage(). This method is called whenever the actor receives a message. It also has a constructor that accepts an argumentmy_value.
- 
Create Actors: To create an actor, you need to create an actor system first and then use it to create your actor. ```python 
 from thespian.actors import ActorSystemactor_system = ActorSystem() 
 actor_ref = actor_system.createActor(MyActor, globalName=’MyActor’, my_value=42)
 ```In this example, ActorSystem()creates a new actor system.actor_system.createActor(MyActor, globalName='MyActor', my_value=42)creates a newMyActoractor withmy_valueset to42. It returns anActorRefthat you can use to interact with the actor.
- 
Send Messages to Actors: You can ask an actor to execute a method by sending it a message. In Thespian, you do this using the tell()method.python future = actor_system.ask(actor_ref, 'get_value') print(future) # prints '42'In this example, actor_system.ask(actor_ref, 'get_value')sends a message to the actor asking it to execute theget_value()method. It returns the result of the method directly.
- 
Stop Actors: When you’re done with an actor, you should stop it to free up its resources. python actor_system.tell(actor_ref, ActorExitRequest())In this example, actor_system.tell(actor_ref, ActorExitRequest())sends a message to the actor telling it to stop.
Notes: Thespian actors can be distributed across multiple machines and support a variety of serialization methods for messages. Actor failure can be detected and managed, and actors can be dynamically added or removed from the system. The messages passed between actors can be any Python object.
Pros and cons of the actor model (in Python)¶
The actor model, as a concurrent computational model, has its strengths and weaknesses. It works best for systems with many independent entities that need to maintain their own state and behavior while occasionally interacting with each other. However, it is probably not be the best choice for problems that require a lot of data sharing or tight coupling between entities.
Page last modified: 2024-11-19 09:38:33