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

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

06.12.2014, 20:05. Показов 3887. Ответов 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
Ответ Создать тему
Новые блоги и статьи
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост.
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 - 2025, CyberForum.ru