.NET Tutorial
An example use case in C#.NET
Introduction
This tutorial will guide you through the basics of using LCM .NET port. As the .NET port is basically a transcription of the original Java library, it tries to be functionally equivalent while maintaining C#.NET naming conventions and other platform specifics. All sample code is written in C# (as well as the port itself), but the principles are applicable to any of the languages supported by the .NET Framework.
The tutorial doesn’t cover the very basics of the LCM (message transmision principles, message definition format etc.) - please see the rest of the documentation before further reading.
Generating C#.NET-specific message files
To demonstrate basic functionality, this tutorial will use the same message
format and application logic as the \ref tut_java to accent similarities and
differences between Java and .NET ports. Let’s have the following type
specification, saved to a file named temperature_t.lcm
:
package exlcm;
struct example_t
{
int64_t timestamp;
double position[3];
double orientation[4];
int32_t num_ranges;
int16_t ranges[num_ranges];
string name;
boolean enabled;
}
In order to obtain C#.NET-specific handler class, we need to call lcm-gen
with
the --csharp
flag:
lcm-gen --csharp example_t.lcm
Besides, the lcm-gen
utility accepts the following .NET-specific options:
Option |
Default value |
Description |
---|---|---|
-csharp-path |
C#.NET file destination directory |
|
-csharp-mkdir |
1 |
Make C#.NET source directories automatically |
-csharp-strip-dirs |
0 |
Do not generate folders |
-csharp-decl |
: LCM.LCM.LCMEncodable |
String |
-csharp-root-nsp |
Root C#.NET namespace |
|
-csharp-default-nsp |
LCMTypes |
Default NET |
As with all tutorials, we will publish and subscribe to the “EXAMPLE” channel.
Initializing LCM
There are at least two ways how to use the .NET port of LCM:
if you don’t want to modify the library and just use it, the simplest way is to build the library and copy resulting lcm.dll to your application; you then need to add a reference to it (Project -> Add Reference… -> Browse) and you are ready to start communicating!
if you plan to do some changes to library source code, the recommended way is to add library Visual Studio project to your solution and reference it (Project -> Add Reference… -> Projects)
Main classes of the library are put in the LCM.LCM namespace (while helper
code is in LCM.Util). This results in quite funny fully qualified name of the
master class - LCM.LCM.LCM (its constructor is even funnier -
LCM.LCM.LCM.LCM() :-) ). It’s logical to use the using LCM.LCM
statement to
shorten calls to the library, but the naming scheme (chosen to agree with the
Java variant) makes it a little difficult - you cannot use bare ‘LCM’ as class
name - the compiler considers it to be the namespace. Instead, you need to
write LCM.LCM to denote the main class.
Generated message handlers are placed in the LCMTypes
namespace by default
(you can change this by specifying lcm-gen option --csharp-default-nsp
).
LCM itself has a mechanism to maintain single instance of the main class - static property LCM.Singleton:
LCM.LCM myLCM = LCM.LCM.Singleton;
You can also instantiate the class and take care of the object’s singularity by yourself:
LCM.LCM myLCM = new LCM.LCM();
In situations where the default connection URL (fetched by LCM from the
environment variable LCM_DEFAULT_URL
or defined by the constant
udpm://239.255.76.67:7667
when the former is empty) is not suitable,
the constructor can accept variable number of parameters specifying individual
connection strings.
For detailed information on the LCM .NET API please see the .NET API reference.
Publishing a message
In order to use LCM types, you can either build an assembly containing generated classes (needed when using LCM from other .NET language then C#), or include the classes directly to application project. Utilization of the generated classes is then fairly straightforward:
exlcm.example_t msg = new exlcm.example_t();
TimeSpan span = DateTime.Now - new DateTime(1970, 1, 1);
msg.timestamp = span.Ticks * 100;
msg.position = new double[] { 1, 2, 3 };
msg.orientation = new double[] { 1, 0, 0, 0 };
msg.ranges = new short[] { 0, 1, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14 };
msg.num_ranges = msg.ranges.Length;
msg.name = "example string";
msg.enabled = true;
myLCM.Publish("EXAMPLE", msg);
The data are simply assigned to appropriate fields inside the message container.
Passing the message object to the Publish
method of the LCM object places
it to specified channel of the communication bus (channel “EXAMPLE” here).
Subscribing to messages
In order to receive messages, you have two options:
write a class implementing the LCMSubscriber interface (just one handler-method MessageReceived) to assynchronously handle incoming messages
use standard class MessageAggregator (that internally implements LCMSubscriber interface) for synchronous blocking or not blocking message delivery
This tutorial exploits the former option - the class SimpleSubscriber
is defined
as internal inside the demo application class:
internal class SimpleSubscriber : LCM.LCMSubscriber
{
public void MessageReceived(LCM.LCM lcm, string channel, LCM.LCMDataInputStream dins)
{
Console.WriteLine("RECV: " + channel);
if (channel == "EXAMPLE")
{
exlcm.example_t msg = new exlcm.example_t(dins);
Console.WriteLine("Received message of the type example_t:");
Console.WriteLine(" timestamp = {0:D}", msg.timestamp);
Console.WriteLine(" position = ({0:N}, {1:N}, {2:N})",
msg.position[0], msg.position[1], msg.position[2]);
Console.WriteLine(" orientation = ({0:N}, {1:N}, {2:N}, {3:N})",
msg.orientation[0], msg.orientation[1], msg.orientation[2],
msg.orientation[3]);
Console.Write(" ranges = [ ");
for (int i = 0; i < msg.num_ranges; i++)
{
Console.Write(" {0:D}", msg.ranges[i]);
if (i < msg.num_ranges-1)
Console.Write(", ");
}
Console.WriteLine(" ]");
Console.WriteLine(" name = '" + msg.name + "'");
Console.WriteLine(" enabled = '" + msg.enabled + "'");
}
}
}
The class instance is then passed to LCM method SubscribeAll
that passes all
received messages to our subscriber class. When selective subscription is needed
(i.e. in almost all real-world cases as we usually don’t to listen to all channels),
method Subscribe
that takes the channel name pattern as an argument is to
be used.
myLCM.SubscribeAll(new SimpleSubscriber());
Putting it all together
Distribution of the LCM library includes a directory of examples. One of them
is a couple of programs implementing all described features. Please go to
examples/csharp/
to find Visual Studio solution ready to be built.
The complete example transmitter application:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using LCM;
namespace LCM.Examples
{
/// <summary>
/// Demo transmitter, see LCM .NET tutorial for more information
/// </summary>
class ExampleTransmit
{
public static void Main(string[] args)
{
try
{
LCM.LCM myLCM = LCM.LCM.Singleton;
exlcm.example_t msg = new exlcm.example_t();
TimeSpan span = DateTime.Now - new DateTime(1970, 1, 1);
msg.timestamp = span.Ticks * 100;
msg.position = new double[] { 1, 2, 3 };
msg.orientation = new double[] { 1, 0, 0, 0 };
msg.ranges = new short[] { 0, 1, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14 };
msg.num_ranges = msg.ranges.Length;
msg.name = "example string";
msg.enabled = true;
myLCM.Publish("EXAMPLE", msg);
}
catch (Exception ex)
{
Console.Error.WriteLine("Ex: " + ex);
}
}
}
}
The complete example receiver application:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using LCM;
namespace LCM.Examples
{
/// <summary>
/// Demo listener, demonstrating interoperability with other implementations
/// Just run this listener and use any of the example_t message senders
/// </summary>
class ExampleDisplay
{
public static void Main(string[] args)
{
LCM.LCM myLCM;
try
{
myLCM = new LCM.LCM();
myLCM.SubscribeAll(new SimpleSubscriber());
while (true)
System.Threading.Thread.Sleep(1000);
}
catch (Exception ex)
{
Console.Error.WriteLine("Ex: " + ex);
Environment.Exit(1);
}
}
internal class SimpleSubscriber : LCM.LCMSubscriber
{
public void MessageReceived(LCM.LCM lcm, string channel, LCM.LCMDataInputStream dins)
{
Console.WriteLine("RECV: " + channel);
if (channel == "EXAMPLE")
{
exlcm.example_t msg = new exlcm.example_t(dins);
Console.WriteLine("Received message of the type example_t:");
Console.WriteLine(" timestamp = {0:D}", msg.timestamp);
Console.WriteLine(" position = ({0:N}, {1:N}, {2:N})",
msg.position[0], msg.position[1], msg.position[2]);
Console.WriteLine(" orientation = ({0:N}, {1:N}, {2:N}, {3:N})",
msg.orientation[0], msg.orientation[1], msg.orientation[2],
msg.orientation[3]);
Console.Write(" ranges = [ ");
for (int i = 0; i < msg.num_ranges; i++)
{
Console.Write(" {0:D}", msg.ranges[i]);
if (i < msg.num_ranges-1)
Console.Write(", ");
}
Console.WriteLine(" ]");
Console.WriteLine(" name = '" + msg.name + "'");
Console.WriteLine(" enabled = '" + msg.enabled + "'");
}
}
}
}
}
Conclusion
The tutorial has provided a basic working demonstration of the LCM library .NET port. For further information, please see the LCM documentation.