Нэш Трей - C# 2010. Ускоренный курс для профессионалов (2010) (1160865), страница 98
Текст из файла (страница 98)
При вызове А11оса сеОаса51ос во всех потоках распределяется новый элемент для хранения ссылки на зкземттляр типа Буясеш. Оьч есс. Метод А11осасепасая1ос возвращает дескриптор в форме экземпляра объекта?,оса1ОасаБпогея1ос. Обращаться к этому месту в памяти можно с помощью методов Бесцаса и яеьпаса потока. Рассмотрим следующую модификацию предмаущего примера: пя1пд Яуягезц пя1пд Яуяпеш.ТЬгеабгпч; рпЬ1гс с1аяз ТОБС)аяя ( зсап1с ТЬБС1аяя() ( с1я51ог = Тпгеаб.А11осагецапа51ог О 1 рггчапе ТЬБС1аяя() ( Сопяо1е.аггее(,гпе( "Создание ТОБС1аяз" )) ) рпьугс япаг с Т15С1аяя Т1яЯ1ос ( оес ОЬ)есп оь) = Тпгеаб.йегсаса( С1з51ос ) 1Г( оЬ) == пп11 ) ( оЬБ = и ТОЯС1азз()т тьгеаб.зессага( г1я51ог, оь) )т гегпгп (ть501аяя) оь)т ) рг1чаге япап1с Ьоса10апавпоге51оп П1я51ог = пп11т ) риЬ1гс с1аяя Еппгуро1пт. ( рг1часе япапгс чогб ТЬгеабГппс() ( Сопяо1е.нг(пе11пе( "Поток (О) запускается...", Тпгеаб.спггепптьгеаб.Мападебтьгеаб1б )т Сопяо1е.нггпсь1пе( "Г1ябага гог Сьья СЬгеаб гя 1"(О) 1"", т1501аяя.т1яя1ог )т Сопяс1е.кг1пеь1пе( "Поток (О) завершается", Тпгеаб.спггепстпгеаб.Мападебтьгеаб1б )т ) ясап1с чсгб Магп () ( Тьгеаб Пьгеаб1 = пеи Тьгеаб( пеи Тьгеабвпагг(Епггуро1пг.тьгеабгппс) )т ТЬгеаб Спгеаб2 = пеи Тпгеаб( пеи Тьгеабвсагс(кпсгуРо1пп.тьгеабгппс) Сьгеаб1.5яагп(); СЬгеаб2.5пагг(); Как видите, использование динамических элементов несколько сложнее, чем применение метода статического поля.
Однако этот вариант обеспечивает некоторую дополнительную гибкость. Обратите внимание, что элемент распределяется в инициализаторе типа, которым является статический конструктор, присутствующий в примере. 372 Глава ) 2 Таким образом, элемент выделяется для всех потоков в точке, где исполняющая среда инициализируег тип для использования. Обратите внимание, что в средстве доступа свойства класса тЬБС1азз элемент проверяетсн на равенство ло11. При распределении элемента с применением А11осасепаса31ос он инициализируется ло11 для каждого потока. Обращаться к специфичному для потока хранилишу может быть удобно по строковому имени, а не через ссылку на экземпляр ьоса1РасазсогеБ1ос.
Для этого элемент ТЕЗ должен быть создан с использованием тЬгеаб. А11осаСеиапебоага31оС. Однако при этом следует разумно выбирать уникальное имя, чтобы применение того же самого имени еще где-либо в коде не привело к нежелательным эффектам. В качестве имени для элемента можно применить, например, строковое представление глобально уникального идентификатора О)ЛР. Для доступа к элементу вызывается метод Беснаюеоаса31ос, который просто транслирует строку в экземпляр ьоса1пасазсоге31ос. Необходимые сведения по именованию элементов локальных хранилищ потоков даны в документации МЗРХ. Большая часть сказанного должна быть знакома разработчикам, использующим локальные хранилища потоков в МЧп32.
Однако здесь есть одно усовершенствование: поскольку управляемые элементы ТЕЗ реализованы иначе, ограничение на количество ТЬЗ-элементов ЪУ1п32 здесь не действует. Как неуправляемые потоки и апартаменты СОМ приспособлены друг к другу Неуправляемые потоки могут входить в управляемую среду извне. Например, управляемые объекты могут быть представлены "родному" колу через промежуточный уровень СОМ.
Когда "родной" поток обращаетсн к объекту, он входит в управляемую среду. Когда это случается, среда СЬН фиксирует данный факт, и если это первый раз, когда неуправляемый поток осуществляет вызов СЬЛ, устанавливаютсн необходимые учетные структуры, которые позволяют потоку выполняться как управляемый поток внутри управляемой исполняющей среды. Как уже упоминалось, потоки, входящие в управляемую среду подобным образом, начинают свое управляемое существование с состояния килоспо [см.
рис. 12.1]. Как только все учетные структуры настроены, то дальше всякий раз, когда тот же неуправляемый поток входит в исполняющую среду, он ассоциируется с тем же управляемым потоком. Подобно тому, как управляемые объекты могут быть представлены внешнему миру в виде объектов СОМ, так и объекты СОМ могут быть представлены управляемому миру в виде управляемых объектов. Когда управляемый поток вызывает таким образом объект СОМ, исполняющая среда ослабляет контроль над состоянием потока до тех пор, пока он не вернется в управляемую среду. Предположим, что объект СОМ, написанный на "родном" С++, вызывает функцию ж!п32 Ар! по имени иасстогзсоо1еОь) есс, чтобы ожидать сигнала от определенного объекта синхронизации.
Затем, если управляемый поток вызовет тЬгеаб. АЬогС или тьгеаб. 1лсеггорс, чтобы разбудить поток, то его пробуждение будет отложено до момента, когда поток войдет обратно в управляемое окружение. Другими словами, этот вызов не имеет эффекта до тех пор, пока поток выполняет неуправляемый код. Поэтому необходимо быть в некоторой мере осведомленным о механизмах синхронизации, используемых родными объектами СОМ, к которым обращается управляемый код. И, наконец, если когда-нибудь в прошлом приходилось выполнять разработку в технологии СОМ, то должньв быть знакомы понятия апартамента СОМ [СОМ арэгнпепС), а также сопровождающих их понятий иронси и заглушек'.
Когда управляемый код про- Детальное описание апартаментов СОМ и их работы можно найти в книге Дона Вокса [Роп Вох) Кээеяяо) СОМ [Абщэоп-уВеэ)еу рго[еээшпа), 13эт с) Многопоточность я Стт 373 изводит вызов объекта СОМ, важно, чтобы управляемый код был настроен на вызов неуправляемого объекта СОМ либо через однопоточный апартамент [з)пя1е-тйтеадег) арагтгпепт — 8ТА), либо через многопоточный апартамент [шп!П-тЬгеабеб арагтшепт— МТА). Это свойство можно установить для нового управляемого потока с помощью метода тьгеаб. яесАрагспгепсясате. Как только поток стартует, состояние апартамента блокируется.
Другими словами, изменить его впоследствии нельзя. При обращении к объекту СОМ из управляемого кода лучше знать тип апартамента. в котором будет выполняться этот СОМ-объект. В этом случае можно взвешенно выбрать тип апартамента СОМ, в котором должен выполняться поток. Выбор неверного типа может плохо отразиться на эффективности, вынуждая все вызовы проходить через прокси и заглушки. В еще худших ситуациях объекты СОМ могут оказаться вообще не вызываемыми из других типов апартаментов. Используя Тпгеаб. ЯеСАрагсгяепСЯСаге, можно контролировать свойство СОМ- апартамента для создаваемых новых управляемых потоков.
Но как насчет главного потока приложения? Дело в том, что когда главный поток управляемого приложения запущен, уже слишком поздно устанавливать свойство состояния апартамента. Причина в том, что управляемая исполняющая среда инициализирует главный поток в состояние МТА при инициализации управляемого приложения. Если нужно изменить состояние апартамента главного потока на 8ТА, то единственный способ сделать это заключается в декорировании метода ма1п атрибутом ятАтьг вас)Ас с гТЬп се.
Кстати, его можно также декорировать атрибутом мтАтпгеабАссг1Ьпсе, но это будет излишне, поскольку МТА является выбором СЕЛ по умолчанию. Ниже показан пример: риЬ11с с1аяя Епггуго1пг ( [ЯТАТЬгеяг)) ягят1с чояб Мя1п[) 1 ) Если ранее приходилось работать с приложениями 1))тшботяз РЬппя, особенно с теми, что генерируются мастерами Ч1зиа) 81пб)о, то, скорее всего, вы уже видели этот атрибут и недоумевали, зачем он нужен. Декорнрование этим атрибутом главного потока графического пользовательского интерфейса приложения упрощает интеграцию элементов управления Аст)яеХ в графический интерфейс, поскольку они обычно выполняются в 8ТА.
Состояние апартамента управляемого потока имеет отношение только к ситуациям взаимодействия СОМ. Обратите внимание, что состояние апартамента управляемого потока не оказывает никакого эффекта на выполнение управляемого кода. И что более важно, когда управляемые объекты потребляются "родным" приложением через промежуточный уровень взаимодействия СОМ, состояние апартамента не управляет тем, в каком апартаменте находится объект с точки зрения "родного" приложения. Со стороны "родного" приложения все управляемые объекты выглядят объектами СОМ, находящимися в МТА и интегрирующими РТМ [г гее ТЪгеабеб МагзЬайег — свободный потоковый маршализатор). К тому же все потоки.
созданные в пуле потоков СЬК, всегда располагаются в МТА для процесса. Синхронизация работы между потоками Синхронизация является, пожалуй, наиболее трудной частью создания многопоточных приложений. Вы можете создавать дополнительные потони для выполнения какойто работы хоть целый день, не заботясь о синхронизации, до тех пор, пока эти потоки не производят параллельный доступ к данным вместе с другими потоками.














