Что такое насос сообщений? - PullRequest
92 голосов
/ 08 февраля 2010

В этой теме (опубликовано около года назад) обсуждается проблема, которая может возникнуть при запуске Word в неинтерактивном сеансе. (Очень сильный) совет, данный там, не должен делать это. В одном сообщении говорится: «Все API-интерфейсы Office предполагают, что вы запускаете Office в интерактивном сеансе на рабочем столе с монитором, клавиатурой и мышью и, что наиболее важно, с обработкой сообщений». Я не уверен, что это такое. (Я программировал на C # всего около года; мой другой опыт программирования был в основном с ColdFusion.)

Обновление:

Моя программа просматривает большое количество RTF-файлов, чтобы извлечь две части информации, использованной для построения номера медицинского заключения. Вместо того, чтобы пытаться понять, как работают инструкции по форматированию в RTF, я решил просто открыть их в Word и вытащить текст оттуда (фактически не запуская GUI). Иногда программа зависала в процессе обработки одного файла и оставляла открытым поток Word, прикрепленный к этому документу (мне все еще нужно выяснить, как закрыть этот файл). Когда я перезапустил программу, конечно, я получил уведомление, что существует поток, использующий этот файл, и я хотел открыть копию только для чтения? Когда я сказал «да», графический интерфейс Word неожиданно появился из ниоткуда и начал обрабатывать файлы. Мне было интересно, почему это произошло; но, может быть, когда диалоговое окно всплыло, насос сообщений начал выдвигать основной графический интерфейс в Windows?

Ответы [ 6 ]

166 голосов
/ 08 февраля 2010

Цикл сообщений - это небольшой фрагмент кода, который существует в любой родной программе Windows. Это выглядит примерно так:

MSG msg;
while (GetMessage(&msg, NULL, 0, 0))
{ 
   TranslateMessage(&msg); 
   DispatchMessage(&msg); 
} 

API Win32 GetMessage () получает сообщение из Windows. Ваша программа обычно проводит там 99,9% времени, ожидая, пока Windows сообщит, что произошло что-то интересное. TranslateMessage () - вспомогательная функция, которая переводит сообщения клавиатуры. DispatchMessage () гарантирует, что оконная процедура вызывается с сообщением.

Каждая программа .NET с графическим интерфейсом имеет цикл обработки сообщений, запускаемый Application.Run ().

Актуальность цикла сообщений для Office связана с COM. Офисные программы - это программы с поддержкой COM, так работают классы Microsoft.Office.Interop. COM заботится о многопоточности от имени COM-класса, это гарантирует, что вызовы, сделанные на COM-интерфейсе, всегда делаются из правильного потока. Большинство COM-классов имеют раздел реестра в реестре, который объявляет их ThreadingModel, безусловно, наиболее распространенные (включая Office) используют «Квартиру». Это означает, что единственным безопасным способом вызова метода интерфейса является выполнение вызова из того же потока, который создал объект класса. Или, другими словами, большинство классов COM не являются поточно-ориентированными.

Каждый поток с поддержкой COM принадлежит квартире COM. Есть два вида, однопоточные квартиры (STA) и многопоточные квартиры (MTA). Класс COM с поточной квартирой должен быть создан в потоке STA. Это можно увидеть в программах .NET, точка входа потока пользовательского интерфейса Windows Forms или WPF-программы имеет атрибут [STAThread]. Модель квартиры для других потоков задается методом Thread.SetApartmentState ().

Большие части сантехники Windows не будут работать правильно, если поток пользовательского интерфейса не является STA. В частности, Drag + Drop, буфер обмена, диалоговые окна Windows, такие как OpenFileDialog, элементы управления, такие как WebBrowser, приложения автоматизации пользовательского интерфейса, такие как программы чтения с экрана. И многие COM-серверы, такие как Office.

Жестким требованием к потоку STA является то, что он никогда не должен блокировать и должен прокачивать цикл сообщений. Цикл сообщений важен, потому что это то, что COM использует для перенаправления вызова метода интерфейса из одного потока в другой. Хотя .NET упрощает маршалинг-вызовы (например, Control.BeginInvoke или Dispatcher.BeginInvoke), сделать это очень сложно. Поток, который выполняет вызов, должен находиться в известном состоянии. Вы не можете просто произвольно прервать поток и заставить его выполнить вызов метода, что вызовет ужасные проблемы повторного входа. Поток должен быть "бездействующим", не занятым выполнением любого кода, который изменяет состояние программы.

Возможно, вы можете увидеть, к чему это приведет: да, когда программа выполняет цикл обработки сообщений, она простаивает. Фактическое маршалинг происходит через скрытое окно, которое создает COM, оно использует PostMessage, чтобы оконная процедура этого окна выполняла код. На нити СТА. Цикл сообщений обеспечивает выполнение этого кода.

11 голосов
/ 08 февраля 2010

«Насос сообщений» является основной частью любой программы Windows, которая отвечает за отправку оконных сообщений в различные части приложения. Это ядро ​​программирования Win32 UI. Из-за его повсеместности многие приложения используют насос сообщений для передачи сообщений между различными модулями, поэтому приложения Office прерываются, если они запускаются без какого-либо пользовательского интерфейса.

Википедия имеет базовое описание .

6 голосов
/ 08 февраля 2010

Джон говорит о том, как система Windows (и другие оконные системы - X Window , оригинальная Mac OS ....) реализуют асинхронные пользовательские интерфейсы, используя события через систему сообщений.

За кулисами для каждого приложения есть система обмена сообщениями, где каждое окно может отправлять события другим окнам или прослушивателям событий - это реализуется путем добавления сообщения в очередь сообщений. Существует основной цикл, который всегда выполняется, просматривая эту очередь сообщений и затем отправляя сообщения (или события) слушателям.

Статья в Википедии Цикл сообщений в Microsoft Windows показывает пример кода базовой программы Windows - и, как вы можете видеть на самом базовом уровне, программа Windows - это просто " насос сообщений ".

Итак, собери все вместе. Причина, по которой программа Windows, разработанная для поддержки пользовательского интерфейса, не может выступать в качестве службы, заключается в том, что ей требуется постоянная работа цикла сообщений для включения поддержки пользовательского интерфейса. Если вы реализуете его как службу, как описано, она не сможет обрабатывать внутреннюю асинхронную обработку событий.

5 голосов
/ 14 июня 2011

В COM насос сообщений сериализует и десериализует сообщения, отправленные между квартирами. Квартира - это мини-процесс, в котором могут быть запущены компоненты COM. Квартиры бывают однопоточными и свободными. Однопоточные квартиры в основном являются устаревшей системой для приложений COM-компонентов, которые не поддерживают многопоточность. Обычно они использовались с Visual BASIC (поскольку он не поддерживает многопоточный код) и устаревшими приложениями.

Я полагаю, что требование к прокачке сообщений для Word вытекает либо из COM API, либо из частей приложения, не являющихся поточно-ориентированными. Имейте в виду, что модели .NET с многопоточностью и сборкой мусора не очень хорошо работают с COM из коробки. COM имеет очень упрощенный механизм сборки мусора и модель потоков, которая требует, чтобы вы работали по-COM. Использование стандартных Office PIA по-прежнему требует явного отключения ссылок на COM-объекты, поэтому необходимо отслеживать каждый созданный дескриптор COM. PIA также будут создавать вещи за сценой, если вы не будете осторожны.

.NET-COM интеграция - это целая тема, и даже есть книги, написанные на эту тему. Даже использование COM API для Office из приложения для интерактивного рабочего стола требует, чтобы вы перепрыгивали через обручи и следили за тем, чтобы ссылки были явно освобождены.

Office можно считать поточно-небезопасным, поэтому вам потребуется отдельный экземпляр Word, Excel или других Office приложений для каждого потока. Вам придется взять на себя начальные накладные расходы или поддерживать пул потоков. Пул потоков должен быть тщательно проверен, чтобы убедиться, что все ссылки COM были правильно освобождены. Даже запуск и завершение работы экземпляров требует от вас правильного освобождения всех ссылок. Если вы не укажете свои «я» и пересекаете ваши «я», это приведет к большому количеству мертвых COM-объектов и даже целым экземплярам Word, пропущенным.

2 голосов
/ 08 февраля 2010

Википедия предполагает, что это означает основной цикл событий программы .

0 голосов
/ 08 февраля 2010

Я думаю, что это обсуждение на канале 9 имеет хорошее краткое объяснение:

Этот процесс связи между окнами стал возможен благодаря так называемому Windows Message Pump. Представьте, что Message Pump - это объект, который обеспечивает взаимодействие между окнами приложений и рабочим столом.

...