реклама на сайте
подробности

 
 
2 страниц V   1 2 >  
Reply to this topicStart new topic
> VS2010 прием по TCP, доступ к главной форме из потока
smk
сообщение Nov 23 2015, 20:35
Сообщение #1


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Цитата
Недопустимая операция в нескольких потоках: попытка доступа к элементу управления 'chart1' не из того потока, в котором он был создан.

В приложении, которое мне необходимо для работы нужно принимать пакет данных по ТСП и "распихивать" их по элементам главной формы. Прием ведется в отдельном потоке:
Код
                Thread th = new Thread(ReceiveRun);
                th.Start();


        // Цикл извлечения сообщений,
        // запускается в отдельном потоке.
        void ReceiveRun()
        {
            while (true)
            {
                try
                {
                    string s = null;
                    while (ns.DataAvailable == true)
                    {
                        //// Определение необходимого размера буфера приема.
                        //byte[] buffer = new byte[_tcpСlient.Available];

                        ns.Read(inBuffer, 0, inBuffer.Length);
                        s += Encoding.Default.GetString(inBuffer);
                    }

                    if (s != null)
                    {
                        ShowReceiveMessage(s);
                        s = String.Empty;
                    }


                    // Вынужденная строчка для экономия ресурсов процессора.
                    // Неизящный способ.
                    Thread.Sleep(100);
                }
                catch
                {
                    ErrorSound();
                }

                if (_stopNetwork == true) break;

            }
        }



При попытке получить пакет появляется ошибка, которую я привел в начале этого сообщения. Ошибка появляется при обработке функции отображения данных:

Код
       // Код доступа к свойствам объектов главной формы  из других потоков
        delegate void UpdateReceiveDisplayDelegate(string message);
        void ShowReceiveMessage(string message)
        {
            if (chart1.InvokeRequired == true)
            {
                UpdateReceiveDisplayDelegate rdd = new UpdateReceiveDisplayDelegate(ShowReceiveMessage);

                // Данный метод вызывается в дочернем потоке,
                // ищет основной поток и выполняет делегат указанный в качестве параметра
                // в главном потоке, безопасно обновляя интерфейс формы.
                Invoke(rdd, new object[] { message });
                chart1.Series[0].Points.Clear();
                chart1.Series[1].Points.Clear();
                chart1.Series[2].Points.Clear();
                chart1.Series[3].Points.Clear();
                chart1.Series["1,3 мкм"].Points.AddY(inBuffer[1] << 8 + inBuffer[0]);
                chart1.Series["2,11 мкм"].Points.AddY(inBuffer[3] << 8 + inBuffer[2]);
                chart1.Series["1,8 мкм"].Points.AddY(inBuffer[5] << 8 + inBuffer[4]);
                chart1.Series["1,93 мкм"].Points.AddY(inBuffer[7] << 8 + inBuffer[6]);
            }
            else
            {
                // Если не требуется вызывать метод Invoke, обратимся напрямую к элементу формы.
                chart1.Series[0].Points.Clear();
                chart1.Series[1].Points.Clear();
                chart1.Series[2].Points.Clear();
                chart1.Series[3].Points.Clear();
                chart1.Series["1,3 мкм"].Points.AddY(inBuffer[1] << 8 + inBuffer[0]);
                chart1.Series["2,11 мкм"].Points.AddY(inBuffer[3] << 8 + inBuffer[2]);
                chart1.Series["1,8 мкм"].Points.AddY(inBuffer[5] << 8 + inBuffer[4]);
                chart1.Series["1,93 мкм"].Points.AddY(inBuffer[7] << 8 + inBuffer[6]);
            }


        }


Как правильно поступить с выводом информации? Прошу помочь. Спасибо.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
XVR
сообщение Nov 25 2015, 06:37
Сообщение #2


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



У вас в ShowReceiveMessage жуткая мешанина из того, что нужно вызывать в потоке приема и того, что нужно вызывать в GUI потоке.

В потоке приема нужно вызывать ТОЛЬКО Invoke от делегата (от ShowReceiveMessage), а в самой ShowReceiveMessage уже обновлять всю GUI часть
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 25 2015, 08:34
Сообщение #3


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Цитата(XVR @ Nov 25 2015, 08:37) *
У вас в ShowReceiveMessage жуткая мешанина из того, что нужно вызывать в потоке приема и того, что нужно вызывать в GUI потоке.

В потоке приема нужно вызывать ТОЛЬКО Invoke от делегата (от ShowReceiveMessage), а в самой ShowReceiveMessage уже обновлять всю GUI часть

Спасибо, что откликнулись. Можно попросить уточнить (на пальцах) что куда перенести? Прошу простить, не силен в этой стороне программирования и прибегаю только когда действительно не обойтись. В идеале если есть у Вас какая-то "коза" с правильной организацией вопроса, то был бы очень признателен. А вообще хотелось бы так чтобы в одном потоке происходил прием, а вдругом обработка. Пришло что-то - выставили флаг. В обработчике увидели флаг - обработали что пришло. Как правильно сделать тсп-клиент под это я не знаю. Потому нужна помощь. С остальным, думаю разберусь. Спасибо.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
XVR
сообщение Nov 25 2015, 10:15
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Конструкцию UpdateReceiveDisplayDelegate rdd = new UpdateReceiveDisplayDelegate(ShowReceiveMessage); вынести из ShowReceiveMessage и поместить в начало ReceiveRun
Конструкцию Invoke(rdd, new object[] { message }); вынести из ShowReceiveMessage и пометить в ReceiveRun вместо вызова ShowReceiveMessage
Прямые обращения к inBuffer в ShowReceiveMessage заменить на обращения к ее декодированному содержимому, декодировать его из параметра message
В ShowReceiveMessage ветку под if (chart1.InvokeRequired == true) убрать
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 25 2015, 11:29
Сообщение #5


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Благодарю. Сделаю как Вы сказали. Еще раз спасибо.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 25 2015, 14:25
Сообщение #6


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Цитата(XVR @ Nov 25 2015, 12:15) *
Конструкцию Invoke(rdd, new object[] { message }); вынести из ShowReceiveMessage и пометить в ReceiveRun вместо вызова ShowReceiveMessage

Сделал. Пишет, что "Элемент "message" не существует в текущем контексте." Можно с этим что-то сделать? Спасибо.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
XVR
сообщение Nov 25 2015, 14:36
Сообщение #7


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



В контексте ReceiveRun этот элемент называется s. Т.е. надо Invoke(rdd, new object[] { s });
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 28 2015, 09:42
Сообщение #8


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Пока получается. Остается вопрос по кнопке "отсоединиться". Отсоединиться получается, но потом не соединяется т.к. элемент _tcpСlient ликвидирован. Что можно сделать? И еще. Я тут "козу" набросал. Посмотрите пожалуйста на предмет правильности и нужно как-то сделать так чтоб void Display() выводила в отдельном потоке. По событию таймера выводит. Заранее прошу прощения за навязчивость, но помощи ждать больше неоткуда. Если удастся привести проект в порядок, то будет мне коза на все случаи жизни.
Прикрепленный файл  myTCP.zip ( 71.43 килобайт ) Кол-во скачиваний: 40


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 28 2015, 13:57
Сообщение #9


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



И еще такой момент. тспклиент принимает ровно 16 посылок. не понимаю что его ограничивает.
Собственно проект, который принмает только 16 посылок. Прикрепленный файл  tcpclient.zip ( 80.66 килобайт ) Кол-во скачиваний: 32

Даже не представляю как так получается. Ничего такого и близко нет. Может кто поможет понять что за чудеса? Спасибо.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
XVR
сообщение Nov 30 2015, 11:47
Сообщение #10


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Цитата(smk @ Nov 28 2015, 12:42) *
Пока получается. Остается вопрос по кнопке "отсоединиться". Отсоединиться получается, но потом не соединяется т.к. элемент _tcpСlient ликвидирован. Что можно сделать?
Я не в курсе, можно ли вызвать Connect у TCPClient после Close, но если нельзя, то можно заново создать TCPClient:
Код
        private void button2_Click(object sender, EventArgs e)
        {
            tcpСlient.Close();
            tcpStream.Close();
            tcpСlient = new TcpClient();
        }

Цитата
нужно как-то сделать так чтоб void Display() выводила в отдельном потоке.
Для этого нужен отдельный поток и вызывать оттуда сам Display через делегат
Вызывать экранные элементы из GUI можно ТОЛЬКО из того потока, где они создавались
Цитата
И еще такой момент. тспклиент принимает ровно 16 посылок. не понимаю что его ограничивает.

Может сервер закрывает соединение?
Go to the top of the page
 
+Quote Post
smk
сообщение Nov 30 2015, 19:25
Сообщение #11


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Цитата(XVR @ Nov 30 2015, 13:47) *
Может сервер закрывает соединение?

Точно нет. проверил. Есть такая программка - hercules-3-2-8. С ней работает на ура. Вот бы посмотреть как сделано.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
XVR
сообщение Dec 1 2015, 11:42
Сообщение #12


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



А что происходит после приема 16 посылок? Останавливается поток или закрывается соединение? Или еще что то?
Цитата
Есть такая программка - hercules-3-2-8. С ней работает на ура.
Может надо что то серверу передавать? Посмотрите WireShark'ом обмен с hercules-3-2-8
Go to the top of the page
 
+Quote Post
smk
сообщение Dec 1 2015, 13:18
Сообщение #13


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



Цитата(XVR @ Dec 1 2015, 13:42) *
А что происходит после приема 16 посылок? Останавливается поток или закрывается соединение? Или еще что то?
Может надо что то серверу передавать? Посмотрите WireShark'ом обмен с hercules-3-2-8

После 16 посылок перестают отображаться новые данные, хотя они передаются. Шарком смотрел. Связь организована именно путем запрос-ответ. Геркулес нормально общается, передает запрос и получает ответ сколько угодно раз. Еще раз сегодня все уточню что в сети происходит для порядка...
Сервер это контроллер с кейловским стеком. Он и реньше нормально работал и сейчас повода для претензий не вижу.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post
sergey sva
сообщение Dec 2 2015, 07:54
Сообщение #14


Гуру
******

Группа: Свой
Сообщений: 2 546
Регистрация: 23-05-07
Из: Самарская область Сызрань
Пользователь №: 27 923



Может что то с отображением данных в gui.
Попробуйте вставить лог в это место
Код
  if (s != null)
{
запись в лог s
ShowReceiveMessage(s);
        s = String.Empty;
}
Go to the top of the page
 
+Quote Post
smk
сообщение Dec 2 2015, 14:39
Сообщение #15


Гуру
******

Группа: Свой
Сообщений: 2 246
Регистрация: 17-03-05
Из: Украина, Киев
Пользователь №: 3 446



В прикрепленном проекте все хорошо кроме приема данных. Соединяется, передает, разъединяется, повторно соединяется. Принимает сколько угодно пакетов. Но при приеме никак не выходит отобразить данные, хотя доподлинно известно, что они есть и валидны. Как только пытаюсь что-то вывести примет первый пакет и все. Счетчик показывает 1 и не меняется. Если часть вывода закоментировать, то выводит нули, хотя знаю что не должно быть нулей. Может как-то неправильно в функцию вывода передается массив, но при этом счетчик выводит правильно. Помогите сделать прием пожалуйста. Я уже не знаю что можно еще предпринять.
Прикрепленный файл  myTCP.zip ( 86.93 килобайт ) Кол-во скачиваний: 46

Как-то трудно на словах описать внятно. Если кто сможет, просто попробуйте.


--------------------
Живи днем так, чтобы ночью ты спал спокойно.
Go to the top of the page
 
+Quote Post

2 страниц V   1 2 >
Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 23rd June 2025 - 05:34
Рейтинг@Mail.ru


Страница сгенерированна за 0.01584 секунд с 7
ELECTRONIX ©2004-2016