Форум программистов, компьютерный форум, киберфорум
C# для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.56/18: Рейтинг темы: голосов - 18, средняя оценка - 4.56
0 / 0 / 1
Регистрация: 01.06.2014
Сообщений: 10

Исключения в потоках

06.12.2014, 20:05. Показов 3962. Ответов 9
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Здравствуйте!

Столкнулся со следующей проблемой:
Написал класс, объект которого должен асинхронно устанавливать tcp-соединение с сервером. Извне о результате соединения пользователь информируется через события:

C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
        public void Connect(IPEndPoint ipep)
        {
            socket.BeginConnect(ipep, ConnectCallBack, socket);
        }
        
        private void ConnectCallBack(IAsyncResult result)
        {
            Socket socket = (Socket)result.AsyncState;
            try
            {
                socket.EndConnect(result);
                OnConnect("Соединение установлено");
                                          ....
            }
            catch
            {
                OnDisconnected("Соединение не установлено");
            }
        }
Исключения, выброшенные в блоке catch не обрабатываются вообще никак, приложение
просто закрывается без каких-либо ошибок. Из-за чего это происходит?

Сначала думал, что это происходит только с потоками из пула потоков, как в примере выше.
C#
1
2
3
4
5
6
  System.Threading.Tasks.Task t = new System.Threading.Tasks.Task( () =>
               {
                   int a = 0;
                   int sd = 1 / a;
               });
           t.Start();
Тут явно выскочит ошибка и exception DivideByZeroException. Всё как положено. А если тот же самый код выполнить в обработчике таймера, то приложение завершит свою работу без каких-либо сообщений вообще:

C#
1
2
3
4
5
6
    System.Threading.Timer t = new System.Threading.Timer(new TimerCallback((o) =>
                {
                    int a = 0;
                    int sd = 1 / a;
                }));
            t.Change(1000, 0);
Но с другой стороны, если создать задачу Task, то исключение будет перехватываться корректно.
Объясните, пожалуйста, причину этих вылетов без сообщений об ошибках.
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
06.12.2014, 20:05
Ответы с готовыми решениями:

Почему нельзя обрабатывать исключения в потоках
Сижу читаю msdn: http://msdn.microsoft.com/ru-ru/library/6kac2kdh.aspx Пять раз перечитал эти предложения плюс 2 раза перечитал...

Исключения в потоках
Кто читал книгу Параллельное программирование на С++ в действии; у меня возник вопрос из 8 главы. Какая роль в нижеприведенном коде класса...

Организация собственного исключения и передача в него исходного исключения
Для обработки исключений при работе с БД я создал собственное исключение: public class SQLEntityException : ApplicationException ...

9
7 / 7 / 4
Регистрация: 12.09.2012
Сообщений: 44
06.12.2014, 21:55
Исключения в потоках пула прерывают работу программы
http://msdn.microsoft.com/ru-r... .110).aspx

С новым синтаксисом можно писать удобный асинхронный код, не мучаясь с кэллбэками
C#
1
2
3
4
EndPoint remoteEp;
Socket s;
try { await Task.Factory.FromAsync(s.BeginConnect(remoteEp, x => { }, null), s.EndConnect); }
catch (SocketException) { ... }
0
0 / 0 / 1
Регистрация: 01.06.2014
Сообщений: 10
06.12.2014, 22:04  [ТС]
Тогда почему исключения в задачах не вызывают такого падения программы, и они отлавливается как обычно? Task же использует потоки из пула.
0
7 / 7 / 4
Регистрация: 12.09.2012
Сообщений: 44
06.12.2014, 22:20
Task хранит возникшее исключение внутри. При обращении к нему (task.Resut, task.Wait(), await task) будет выброшено исключение. В книге Джеффри Рихтера я читал, что если исключение нигде не обрабатывается, то при сборке мусора исключение будет выброшено в потоке gc, программа прекратит работу, а сделано это было командой разработчиков для того, чтобы исключения Task обрабатывали. Но в с выходом Net FW 4.5 всё поменялось.
http://habrahabr.ru/post/231665/
0
0 / 0 / 1
Регистрация: 01.06.2014
Сообщений: 10
06.12.2014, 22:31  [ТС]
Получается наоборот, Task выбрасывает исключение не при обращении к нему, а во время выполнения

C#
1
2
3
4
5
6
  System.Threading.Tasks.Task t = new System.Threading.Tasks.Task( () =>
               {
                   int a = 0;
                   int sd = 1 / a;
               });
           t.Start();
На 4 строчке выбрасывается исключение. Если тоже самое сделать в обработчике таймера (System.Threading.Timer), исключения не будет и тут, возможно, исключение выбрасывается в потоке GC и приложение не успев никак отреагировать просто закрывается без всяких ошибок.
0
7 / 7 / 4
Регистрация: 12.09.2012
Сообщений: 44
06.12.2014, 22:44
Таймер выполняет код в потоке пула, это значит, что ваш код сразу завершит приложение. Task же хранит необработанное исключение внутри себя, сборщик мусора вызывает task.Dispose(), который(или финализатор, надо смотреть исходники Task) в net fw 4.0 как раз и завершит приложение. Чтобы убедиться в этом, скопируйте код с хабра, отключите в visual studio исключения(Debug-Exceptions-CLR exceptions и снимите галочку).
1) Если у вас Net FW 4.5, ничего не случится.
2) Если я правильно понимаю, проверять надо на компьютере, на котором нету Net FW 4.5, потому что 4.5 считается не отдельной версией, а обновлением, заменяющим 4.0. Тогда приложение завершится.
1
0 / 0 / 1
Регистрация: 01.06.2014
Сообщений: 10
06.12.2014, 22:57  [ТС]
lyialat, спасибо за ответы, буду разбираться
0
0 / 0 / 1
Регистрация: 01.06.2014
Сообщений: 10
07.12.2014, 23:27  [ТС]
Что бы не создавать новую тему спрошу здесь.

Задача такая: клиент должен коннектиться к серверу и обмениваться всякими запросами/командами. Что бы не было тонны кода классе главной формы, хочу поместить всё взаимодействие клиента с сервером в отдельный класс. Для информирования о результате этих операций использую события. И их достаточно большое кол-во (коннект, дисконнект, прием/передача файла итд)
C#
1
2
3
4
5
6
7
8
9
10
Client.Connected += (sender, args) =>
            {
                this.Invoke(new Action(() =>
                {
                      richTextBoxLog.SelectionColor = Color.Green;
                      richTextBoxLog.AppendText("Соединение установлено\r\n");
                      richTextBoxLog.SelectionColor = Color.Black;
                      richTextBoxLog.ScrollToCaret();
                }));
            };
Вопрос по конструированию приложения, как лучше его структурировать? То как это сделано сейчас мне не нравится.
При работе возникают проблемы:
1) Так как событий много, и почти при каждом из них я что-то обновляю на форме, то бывают случаи, когда, например, я хочу что-то написать в richTextBoxLog красным шрифтом, но из-за возникшего события и прерывания this.Invoke, текст выводится другим цветом. Синхронизация тут не поможет, т.к. всё происходит в одном потоке.
2) При закрытии приложения выскакивает ошибка : InvalidOperationException "Невозможно вызвать Invoke или BeginInvoke для элемента управления до завершения создания дескриптора окна." Во время обработки какого-либо события. Возможно при закрытии формы можно отписаться от всех событий и после уже завершить работу, но мне кажется что мой такой подход с событиями в принципе не очень хороший или неправильный вовсе.
0
0 / 0 / 0
Регистрация: 13.11.2013
Сообщений: 7
08.12.2014, 13:55
Подскажите, пожалуйста.
Разбираюсь с отменой задачи и возник вопрос: если запустить этот пример http://msdn.microsoft.com/ru-r... .110).aspx, то всё работает, однако, если перед tokenSource2.Cancel(); написать Thread.Sleep(100); для эмитации вычислений в главном потоке, то выбрасывается исключение "OperationCanceledException не обработано пользовательским кодом" в задаче ct.ThrowIfCancellationRequested(); Из-за чего это происходит? казалось бы всего-навсего приостановили главный поток, потом запустили и отменили задачу...
0
7 / 7 / 4
Регистрация: 12.09.2012
Сообщений: 44
11.12.2014, 20:37
swdclk, с кодом async было бы так:
C#
1
2
3
4
5
6
7
8
9
10
async void button1_Click()
{
    bool result = true;
    try { await client.ConnectAsync(); }
    catch (SocketException) { result = false; }
    richTextBoxLog.SelectionColor = result ? Color.Green : Color.Red;
    richTextBoxLog.AppendText(String.Format("Соединение{0}установлено{1}", result ? " " : " не ", Environment.NewLine));
    richTextBoxLog.SelectionColor = Color.Black;
    richTextBoxLog.ScrollToCaret();
}
async умеет захватывать контексты, так что invoke не нужен.
Решать вам
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
11.12.2014, 20:37
Помогаю со студенческими работами здесь

Синхронизация в потоках
В программе я хочу что бы повар приготовил еду, а официант подал её. При этом не должно быть так, что официант отнес еще не сделанную еду....

Ошибка в потоках
Не понимаю что не так? Обьясните пожалуйста как исправить? ...

Ошибки в потоках
void test() { Thread ^t = gcnew Thread(gcnew ThreadStart(this, &Form1::hide_button)); t->Start(); //Создаю новый поток //Далее...

InvalidOperationException в потоках
using System; using System.Collections.Generic; using System.Threading; using System.Linq; struct StaticThread { static...

Сортировка в потоках
Здравствуйте. Нужно отсортировать массив 3-мя различными сортировками. Выполнить это в потоках. Не могли бы Вы подсказать, с чего начать и...


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

Или воспользуйтесь поиском по форуму:
10
Ответ Создать тему
Новые блоги и статьи
My Business CRM
MaGz GoLd 16.04.2026
Всем привет, недавно возникла потребность создать CRM, для личных нужд. Собственно программа предоставляет из себя базу данных клиентов, в которой можно фиксировать звонки, стадии сделки, а также. . .
Знаешь почему 90% людей редко бывают счастливыми?
kumehtar 14.04.2026
Потому что они ждут. Ждут выходных, ждут отпуска, ждут удачного момента. . . а удачный момент так и не приходит.
Фиксация колонок в отчете СКД
Maks 14.04.2026
Фиксация колонок в СКД отчета типа Таблица. Задача: зафиксировать три левых колонки в отчете. Процедура ПриКомпоновкеРезультата(ДокументРезультат, ДанныеРасшифровки, СтандартнаяОбработка) / / . . .
Настройки VS Code
Loafer 13.04.2026
{ "cmake. configureOnOpen": false, "diffEditor. ignoreTrimWhitespace": true, "editor. guides. bracketPairs": "active", "extensions. ignoreRecommendations": true, . . .
Оптимизация кода на разграничение прав доступа к элементам формы
Maks 13.04.2026
Алгоритм из решения ниже реализован на нетиповом документе, разработанного в конфигурации КА2. Задачи, как таковой, поставлено не было, проделанное ниже исключительно моя инициатива. Было так:. . .
Контроль заполнения и очистка дат в зависимости от значения перечислений
Maks 12.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеПерсонала", разработанного в конфигурации КА2. Задача: реализовать контроль корректности заполнения дат назначения. . .
Архитектура слоя интернета для сервера-слоя.
Hrethgir 11.04.2026
В продолжение https:/ / www. cyberforum. ru/ blogs/ 223907/ 10860. html Знаешь что я подумал? Раз мы все источники пишем в голове ветки, то ничего не мешает добавить в голову такой источник, который сам. . .
Подстановка значения реквизита справочника в табличную часть документа
Maks 10.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеПерсонала", разработанного в конфигурации КА2. Задача №1: при указании работ (справочник РаботыПоРемонтуСпецтехники),. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru