Форум программистов, компьютерный форум, киберфорум
C# .NET
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.90/29: Рейтинг темы: голосов - 29, средняя оценка - 4.90
25 / 24 / 18
Регистрация: 16.10.2009
Сообщений: 1,163

Tcp сервер - как отследить отключение клиента?

22.02.2019, 15:45. Показов 5831. Ответов 7
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Уважаемые Гуру!
Попробовал запустить тестовый tcp сервер. Tcp-клиенты подключаются так

C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
        
 
 List<TcpClient> clientsLst = new List<TcpClient>();   
 void AcceptClients()
        {
            while (true)
            {
                try
                {
                    TcpClient client = _server.AcceptTcpClient();
                    clientsLst.Add(client);
                    int clientIndex = clientsLst.IndexOf(client);
                    Thread readThread = new Thread(ReceiveRun);
                    readThread.Start(clientIndex);
                    Invoke(new UpdateClientsDisplayDelegate(UpdateClientsDisplay));  // обновим в осн потоке гл формы
                }
                catch
                {
                    ErrorSound(); 
                }
                if (_stopNetwork == true) 
                {
                    break;
                }
            }
        }
Видимо в результате каких то ошибок, список за сутки увеличился до 80 подключений.
Задача - удалять "отвалившиеся" соединения.
Предполагаю в основном потоке старт потока обработки подключения заменить на


C#
1
readThread.Start(client);
т.е. передавать самого TcpClient-а и сделать в основном потоке словарь, в котором будет храниться время получения очередного сообщения

C#
1
        Dictionary<TcpClient, DateTime> connectionDictionary;
В котором будет отслеживаться то, что коннект не закрылся на стороне клиента. Если время отсутствия пакетов превысит
скажем 10 минут, то это соединение закрывается на стороне сервера и этот клиент из списка удаляется.

C#
1
2
3
4
5
6
7
8
        // Вызывается из клиентского потока для отображения заголовка пришедшего пакета от клиента
        public void UpdateReceiveDisplay(TcpClient client, string message)
        {
            listBox1.Items.Add(DateTime.Now.ToShortTimeString() + " : " + message);            
            connectionDictionary[client] = DateTime.Now;
           // writeLog(clientnum, message);
        }
        protected delegate void UpdateReceiveDisplayDelegate(int clientcount, string message);       // Делегат доступа к элементу формы listBox1 из вспомогательного потока.
В основном потоке по таймеру, раз в 10 минут проверяется приходили ли пакеты по данному соединению.
Если нет, то соединение закрывается

C#
1
2
3
4
5
if(client != null){
 
  client.Close();
  clientsLst.Remove(client);
  connectionDictionary.Remove(client);
Такой подход годится?
И что может дать функция client.Connected ?
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
22.02.2019, 15:45
Ответы с готовыми решениями:

Протокол TCP. Передать число с клиента на сервер
Привет всем. Нужно передать число с клиента на сервер, чтобы возвращался синус этого числа. Код клиента: using System; using...

TCP прием от клиента: Сервер считывает 21 байт не за раз, а за 7 циклов
Превью. Лог от сервера И так. От клиента поступают запросы. в среднем от 20 до 30 байт. Первый запрос приходит успешно....

ServerSocket отследить аварийное отключение клиента
Здравствуете! Пытаюсь сделать текстовый с одним сервером и некоторым числом клиентов, подключающихся к этому серверу. Работает это всё...

7
Эксперт .NETАвтор FAQ
 Аватар для Storm23
10425 / 5155 / 1825
Регистрация: 11.01.2015
Сообщений: 6,226
Записей в блоге: 34
22.02.2019, 19:21
Лучший ответ Сообщение было отмечено АТерентьев как решение

Решение

Цитата Сообщение от АТерентьев Посмотреть сообщение
передавать самого TcpClient-а и сделать в основном потоке словарь, в котором будет храниться время получения очередного сообщения
Не нужно никаких словарей.
Создайте класс Client, который будет хранить все данные, относящиеся к клиенту: его TcpClient, время последнего обмена данными и другие данные, если они будут.
C#
1
2
3
4
5
class Client
{
     public TcpClient TcpClient { get; set; }
     public DateTime LastContactTime { get; set; }
}
Этот класс и передавайте в readThread.Start().

Далее, создайте ConcurrentBag<Client> в котором вы будете хранить список клиентов. Не используйте List<Client> потому что у вас будет доступ к этому списку из разных потоков.

По таймеру - пробегайтесь по списку клиентов, найдите подвисших и удаляйте их.

Да и еще добавьте в класс Client флаг Stopped, что бы цикл принятия сообщений (или цикл отправки данных) узнали, что пора завязывать.
1
25 / 24 / 18
Регистрация: 16.10.2009
Сообщений: 1,163
25.02.2019, 10:24  [ТС]
Спасибо! Не ясно только маленько.
В текущей версии в readThread.Start передавался номер в массиве TcpClient-ов. Там по номеру вытаскивался сам TcpClient,
причем из разных потоков. Почему это работает без ConcurrentBag?

Добавлено через 1 час 29 минут
Не увидел пока как задумано удалять конкретный элемент из ConcurrentBug.

https://stackoverflow.com/ques... currentbag

Предлагается использовать
https://docs.microsoft.com/en-... work-4.7.2


Вроде словарь нормальное решение? и в классе Client хранить ключ в этом словаре
0
Эксперт .NETАвтор FAQ
 Аватар для Storm23
10425 / 5155 / 1825
Регистрация: 11.01.2015
Сообщений: 6,226
Записей в блоге: 34
25.02.2019, 10:56
Лучший ответ Сообщение было отмечено АТерентьев как решение

Решение

Цитата Сообщение от АТерентьев Посмотреть сообщение
В текущей версии в readThread.Start передавался номер в массиве TcpClient-ов. Там по номеру вытаскивался сам TcpClient,
причем из разных потоков. Почему это работает без ConcurrentBag?
Потому что
1) Вы только читаете данные из массива. Эффекты же многопоточности обычно сказываются тогда, когда происходит одновременное чтение и запись данных из разных потоков. Когда вы начнете удалять элементы из списка, вот тогда и начнутся глюки.
2) Ошибки при многопточности могут возникать не сразу. Оно может проявляться очень редко.

Цитата Сообщение от АТерентьев Посмотреть сообщение
Вроде словарь нормальное решение? и в классе Client хранить ключ в этом словаре
Нет, не нужно словарь. Используйте просто List<Client>, но при доступе к нему используйте lock.
1
25 / 24 / 18
Регистрация: 16.10.2009
Сообщений: 1,163
25.02.2019, 11:25  [ТС]
Цитата Сообщение от Storm23 Посмотреть сообщение
Нет, не нужно словарь. Используйте просто List<Client>, но при доступе к нему используйте lock.
Т.е. вы предлагаете вместо ConcurrentBag использовать List<Client> c lock?
Я правильно понял?
0
Эксперт .NETАвтор FAQ
 Аватар для Storm23
10425 / 5155 / 1825
Регистрация: 11.01.2015
Сообщений: 6,226
Записей в блоге: 34
25.02.2019, 13:59
Лучший ответ Сообщение было отмечено АТерентьев как решение

Решение

Цитата Сообщение от АТерентьев Посмотреть сообщение
Т.е. вы предлагаете вместо ConcurrentBag использовать List<Client> c lock?
Я правильно понял?
Да, правильно. Я просто забыл, что из ConcurrentBag нельзя удалять элементы. Значит он вам не подходит. А заводить словарь - это уже избыточно.
1
25 / 24 / 18
Регистрация: 16.10.2009
Сообщений: 1,163
25.02.2019, 14:24  [ТС]
Вот код получения сообщений:

C#
1
2
3
4
5
6
7
8
using (var br = new BinaryReader(ns))
     while (true)
         {
             try
                   {
                        HEADER = (byte[])br.ReadBytes(11);//read 11 bytes to get  packet size
                        if (HEADER.Length == 0)
                                  continue ;// Может быть в этом случае нужно закрывать коннект?
Очень редко , но иногда при чтении заголовка возвращается "0" байт вместо заказанных 11
Значит ли это что коннект разорван на стороне клиента ?

P.S. Некоторые версии их серверов отправляют сообщения с правильным заголовком (11 байт), но нулевой
длиной данных прикладного уровня. Это не ошибка, просто чтобы показать что коннект "жив", по видимому. В новых версиях "Для этого настраивается стандартный keev alive таймер."
0
Эксперт .NETАвтор FAQ
 Аватар для Storm23
10425 / 5155 / 1825
Регистрация: 11.01.2015
Сообщений: 6,226
Записей в блоге: 34
25.02.2019, 14:27
Цитата Сообщение от АТерентьев Посмотреть сообщение
Очень редко , но иногда при чтении заголовка возвращается "0" байт вместо заказанных 11
Значит ли это что коннект разорван на стороне клиента ?
Да, пакет нулевой длины - это стандартное сообщение о том, что противоположная сторона закрыла соединение.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
25.02.2019, 14:27
Помогаю со студенческими работами здесь

Как из TCP пакета пришедшего на сервер выделить IP клиента
Как из TCP пакета пришедшего на сервер выделить IP клиента?? Где в передаваемом пакете на сервер будет находится ip клиента вначале или в...

TCP Сервер не принимает данные от клиента
Я использую Ubuntu 16.04 (telnet есть) и Raspberry Pi Zero W. На Ubuntu сервер на Qt, а на RPi клиент. Я на писал сервер на Qt, но он...

Передать файл по TCP протоколу от клиента на сервер
Товарищи прошу вас помощи, задача - необходимо передать файл по TCP протоколу от клиента на сервер. Самому писать долго, может есть уже...

Сцена зависает при запуске TCP-клиента, когда он подключен к TCP - серверу, при этом TCP-клиент полностью функционирует
Проблема описана в заголовке, и хотелось бы услышать ваше мнение, о том как можно решить проблему. Скрипт TCP-клиента на сцене: ...

как создать TCP клиент, TCP сервер ? На С++
Очень нужна помощь!Как написать TCP клиент, TCP сервер. Например,клиент вводит строку с клавиатуры и отсылает ее серверу.только перед...


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

Или воспользуйтесь поиском по форуму:
8
Ответ Создать тему
Новые блоги и статьи
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
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru