163 lines
5.5 KiB
Markdown
163 lines
5.5 KiB
Markdown
|
# Отчёт по лабораторной работе №4
|
|||
|
## Работа с брокером сообщений
|
|||
|
|
|||
|
Выполнила: студентка гр. ИСЭбд-41 Халитова А.М.
|
|||
|
|
|||
|
## Установка брокера сообщений RabbitMQ
|
|||
|
Была выполнена установка RabbitMQ и Erlang:
|
|||
|
data:image/s3,"s3://crabby-images/400d9/400d96c5a3c4079b94d76c199cba8a2cdffdaa23" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/9e416/9e4169596cfb0d5a8170ac704c8706d9d244486e" alt=""
|
|||
|
|
|||
|
И выполнен вход на http://localhost:15672/ в качестве гостя:
|
|||
|
data:image/s3,"s3://crabby-images/0e70c/0e70ce9f4364f7c4e72d9779ace632c6503accda" alt=""
|
|||
|
|
|||
|
## Прохождение tutorial
|
|||
|
|
|||
|
tutorial1:
|
|||
|
data:image/s3,"s3://crabby-images/6b847/6b8471c270f17ff415f3a1ad2be9b5b1fe57fc5b" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/21e30/21e309f95f4097438537551fbb1412447d1c72df" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/90aa5/90aa5a6aadd897425368aba9a2d32fa1bbfc7fc2" alt=""
|
|||
|
|
|||
|
tutorial2:
|
|||
|
data:image/s3,"s3://crabby-images/00c28/00c28231b9ac9396fd23521a867b1237686d6941" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/38185/38185ea88d4497ef36bfecad8e5a0c197a44f4b4" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/6b792/6b792026b3b37f237b8034e70cfb084eb6830e9b" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/6e4fa/6e4fad1e059b073824ae80a9b4e11c65ef3a83d7" alt=""
|
|||
|
|
|||
|
tutorial3:
|
|||
|
data:image/s3,"s3://crabby-images/d6d14/d6d147fd99b0699860666305d7d17ff800551669" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/b8c8b/b8c8b148d487760af8269ffe753c644ce1ef05f8" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/a52f4/a52f403b328381223a59e19c49c3e2db046e0971" alt=""
|
|||
|
|
|||
|
## Разработка демонстрационных приложений
|
|||
|
|
|||
|
Предметная область - заказ работ на исполнение. Производитель - заказчик работ. Потребитель 1 - исполнитель принимает заказ и выполняет работу. Потребитель 2 - исполнитель откладывает выполнение задачи.
|
|||
|
|
|||
|
1. Publisher:
|
|||
|
```
|
|||
|
using System.Text;
|
|||
|
using RabbitMQ.Client;
|
|||
|
|
|||
|
var factory = new ConnectionFactory { HostName = "localhost" };
|
|||
|
using var connection = factory.CreateConnection();
|
|||
|
using var channel = connection.CreateModel();
|
|||
|
|
|||
|
channel.ExchangeDeclare(exchange: "job", type: ExchangeType.Fanout);
|
|||
|
Random rand = new Random();
|
|||
|
foreach (var item in Enumerable.Range(0, 1000))
|
|||
|
{
|
|||
|
var message = rand.Next().ToString();
|
|||
|
|
|||
|
var body = Encoding.UTF8.GetBytes(message);
|
|||
|
channel.BasicPublish(exchange: "job",
|
|||
|
routingKey: string.Empty,
|
|||
|
basicProperties: null,
|
|||
|
body: body);
|
|||
|
|
|||
|
Console.WriteLine($" [x] Поступила работа {message}");
|
|||
|
await Task.Delay(500);
|
|||
|
}
|
|||
|
|
|||
|
Console.WriteLine(" Press [enter] to exit.");
|
|||
|
Console.ReadLine();
|
|||
|
|
|||
|
static string GetMessage(string[] args)
|
|||
|
{
|
|||
|
return ((args.Length > 0) ? string.Join(" ", args) : "info: Принято!");
|
|||
|
}
|
|||
|
```
|
|||
|
2. Consumer 1:
|
|||
|
```
|
|||
|
using System.Text;
|
|||
|
using RabbitMQ.Client;
|
|||
|
using RabbitMQ.Client.Events;
|
|||
|
|
|||
|
var factory = new ConnectionFactory { HostName = "localhost" };
|
|||
|
using var connection = factory.CreateConnection();
|
|||
|
using var channel = connection.CreateModel();
|
|||
|
|
|||
|
channel.QueueDeclare(queue: "accepted");
|
|||
|
channel.QueueBind(queue: "accepted",
|
|||
|
exchange: "job",
|
|||
|
routingKey: string.Empty);
|
|||
|
|
|||
|
Console.WriteLine(" [*] Waiting for messages.");
|
|||
|
|
|||
|
var consumer = new EventingBasicConsumer(channel);
|
|||
|
consumer.Received += async (model, ea) =>
|
|||
|
{
|
|||
|
byte[] body = ea.Body.ToArray();
|
|||
|
var message = Encoding.UTF8.GetString(body);
|
|||
|
|
|||
|
string outputText = $"Работа принята к исполнению {message}";
|
|||
|
Console.WriteLine($" [x] Done. {outputText}");
|
|||
|
channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);
|
|||
|
};
|
|||
|
channel.BasicConsume(queue: "accepted",
|
|||
|
autoAck: false,
|
|||
|
consumer: consumer);
|
|||
|
|
|||
|
Console.WriteLine(" Press [enter] to exit.");
|
|||
|
Console.ReadLine();
|
|||
|
```
|
|||
|
|
|||
|
3. Consumer 3:
|
|||
|
```
|
|||
|
using System.Text;
|
|||
|
using RabbitMQ.Client;
|
|||
|
using RabbitMQ.Client.Events;
|
|||
|
|
|||
|
var factory = new ConnectionFactory { HostName = "localhost" };
|
|||
|
using var connection = factory.CreateConnection();
|
|||
|
using var channel = connection.CreateModel();
|
|||
|
|
|||
|
Random rand = new Random();
|
|||
|
string queueName = $"denied{rand.Next()}";
|
|||
|
|
|||
|
channel.QueueDeclare(queue: queueName);
|
|||
|
channel.QueueBind(queue: queueName,
|
|||
|
exchange: "job",
|
|||
|
routingKey: string.Empty);
|
|||
|
|
|||
|
Console.WriteLine(" [*] Waiting for messages.");
|
|||
|
|
|||
|
var consumer = new EventingBasicConsumer(channel);
|
|||
|
consumer.Received += async (model, ea) =>
|
|||
|
{
|
|||
|
byte[] body = ea.Body.ToArray();
|
|||
|
var message = Encoding.UTF8.GetString(body);
|
|||
|
|
|||
|
int waitTime = rand.Next(20, 90);
|
|||
|
Thread.Sleep(waitTime * 100);
|
|||
|
|
|||
|
string outputText = $"Работа отложена {message} на {waitTime} минут";
|
|||
|
Console.WriteLine($" [x] Done. {outputText}");
|
|||
|
channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);
|
|||
|
};
|
|||
|
channel.BasicConsume(queue: queueName,
|
|||
|
autoAck: false,
|
|||
|
consumer: consumer);
|
|||
|
|
|||
|
Console.WriteLine(" Press [enter] to exit.");
|
|||
|
Console.ReadLine();
|
|||
|
```
|
|||
|
|
|||
|
## Результаты выполнения работ
|
|||
|
|
|||
|
Запуск каждой программы по одному экземпляру:
|
|||
|
data:image/s3,"s3://crabby-images/b03f4/b03f440b11d4fcdcff9ac0b6a5cac1aafaf8a6c5" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/6ad9e/6ad9ecccd074c8b6e78cc9d62a3ab486b83d587d" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/a5d45/a5d458e3332858a7b43c2fca98cf9b0654bfe525" alt=""
|
|||
|
|
|||
|
Результаты обработки:
|
|||
|
data:image/s3,"s3://crabby-images/477d4/477d4806add91439c9fe79e7e47eac2069c98622" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/fb866/fb866b87d5fc6cab539392e6dd0c2009f895f751" alt=""
|
|||
|
|
|||
|
Потребителю один соответствует очередь accepted, а потребителю два - очередь denied. Как видно, первый потребитель работает без задержки, а очередь второго потребителя переполенена. Из этого следует, что скорость первого в разы выше, и он потребляет намного меньше памяти за счет постоянно пустой очереди.
|
|||
|
|
|||
|
Запуск двух экземпляров потребителя два и по одному экземпляру потребителя один и заказчика:
|
|||
|
data:image/s3,"s3://crabby-images/cb22e/cb22e640a69a92b10619cf9e802856701054080d" alt=""
|
|||
|
data:image/s3,"s3://crabby-images/38ee5/38ee5cc3f9741c6b88c5b6cbfa3f178793d1e3da" alt=""
|
|||
|
|
|||
|
Ситуация в очередью потребителя один не изменилась, а экземпляры потребителя два теперь затрачивают в 2 раза больше времени и памяти, их очереди так же переполнены, как и в ситуации выше.
|