Last week, I explained how we can start asynchronous Java threads to run in parallel to the main Matlab processing. Today I continue the series by examining .Net threads. Next week I will discuss C++ threads, followed by a final article on timers and process-spawning.
I continue using last week’s example, where we compute some data, save it to file on a relatively slow USB/network disk, and then proceed with another calculation. The purpose of multi-threading would be to offload the I/O onto a separate thread, so that the Matlab computation can continue in parallel without needing to wait for the slow I/O.
Dot-Net (.Net), like Java and C++, also enables multithreading. .Net libraries (assemblies) are commonly distributed as DLL files, which can be loaded into Matlab using the NET.addAssembly function, similarly to the javaaddpath function for Java classes. Using these assemblies in Matlab is then as straight-forward as in Java:
NET.addAssembly('C:\Yair\Code\NetThread.dll'); data = rand(5e6,1); % pre-processing (5M elements, ~40MB) start(My.NetThread('F:\test.data',data)); % start running in parallel data = fft(data); % post-processing (.Net I/O runs in parallel) |
As with Java, the assembly only needs to be loaded once per Matlab session, not repeatedly. The definition of the .Net class closely follows that of the Java class. Unfortunately, .Net’s System.Threading.Thread
class is sealed (non-inheritable), so we cannot extend it as we did in Java. However, we can instantiate an internal Thread
object within our class and use it. Here is the corresponding C# source code (thanks to Ronnie Shmuel who assisted):
using System; using System.IO; using System.Threading; namespace My { public class NetThread { string filename; double[] doubleData; Thread thread; public NetThread(string filename, double[] data) { this.filename = filename; this.doubleData = data; thread = new Thread(this.run); } public void start() { thread.Start(); // note the capital S in Start() } private void run() { try { BinaryWriter out = new BinaryWriter( File.Open(filename,FileMode.Create)) for (int i = 0; i < doubleData.Length; i++) { out.Write(doubleData[i]); } out.Close(); } catch (Exception ex) { Console.WriteLine(ex.Message); } } } } |
Note that C# Thread
uses the Start() method (uppercase S), unlike java.lang.Thread
that uses start() (lowercase s). Also, while java.lang.Thread.start()
always invokes a run() method (the “run” name is predefined), C# Thread.Start()
can invoke any delegate method. In this example I chose to name it run() for consistency with the Java example, but in fact I could have named it anything else.
In this example I’ve used C# source code, but in fact I could also use C++, F# or VB code, since the System.Threading.Thread
class behaves very similarly in all of them. Once we compile it in our favorite compiler [hmmm, Microsoft Visual Studio anyway…], the resulting .Net assembly aught to work exactly the same in Matlab.
Compatibility and synchronization
Unlike Java and C++, .Net is not platform-independent: it only works on Windows, so if we need a cross-platform solution then we need to use Java or one of the other alternatives.
Also note that the .Net solution will only work on platforms that installed .Net of a supporting .Net version (Framework). .Net is commonly installed on Win7+, but not for example on XP. On Windows platforms where .Net is not installed, we need to download and install it before we can use any .Net features (duh!). To ensure .NET is supported on our current platform, use the NET.isNETSupported function:
>> NET.isNETSupported % returns true/false ans = 1 |
As with Java, when compiling a .Net class that should be used within Matlab, we must ensure that we are compiling for a .Net Framework that is equal to or lower than what Matlab will actually use in the target platform, as reported by System.Environment.Version
:
% My system uses .Net 4.0, so we should compile for .Net 4.0, not 4.5 >> char(System.Environment.Version.ToString) ans = 4.0.30319.18034 |
As with Java threads, .Net threads are most effective when they can run independently of Matlab and of each other. I do not know of any mechanism for data synchronization between .NET threads and Matlab. However, we can use one of the standard IPC mechanisms, as I shall discuss in a future article. For example, it is possible to open a COM connection to the invoking Matlab (serving as an automation server) and then use it in the .Net code.