Версия для печати темы
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум программистов > Разработка Windows Forms > Завершение потока и Invoke


Автор: amarenkov 28.4.2008, 10:59
Добрый день.

У меня во вторичном потоке происходят какие-то действия и иногда вызывается следующая функция:
Код

  private void OnDCRefreshImage(object sender)
        {
            if (!fobj_picture.InvokeRequired)
            {
               //Здесь даже может ничего не делаться...
            }
            else
            {   
                fobj_picture.Invoke(new MethodInvoker(delegate { OnDCRefreshImage(sender); }));
            }
        }


На закрытии формы я выставляю вторичному потоку флаг на завершение. Если строчка с fobj_picture.Invoke закоментарена, то все нормально завершается. Если нет, то второй поток продолжает висеть и не завершается вообще никогда. 

Возможно кто-то знает, в чем дело?

Заранее спасибо. 

Автор: mr.DUDA 28.4.2008, 11:07
Очень похоже на deadlock. Гуишный поток говорит второму потоку "завершайся" и ждёт, пока тот не завершится одним из способов (через событие, WaitHandle, проверкой IsAlive или Join-ом). В это время второй поток безуспешно пытается вызвать Invoke, который попросту не может выполниться т.к. Invoke основан на очереди оконных сообщений и не выполнится пока гуишный поток не выйдет из ожидания. Примерно так.

Выходом может быть вызов из гуишного потока thread.Abort и thread.Join(таймаут) второму потоку. В корневой функции второго потока нужно поставить try..catch(ThreadAbortException), чтобы сообщение об ошибке не вылетало.

Автор: amarenkov 28.4.2008, 11:18
mr.DUDA, у меня немного не так организовано приложение.

Вторичный поток крутится в бесконечном цикле, проверяя некоторый флаг. Когда флаг становится в true, вторичный поток просто завершает свою процедуру и завершается сам.

То есть, в основном потоке завершение вторичного потока выглядит просто как установка флага в true. После чего, основной поток продолжает свое завершение.

Есть подозрение, что в тот момент, когда вторичный поток делает Invoke, основного потока уже нет smile. Может такое быть? Как на это проверить?

Автор: mr.DUDA 28.4.2008, 13:40
Если основной поток завершился, Invoke рухнет с исключением. Проверить можно поставив try..catch во втором потоке.

Кстати, ещё можно запустить в отладке, дойти до места где виснет и нажать "паузу" отладчика. В окне Debug/Windows/Threads посмотреть какие потоки выполняются и даблкликом пройтись по ним - покажет в каком месте в исходнике сейчас находится каждый поток.

Автор: amarenkov 28.4.2008, 13:43
mr.DUDA, сделал Abort и Join. Поток завершился, но исключения не вылетело. Что это может значить?

Добавлено через 2 минуты и 29 секунд
Хм, показывает, что второй поток стоит на строчке 
Код

fobj_picture.Invoke(new MethodInvoker(delegate { OnDCRefreshImage(sender); }));

и ни с места smile.

Добавлено через 5 минут и 9 секунд
Сделал так:
Код

MethodInvoker mi = new MethodInvoker(delegate { OnDCRefreshImage(sender); });
fobj_picture.Invoke(mi);

Стоит мертво на fobj_picture.Invoke(mi);

Автор: mr.DUDA 28.4.2008, 13:57
А в анонимный метод заходит? Я бы вынес для отладки OnDCRefreshImage в отдельный метод и передавал делегат на него.

Автор: amarenkov 28.4.2008, 14:27
Вынес. Нет, не заходит. Когда жму на паузу и смотрю данные объектов, то во всех полях fobj_picture написано, дескать "Текущий поток спит, ждет или джоин".

Вообще в Threads видно 7 каких-то потоков. Один - мой вторичный. Главный поток имел ID 3704. Так в момент подвеса этого потока уже вообще не видать в списке потоков. Он завершился что ли?

Автор: amarenkov 28.4.2008, 15:00
Чуть подробнее о вторичном потоке.

Основная функция:
Код

public void Draw()
        {
                fobj_mutex = new Mutex();
                fobj_arEvent = new AutoResetEvent(false);

                while (true)
                {
                    switch (fenm_drawState)
                    {
                        case EMDThreadState.edtsReStart:
                            // Всякое рисование
                            break;
                        case EMDThreadState.edtReSized:
                            OnResize();

                            fenm_drawState = EMDThreadState.edtsReStart;
                            break;
                        case EMDThreadState.edtAbort: return;
                        default: fobj_arEvent.WaitOne(); break;
                    }
                }           
        }


Пытаюсь его выключить:
Код

fobj_mapDraw.DrawState = CMapDraw.EMDThreadState.edtAbort;


Получается описанная выше проблема. Нашел такое еще решение:
Код

public void Exit()
        {
            fobj_mapDraw.DrawState = CMapDraw.EMDThreadState.edtAbort;
            while (fobj_mapDraw.Thread.IsAlive)
                Application.DoEvents();
         }


т.е. пока вторичный поток не завершится, основной обрабатывает свои сообщения. Так делать нормально вообще?

Второй вариант:
Код

public void Exit()
        {
            fobj_mapDraw.DrawState = CMapDraw.EMDThreadState.edtAbort;
            fobj_mapDraw.Thread.Abort();
            fobj_mapDraw.Thread.Join();
        }


Тоже работает корректно - поток завершается. Никаких исключений не вызывается. Но мне как-то этот вариант не нравится.

Powered by Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)