С Новым годом! Форум программистов, компьютерный форум, киберфорум
C# .NET
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.54/13: Рейтинг темы: голосов - 13, средняя оценка - 4.54
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37

Как избежать зависания на Socket.Send()?

09.01.2019, 16:22. Показов 2790. Ответов 11

Студворк — интернет-сервис помощи студентам
Вообщем при большом трафике на сервере (C# Mono Ubuntu 16.04) как я понял если клиент отключается в момент отправки пакета, то весь поток зависает на Socket.Send() бывает 2 исхода либо выкидывается исключение и сервер его успешно обрабатывает либо поток намертво виснет. Знаю, при использовании голых UNIX сокетов (C++) есть такая же проблема, но она легко решается добавлением флага MSG_NOSIGNAL, есть ли в C# какие-то похожие механизмы?
P.S. Асинхронные сокеты не предлагать

C#
1
2
3
4
5
6
7
8
9
        private void sendBytes(byte[] bf)
        {
            if (_userSocket == null || !_userSocket.Connected)
                return;
 
 
            lock (_socket_sync)
                _userSocket.Send(bf, 0, bf.Length, SocketFlags.None);
        }
Функция отправки пакетов пользователю трафик на сервере (10-11к вход. / 9-10к исход.) пакетов в секунду
Средний размер пакета около 40 байт ~ (передвижение)
0
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
09.01.2019, 16:22
Ответы с готовыми решениями:

Socket Send и Recieve
private void update_table() { Stopwatch sw = new Stopwatch(); sw.Start(); string status...

Send struct from socket
Здравствуйте, помогите пожалуйста отправить структуру на ++... typedef struct { unsigned int size; unsigned int cmd; ...

Socket.Send.Receive - Сообщение делится на фрагменты произвольной длины
клиент шлет массив байт(с <eof>), на сервере дописывается имя клиента в начало массива и возвращается клиенту. Почему сообщение не...

11
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37
13.01.2019, 17:12  [ТС]
Проблема решается либо использованием асинхронных сокетов Socket.SendAsync() (дорого но решает проблему) либо
добавлением тайм аута на отправку socket.SendTimeout = 2500 (мс.) выяснилось, что в C# по дефолту в тайм-ауте стоит значение 0 т.е. бесконечно ожидает отправки сообщения и в случае какой то ошибки отправка просто намертво вешает поток, при превышении же тайм аута выбрасывается соответствующее исключение и его можно штатно обработать.

Может быть кому-то будет полезно
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,573
13.01.2019, 18:54
Цитата Сообщение от WODICHKA Посмотреть сообщение
дорого но решает проблему
В смысле дорого? По моему самое оптимальное решение.
0
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37
13.01.2019, 20:00  [ТС]
Как показали тесты конструкция
C#
1
2
3
4
5
                SocketAsyncEventArgs sendEvent = new SocketAsyncEventArgs();
                sendEvent.SetBuffer(resultBuffer, 0, resultLenght);
 
                lock (_userSocket)
                    _userSocket.SendAsync(sendEvent);
В разы дороже чем

C#
1
2
                lock (_userSocket)
          _userSocket.Send(buffer, 0, buffLen);
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,573
13.01.2019, 21:24
Цитата Сообщение от WODICHKA Посмотреть сообщение
дороже
В каком плане дороже?

Насколько я знаю SendAsync/ReceiveAsync самый производительный способ работы с сокетами на C# под .NET Framework (на .NET Core есть решения получше, например System.IO.Pipelines).

При этом объекты SocketAsyncEventArgs должны переиспользоваться, а не создаваться новый на каждой операции приема или отправки.
0
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37
13.01.2019, 22:47  [ТС]
Цитата Сообщение от Someone007 Посмотреть сообщение
В каком плане дороже?

Насколько я знаю SendAsync/ReceiveAsync самый производительный способ работы с сокетами на C# под .NET Framework (на .NET Core есть решения получше, например System.IO.Pipelines).

При этом объекты SocketAsyncEventArgs должны переиспользоваться, а не создаваться новый на каждой операции приема или отправки.
Разумеется во время тестов я переиспользовал объект, сервер показал работу хуже при использовании асинхронного варианта.

P.S. Не особо понимаю как асинхронная операция может быть производительней блокирующей это всё равно что говорить
число A больше чем A + 1
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,573
13.01.2019, 23:11
Цитата Сообщение от WODICHKA Посмотреть сообщение
как асинхронная операция может быть производительней блокирующей
На то она и асинхронная, чтобы не было лишних ожиданий как в блокирующем варианте, поэтому должна быть намного производительней.
0
 Аватар для IamRain
4694 / 2702 / 734
Регистрация: 02.08.2011
Сообщений: 7,228
13.01.2019, 23:18
Цитата Сообщение от Someone007 Посмотреть сообщение
чтобы не было лишних ожиданий как в блокирующем варианте,
В связи с этим я не понимаю, зачем нужен lock в коде ТС-а:
C#
1
2
lock (_userSocket)
  _userSocket.SendAsync(sendEvent);
Критичного участка кода, ятп нету ведь, к которому нужен монопольный доступ. - отправил и забыл, не?
0
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37
13.01.2019, 23:21  [ТС]
Цитата Сообщение от Someone007 Посмотреть сообщение
На то она и асинхронная, чтобы не было лишних ожиданий как в блокирующем варианте, поэтому должна быть намного производительней.
Не буду врать плохо знаю C# в основном мой инструмент C++, не знаю как устроенны асинхронные операции в C#, но так или иначе она либо создает дополнительный поток (что крайне дорого) для выполнения задачи либо пушит эту задачу в какой-то ThreadPoll что тоже не дешего потому что редко такие вещи обходятся без синхронизации.
А задача которую она пушит и есть блокирующая операция.

Могу ошибаться, но примерно как-то так.

P.S. То что я назвал "ухудшением работы сервера", было повышение использования CPU,и не значительное увеличение использования RAM.

Добавлено через 2 минуты
Цитата Сообщение от IamRain Посмотреть сообщение
В связи с этим я не понимаю, зачем нужен lock в коде ТС-а:
C#
1
2
lock (_userSocket)
  _userSocket.SendAsync(sendEvent);
Критичного участка кода, ятп нету ведь, к которому нужен монопольный доступ. - отправил и забыл, не?
Это не код сервера
Я просто скопировал лишнее
0
 Аватар для IamRain
4694 / 2702 / 734
Регистрация: 02.08.2011
Сообщений: 7,228
13.01.2019, 23:25
Цитата Сообщение от WODICHKA Посмотреть сообщение
Это не код сервера
Да неважно, надо смотреть по факту весь ваш код, чтобы посмотреть, почему Async, по вашему мнению, остался виноватым.
0
5 / 5 / 0
Регистрация: 26.03.2015
Сообщений: 37
13.01.2019, 23:32  [ТС]
Цитата Сообщение от IamRain Посмотреть сообщение
Да неважно, надо смотреть по факту весь ваш код, чтобы посмотреть, почему Async, по вашему мнению, остался виноватым.
Весь код функции отправки предоставлен в самом первом сообщении там 3 строчки.
Функцию sendBytes() вызывает функция SendPacket() вызовов которой в коде аш 846
C#
1
2
3
4
        public void SendPacket(Packet p)
        {
            sendBytes(p.PacketBuffer);
        }
C#
1
2
3
4
5
6
7
8
9
10
11
12
public class Packet
{
        public DateTime TimeGet;
 
        public byte[] PacketBuffer;
        public PacketIDs ID;
        public int Size;
        public bool Encrypted;
        public int PacketNumber;
 
// methods
}
P.S. Не считаю Async виноватым, просто при использовании Async приложение потребляло больше ресурсов, а это очень критично учесть что функция используется в 846 местах
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,573
13.01.2019, 23:51
Тот факт что потребляется больше ресурсов совершенно не означает что Async хуже работает. При работе с сетью важны и другие параметры, например latency (тут имеется в виду как быстро ваш сервер обрабатывает приходящие пакеты и отправляет на них ответы, а не latency из-за расстояния между сервером и клиентом), пропускная способность и т.д.

Например повышенное использование процессора можно объяснить тем, что асинхронный код способен обрабатывать больше пакетов за единицу времени... А повышенное потребление памяти кешированием тех же SocketAsyncEventArgs...
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
13.01.2019, 23:51
Помогаю со студенческими работами здесь

Проблема с методом Send у класса System.Net.Sockets.Socket
Привет всем! Вот фрагмент кода: private Socket FSocket; protected override void OnProcess(object message) { try { ...

Сервер продолжает получать пустые сообщения после вызова метода Socket.Send
Есть сервер с сокетом, который выдает ответы на запросы. Связь с ним нормально работает через модем. Необходимо подключиться к нему с...

Socket.Send + Async & Await: свой асинхронный метод отправки данных
Привет. Хочу написать свой асинх. метод для отправки данных при помощи класса Socket и его метода Send. Работать должно так. Есть некий...

Вылет при socket.send()
Когда вызываю сокет.send или BeginSend, то прога тупо вылетает, без ошибок и eceptionoв. Если поставить брекпоинт, то доходит до }...

Как избежать зависания формы при использовании n потоков?
Здравствуйте. У меня есть код, в котором выполняется ёмкий цикл, чтобы не зависала форма я его запускаю отдельным потоком(bw), но если я...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
12
Ответ Создать тему
Новые блоги и статьи
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост.
Programma_Boinc 28.12.2025
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост. Налог на собак: https:/ / **********/ gallery/ V06K53e Финансовый отчет в Excel: https:/ / **********/ gallery/ bKBkQFf Пост отсюда. . .
Кто-нибудь знает, где можно бесплатно получить настольный компьютер или ноутбук? США.
Programma_Boinc 26.12.2025
Нашел на реддите интересную статью под названием Anyone know where to get a free Desktop or Laptop? Ниже её машинный перевод. После долгих разбирательств я наконец-то вернула себе. . .
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка.
Programma_Boinc 23.12.2025
Рецензия / Мнение/ Перевод Нашел на реддите интересную статью под названием The Thinkpad X220 Tablet is the best budget school laptop period . Ниже её машинный перевод. Thinkpad X220 Tablet —. . .
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru