This is an “is this possible, and if so can you give me a quick example because I can’t find one online?” kind of question.
I have a number of completely separate (i.e. “embarrassingly parallel”) processes that I want to run in parallel using the Task Parallel library in .NET Framework 4 using C#. Some of these processes require the use of software that can be accessed via COM/OLE automation.
Specifically, there is a Parallel.Foreach() loop that divides up the tasks from a list of items, basically calling a different function inside the Parallel.Foreach to handle the processing (so some of these functions employ COM libraries to work).
Is this possible? Thanks.
It’s 100% possible to use COM objects with the TPL. While it’s true that, by default, the TPL will use the standard .NET ThreadPool, the TPL has an extension point via the
TaskSchedulerclass which enables you to provide your own scheduler which can dispatch work to threads which you’ve created.In the case of of using COM objects you first need to know if the COM class requires STA threading or MTA threading. If MTA threading, then there’s nothing special that needs to be done because the COM class can already be used from any random thread. Unfortunately most classic COM objects tend to rely on STA threading and that’s when you’d need to employ a custom
TaskSchedulerso that whatever .NET thread you’re using them from has been initialized as an STA compatible thread.While TaskSchedulers are not exactly trivial to write, they’re not really that hard to write either if you’ve got a basic understanding of threading. Luckily the ParallelExtensions Extras library already provides an
StaTaskSchedulerclass so you don’t even need to write anything yourself. There’s a great blog post here by the PFX team that discusses the implementation of and some use cases for the theStaTaskSchedulerclass.Basically though, you’ll want to initialize a new
StaTaskScheduleras a static somewhere on one of your classes and then just start yourTasksspecifying that they are scheduled by that instance. That would look something like this: