25 / 24 / 18
Регистрация: 16.10.2009
Сообщений: 1,163

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

22.02.2019, 15:45. Показов 5859. Ответов 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
Ответ Создать тему
Опции темы

Новые блоги и статьи
Модель микоризы: классовый агентный подход 3
anaschu 06.01.2026
aa0a7f55b50dd51c5ec569d2d10c54f6/ O1rJuneU_ls https:/ / vkvideo. ru/ video-115721503_456239114
Owen Logic: О недопустимости использования связки «аналоговый ПИД» + RegKZR
ФедосеевПавел 06.01.2026
Owen Logic: О недопустимости использования связки «аналоговый ПИД» + RegKZR ВВЕДЕНИЕ Введу сокращения: аналоговый ПИД — ПИД регулятор с управляющим выходом в виде числа в диапазоне от 0% до. . .
Модель микоризы: классовый агентный подход 2
anaschu 06.01.2026
репозиторий https:/ / github. com/ shumilovas/ fungi ветка по-частям. коммит Create переделка под биомассу. txt вход sc, но sm считается внутри мицелия. кстати, обьем тоже должен там считаться. . . .
Расчёт токов в цепи постоянного тока
igorrr37 05.01.2026
/ * Дана цепь постоянного тока с сопротивлениями и напряжениями. Надо найти токи в ветвях. Программа составляет систему уравнений по 1 и 2 законам Кирхгофа и решает её. Последовательность действий:. . .
Новый CodeBlocs. Версия 25.03
palva 04.01.2026
Оказывается, недавно вышла новая версия CodeBlocks за номером 25. 03. Когда-то давно я возился с только что вышедшей тогда версией 20. 03. С тех пор я давно снёс всё с компьютера и забыл. Теперь. . .
Модель микоризы: классовый агентный подход
anaschu 02.01.2026
Раньше это было два гриба и бактерия. Теперь три гриба, растение. И на уровне агентов добавится между грибами или бактериями взаимодействий. До того я пробовал подход через многомерные массивы,. . .
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост.
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? Ниже её машинный перевод. После долгих разбирательств я наконец-то вернула себе. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru