System Object Model
System Object Model (SOMObjects) | |
---|---|
![]() | |
Разработчики | CILabs (IBM, Apple Computer и др.) |
Операционные системы | Mac OS, OS/2, AIX, Windows, DOS |
Последняя версия | 3.0 (декабрь 1996 года) |
System Object Model (SOM) — система объектно-ориентированных динамических библиотек, разработанная CILabs (IBM, Apple, OMG, Adobe, Oracle и др.). DSOM, основанная на CORBA распределённая версия SOM, позволяющая распределять объекты по различным вычислительным системам. Существуют реализации для Windows NT, MacOS Classic, OS/2, AIX, DOS, Copland, OS/390, NonStop OS. Для Windows NT, MacOS и OS/2 существует реализация компонентной разработки приложений OpenDoc на базе SOM/DSOM.
Сравнение с другими объектными моделями
IBM SOM концептуально похож на Microsoft Component Object Model. Обе системы решают проблему создания стандартного формата библиотеки, которую можно было бы вызывать из более, чем одного языка. SOM считается более функциональным, чем COM. COM предлагает два способа вызывать методы объекта, и объект может реализовать один из них или оба. Первый — это динамический вызов и позднее связывание (IDispatch), и, аналогично SOM, не зависит от языка программирования. Второй способ, через частный интерфейс, использует таблицу функций, которую можно сконструировать на C, либо использовать совместимую на нижнем уровне таблицу виртуальных методов объекта C++. Используя совместимые компиляторы C++, можно объявлять частные интерфейсы как чисто виртуальные классы C++. Частные интерфейсы — это компромисс между функциональностью и производительностью. Как только интерфейс опубликован в выпущенном продукте, в него нельзя вносить изменения, поскольку приложения–пользователи интерфейса были скомпилированы под конкретное устройство таблицы на нижнем уровне. Это пример проблемы хрупкого базового класса, которая может привести к DLL hell, когда после установки новой версии разделяемой библиотеки все программы, использующие старую версию, перестают работать корректно. Чтобы избежать этой проблемы, COM разработчикам необходимо всегда помнить о том, что нельзя изменять уже опубликованные интерфейсы. Если требуется добавить новые методы или внести другие изменения, нужно определять новые интерфейсы.
SOM предотвращает эти проблемы, предоставляя только позднее связывание и позволяя компоновщику времени исполнения перестраивать таблицы на лету. Таким образом, изменения в нижележащие библиотеки пересчитываются при их загрузке в программы ценой небольшой потери производительности.
SOM также более функционален в том, что касается полной поддержки различных ОО языков. В то время, как разработка на COM сводится к использованию урезанной версии C++, SOM поддерживает почти весь набор обычных возможностей и даже немного эзотерических. Например, SOM поддерживает множественное наследование, метаклассы и динамические вызовы. Некоторые из этих возможностей не отражены в большинстве языков, в связи с чем многие SOM/COM–подобные системы реализованы проще ценой поддержки меньшего набора языков. Полная гибкость многоязыковой поддержки была важна для IBM, в связи с необходимостью поддерживать как Smalltalk (одиночное наследование, динамическое связывание), так и C++ (множественное наследование и статическое связывание).
Наиболее заметное отличие между SOM и COM — это поддержка наследования — у COM отсутствует вовсе. Может показаться странным, что Microsoft произвела систему библиотек объектов, не поддерживающую наиболее фундаментальный принцип ООП. Главным препятствием для этого является сложность определения нахождения базового класса в системе, в то время, как библиотеки загружаются в потенциально произвольном порядке. COM требует от разработчика точно указывать базовый класс на этапе компиляции, делая невозможной вставку других наследованных классов в середину (по крайней мере, в чужие библиотеки COM).
Напротив, SOM использует простой алгоритм, просматривая дерево наследования в поисках потенциального базового класса и останавливаясь на первом подходящем. В большинстве случаев это основной принцип наследования. Обратная сторона этого подхода — вероятность отказа работать новых версий базового класса, несмотря на неизменный API. Эта вероятность существует в любой программе, не только в использующих разделяемые библиотеки, но проблему становится очень сложно отследить, если она существует в чужом коде. В SOM единственное решение — это полное тестирование новых версий библиотек, что не всегда легко.
Поддерживаемые языки программирования
C, C++
Эмиттеры для C и C++ входят в собственно поставку SOMobjects Developer Toolkit и позволяют как вызывать методы объектов, так и наследовать от классов. В некоторых компиляторах C++, сначала MetaWare High C++, потом IBM VisualAge C++, была реализована возможность Direct-to-SOM. В VisualAge C++ для Windows эта возможность появилась в версии 3.5[1], и эта же версия одновременно стала последней, в которой эта возможность поддерживалась.
REXX
ObjectREXX, поставляемый с OS/2, интегрирован с SOM, позволяет вызывать методы объектов и наследовать от классов. При передаче исходных текстов ObjectREXX сообществу open source все файлы, требуемые для функционирования этой интеграции, не были переданы, и в open source версию эта возможность не попала. Некоторое время в репозитории были следы интеграции с SOM, но скомпилировать было невозможно, а впоследствии всё, что было связано с SOM, было удалено совсем.
SmallTalk
Пакет SOMSupport для VisualAge SmallTalk позволяет вызывать методы SOM объектов и создавать SOM обёртки для классов SmallTalk.
COBOL
IBM ObjectCOBOL изначально использовал SOM как объектную систему в режиме Direct-to-SOM. Впоследствии ObjectCOBOL был перенесён на Java и начал пользоваться объектной системой Java вместо SOM.
Basic
В некоторых версиях VisualAge for Basic имелась интеграция с SOM[2]. Кроме того, Lotus Script, входящий в поставку OpenDoc, посредством OpenDoc Direct Scripting (ODDS) также может работать с SOM объектами[3].
Java
В SOMObjects Java Client[4] была возможность вызывать SOM объекты только удалённо, через DSOM. В примере с демонстрацией были классы, которые делались доступными на сервере DSOM, а затем Java апплет размещался на Интернет–ресурсе, создавал удалённые объекты и вызывал их методы. Локальный вызов методов не предусмотрен.
Pascal
Частным лицом были разработаны эмиттеры для Virtual Pascal, позже портированы на Free Pascal[5]. Позволяют вызывать методы и создавать свои классы.
Ada
Разработчики компилятора PowerAda сделали эмиттеры[6] и примеры использования SOM. PowerAda был доступен только на AIX, и для запуска эмиттера нужен SOM 3.0 Beta, тоже для AIX. SOM 3.0 для AIX утерян.
Modula-2
Canterbury Modula-2 для OS/2 имела объектно–ориентированные расширения, аналогичные Oberon-2, и поддерживала режим компиляции Direct-to-SOM в профессиональной версии.[7]
Oberon-2
Oberon Мicrosystems анонсировала поддержку Direct-to-SOM на Mac OS Classic, но состояние этого проекта неизвестно.[8]
Способы интеграции с SOM
Эмиттеры
Обычно разработка для SOM происходит по следующей схеме:
В режиме потребителя:
Разработчик запускает компилятор SOM с эмиттером для желаемого языка программирования, указав, для каких файлов IDL желаемой библиотеки, делать привязки. Например:
sc -sada somcm.idl
Эмиттер создаёт один или несколько файлов в том формате, который понимает компилятор выбранного языка программирования. При помощи этих файлов становится возможным создавать объекты описанных классов и вызывать их методы.
В режиме производителя:
Разработчик пишет свои .idl файлы, в которых делается #include других .idl файлов, и от описанных в других .idl классов производятся наследники. Затем разработчик запускает специальный эмиттер, который создаст файлы с вспомогательным кодом и файлы с пустой реализацией методов класса.
Например:
sc -sih animals.idl
sc -sc animals.idl
Первый вызов создаст animals.ih, который будет содержать, например, реализацию Animals_AnimalNewClass, которая запустит somBuildClass2, передав ей сложную структуру, синтезированную на основе входного .idl. Кроме этого вызова в этом файле есть сама эта структура и некоторые другие вспомогательные элементы, которые разработчик не должен изменять вообще. Второй вызов создаст animals.c с пустыми реализациями методов. Эмиттер C и C++ от IBM могут работать инкрементально, добавляя пустые новые методы, не трогая код существующих методов.
Кроме этого, если эмиттеры для создания .dll. Один эмиттер синтезирует главную функцию .dll, два других синтезируют .def и .nid файлы.
Эмиттер представляет собой библиотеку с названием emit*.dll, где * — это параметр аргумента -s компилятора SOM. Библиотека должна экспортировать процедуру emit (SOM 2.1) или emitSL (SOM 3.0), которая, будучи вызвана из компилятора SOM, выполняет работу, специфичную для выбранного эмиттера. Работа может быть любая. Для создания новых эмиттеров есть скрипт newemit.
Динамические языки программирования
VisualAge SOMSupport и ObjectREXX не требуют вызова эмиттера, вместо этого для интеграции с SOM применяется рефлексия встроенными средствами SOM (SOM Interface Repository).
Кроме того, стандарт OpenDoc включает в себя OpenDoc Direct Scripting (ODDS), и интерпретаторы сценарных языков программирования, реализующие интерфейс ODScriptComponent, могут тем самым обращаться к SOM классам посредством ODDS. Примером подобного языка программирования является Lotus Script, поставляемый в комплекте с OpenDoc[3].
Интеграция SOM и COM
В Novell разработали мост, при помощи которого объекты SOM стали доступны из языков, поддерживающих OLE Automation. Кроме того, Novell ComponentGlue позволяет приложениям, использующим одну из технологий OLE или OpenDoc, задействовать компоненты, сделанные по другой технологии, а также оформить OpenDoc part как компонент OLE (OCX). При этом используется утилита ctypelib. При использовании этой утилиты никакого программного кода во время компиляции не создаётся. В реестре регистрируется одна и та же DLL из состава OpenDoc, которая способна загрузить в память библиотеку SOM и создать во время выполнения таблицы виртуальных методов, трамплины и другие элементы, необходимые для прокси–объектов COM. Обычно ComponentGlue реализует только интерфейс IDispatch, но, чтобы ускорить работу, существует возможность объявить и реализовать собственный COM интерфейс, если пометить интерфейс SOM модификатором ODdual и привести в соответствие всем правилам для OLE интерфейсов.
Другим средством интеграции SOM и COM является утилита emitcom, создающая COM–обёртки для классов SOM на языке C++. emitcom входил в состав SOM 3.0 Beta (февраль 1996), но в состав SOM 3.0 Release (декабрь 1996), как и многие другие возможности, не вошёл.
Следует, однако, отметить, что поскольку COM никак не решает проблему хрупкого базового класса, следует относиться с осторожностью к подобным мостам. Обёртки COM, производимые emitcom, соответствуют слепку интерфейса класса в момент создания, и при изменениях в интерфейсе нужно создавать новые версии обёрток с новыми GUID COM–интерфейсов, которые бы, тем не менее, поддерживали COM–интерфейсы старых версий обёрток. Интерфейсы COM, создаваемые утилитой ctypelib для классов SOM, помеченных модификатором ODdual, не следует использовать из компилируемых языков программирования, поскольку низкоуровневое представление такого интерфейса не стабильно. ctypelib обычно перезаписывает библиотеку типов COM, а поддержание параллельно нескольких разных версий интерфейса не предусмотрено.
Direct-to-SOM (D2SOM, DTS)
При использовании эмиттеров в компилируемых языках программирования, таких, как C++, эмиттер C++ производит видимость того, что SOM класс является C++ классом. somInit проецируется на стандартный конструктор, а somAssign — на operator=. Однако, при реализации своих классов основную роль играет написание .idl, а реализация методов не выглядит как реализация методов класса. Необходимо постоянно вызывать компилятор SOM для обновления файлов. SOM получается чем–то инородным для языков программирования, компиляторы которых не имеют встроенную поддержку SOM.
Компилятор Direct-to-SOM C++ позволяет обойтись без написания .idl файлов. .idl файлы генерируются на основе заголовочных файлов DTS C++, а не наоборот. Таким образом, компилятор DTS C++ предоставляет полноценную однородную среду разработки, позволяющую писать всё на одном языке. Работать с som.dll в DTS C++ подобно тому, как работать с objc.dll в Objective-C.
Эмиттеры по–прежнему нужны, но только для импорта сторонних библиотек. В Microsoft C++ есть возможность писать #import <something.tlb>. Аналогично можно было бы поступить с IDL в DTS C++, но этого реализовано не было. Вместо этого нужно применить эмиттер, который создаст .hh файлы, требуемые для компилятора DTS C++. Компилятор DTS C++ поддерживает как обычные C++ классы, так и SOM классы, наследуемые от SOMObject (явно или неявно, при #pragma SOMAsDefault (on)). Как и в другом гибриде, Objective-C++, возможность смешивать классы из разных иерархий ограничена.
Direct-to-SOM C++ появился в MetaWare High C++ и позже продублирован в VisualAge C++, причём, эти реализации несовместимы напрямую, только через импорт/экспорт в .idl. В книге «Putting Metaclasses to Work» был описан ещё один, третий известный диалект DTS C++, компилятора для которого ещё не существует.
Альтернативные реализации
Существует открытая реализация SOM - somFree[9]. Проект заявляет двоичную совместимость с оригинальной реализацией от IBM. Netlabs.org поддерживает реализацию NOM, которая основана на принципах SOM, но не является совместимой ни на уровне исходного кода, ни на двоичном уровне.
Примечания
- ↑ VisualAge C++ 3.5 for Windows | Dr Dobb's
- ↑ VisualAge for Basic Ships
: The new VisualAge for Basic also incorporates IBM System Object Model (SOM)* technology, which allows applications to access and use diverse software components, even when they are written in different programming languages. Development becomes easier because SOM technology provides a language-neutral programming environment and manages local and remote communication among objects. - ↑ 1 2 Lotus Script Scripting
- ↑ Apache2 Ubuntu Default Page: It works
- ↑ p/osfree/code - Revision 1153: /trunk/OS2/SOM/Frameworks/Emitter/Emitters/Pas/Animals
- ↑ http://ocsystems.com/download/powerada/aix/powerada_som.tar.Z
http://octagram.name/pub/somobjects/ada/powerada/contrib/som/ - ↑ Возможности Canterbury Modula-2 for OS/2
Canterbury Modula-2 в EDM/2 wiki - ↑ Oberon Compiler Supports SOM And COM
Leigh C. Make Way for Oberon/F, 1997 - ↑ Домашняя страница проекта somFree . somFree.
Ссылки
- IBM SOMobjects Developer's Toolkit Version 3.0 for Windows NT, OS/2 Warp, and AIX Documentation (англ.)
- IBM SOMobjects Developer's Toolkit Version 3.0 (англ.) (архивная версия страницы загрузки, скачивание невозможно)
- IBM SOMobjects Developer's Toolkit Version 3.0 for WinNT (только WinNT)
- IBM SOMobjects Developer's Toolkit Version 3.0 for OS/2 (англ.) (только OS/2)
- IBM SOMobjects Developer's Toolkit Version 3.0 Readme (англ.)
- Лучшее из OS/2 на платформе Linux
- Исследование модели SOM