Azure IoT 技术研究系列3-设备到云、云到设备通信

上篇博文中我们将模拟设备注册到Azure IoT Hub中:我们得到了设备的唯一标识。

Azure IoT 技术研究系列2-设备注册到Azure IoT Hub

本文中我们继续深入研究,设备到云、云到设备通信。

1. 在Azure IoT Hub中接收模拟设备的消息

读取设备到云消息的Event Hub兼容终结点,使用 AMQP 协议。

我们新建一个Console控制台工程:IoTServer,添加Nuget引用:WindowsAzure.ServiceBus

核心的命名空间:using Microsoft.ServiceBus.Messaging;

核心类:EventHubClient 

通过EventHubClient创建一个EventHubReceiver,不间断的接收设备侧的消息。

1         static string connectionString = "HostName=IoTTest.*******;SharedAccessKeyName=iothubowner;SharedAccessKey=***";
2         static string iotHubD2cEndpoint = "messages/events";
3         static EventHubClient eventHubClient;

ReceiveMessagesFromDeviceAsync方法:

 1         /// <summary>
 2         /// 接收设备侧的消息
 3         /// </summary>
 4         /// <param name="partition">分区</param>
 5         /// <param name="ct">取消标识</param>
 6         /// <returns>Task</returns>
 7         private static async Task ReceiveMessagesFromDeviceAsync(string partition, CancellationToken ct)
 8         {
 9             var eventHubReceiver = eventHubClient.GetDefaultConsumerGroup().CreateReceiver(partition, DateTime.UtcNow);
10             while (true)
11             {
12                 if (ct.IsCancellationRequested) break;
13                 EventData eventData = await eventHubReceiver.ReceiveAsync();
14                 if (eventData == null) continue;
15 
16                 string data = Encoding.UTF8.GetString(eventData.GetBytes());
17                 Console.WriteLine("Message received. Partition: {0} Data: '{1}'", partition, data);
18 
19                 //防止CPU被占满
20                 Task.Delay(1).Wait();
21             }
22         }

Main函数中我们将整个IoTServer Run起来:

 1         static void Main(string[] args)
 2         {
 3             Console.WriteLine("Azure IoT Hub 接收消息..., Press Ctrl-C to exit.\n");
 4             eventHubClient = EventHubClient.CreateFromConnectionString(connectionString, iotHubD2cEndpoint);
 5 
 6             var d2cPartitions = eventHubClient.GetRuntimeInformation().PartitionIds;
 7 
 8             CancellationTokenSource cts = new CancellationTokenSource();
 9 
10             System.Console.CancelKeyPress += (s, e) =>
11             {
12                 e.Cancel = true;
13                 cts.Cancel();
14                 Console.WriteLine("Exiting...");
15             };
16 
17             var tasks = new List<Task>();
18             foreach (string partition in d2cPartitions)
19             {
20                 tasks.Add(ReceiveMessagesFromDeviceAsync(partition, cts.Token));
21             }
22 
23             Task.WaitAll(tasks.ToArray());
24         }

2. 模拟设备发送消息到Azure IoT Hub

我们同样新建一个Console控制台工程:Device,用于模拟向Azure IoT Hub 发送消息。

首先添加Nuget引用:Microsoft.Azure.Devices.Client,这个Nuget依赖的Nuget很多,不要着急,慢慢Install吧

核心的命名空间:

using Microsoft.Azure.Devices.Client;
using Newtonsoft.Json;

核心类:

Microsoft.Azure.Devices.Client.DeviceClient

模拟设备往Azure IoT Hub发消息时,用到了设备的Key(唯一标识)和IoT Hub HostName, 上篇博文中提到的主机名:Azure IoT 技术研究系列2-设备注册到Azure IoT Hub

1         static DeviceClient deviceClient;
2         static string iotHubUri = "IoTTest.******";          //iot hub hostname
3         static string deviceKey = "+jDqO+Nu2g************="; //device key

添加一个循环向Azure IoT Hub发送消息的方法:SendDeviceToCloudMessagesAsync,1s 一条消息

 1         /// <summary>
 2         /// 循环向Azure IoT Hub发送消息
 3         /// </summary>
 4         private static async void SendDeviceToCloudMessagesAsync()
 5         {
 6             double avgWindSpeed = 10; // m/s
 7             Random rand = new Random();
 8 
 9             while (true)
10             {
11                 //发送遥测数据
12                 double currentWindSpeed = avgWindSpeed + rand.NextDouble() * 4 - 2;
13                 var telemetryDataPoint = new
14                 {
15                     deviceId = "TeldPile001",
16                     windSpeed = currentWindSpeed
17                 };
18                 var messageString = JsonConvert.SerializeObject(telemetryDataPoint);
19                 var message = new Message(Encoding.ASCII.GetBytes(messageString));
20 
21                 await deviceClient.SendEventAsync(message);
22                 Console.WriteLine("{0} > Sending message: {1}", DateTime.Now, messageString);
23 
24                 //1s 一条
25                 await Task.Delay(1000);
26             }
27         }

然后,在Main函数中启动模拟设备发送消息:

1         static void Main(string[] args)
2         {
3             Console.WriteLine("模拟设备通信...\n");
4             deviceClient = DeviceClient.Create(iotHubUri, new DeviceAuthenticationWithRegistrySymmetricKey("TeldPile001", deviceKey), TransportType.Mqtt);
5 
6             SendDeviceToCloudMessagesAsync();
7             Console.ReadLine();
8         }

 3. 启动运行测试

在解决方案上设置双启动项目:Device和IoTServer

F5 Run:

可以发现,设备侧消息发送、Azure IoT Hub接收是同步的

我们查看Azure Portal中的统计:

 

总结: 通过这两篇博文,我们研究验证了Azure IoT Hub 注册设备、设备和云之间的通信,感觉整个Azure 的 IoT Hub还是非常好用、易用,比较容易理解和操作,基于PaaS层的IoT Hub,可以做很多有价值的设计和方案。

 

周国庆

2017/4/18

posted @ 2017-04-18 09:19  Eric zhou  阅读(2493)  评论(0编辑  收藏  举报