Реализация промежуточного объектно-реляционного слоя в 


Мы поможем в написании ваших работ!



ЗНАЕТЕ ЛИ ВЫ?

Реализация промежуточного объектно-реляционного слоя в



Среде Oracle 9

 

В настоящее время в рамках проекта создания программной платформы для интеграции приложений ведется разработка промежуточного объектно-реляционного слоя общего назначения. Объектно-реляционный слой предназначен для работы с произвольными прикладными объектно-ориентированными данными, модели которых описаны на языке EXPRESS.

Для интегрируемых приложений объектно-реляционный слой предоставляет программные объектно-ориентированные интерфейсы доступа к прикладным данным на некоторых популярных языках реализации). Организация этих интерфейсов следует перечисленным выше принципам прозрачного манипулирования хранимыми данными, декларируемым Манифестом объектно-ориентированных баз данных. Интерфейсы предоставляют функционально развитый набор операций для манипулирования хранимыми и временными объектами, включая операции создания, модификации, удаления объектов, навигации по их однонаправленным и двунаправленным ассоциативным связям и выборки объектов на основе языка запросов. Запросы базируются на конструкции QUERY языка EXPRESS, позволяющей задать произвольный предикат на множестве объектов и отобрать те из них, которые удовлетворяют условию данного предиката. Интерфейсы предусматривают несколько пессимистических и оптимистических моделей транзакций с различными способами изоляции на уровне отдельных прикладных объектов и самостоятельных объектных популяций, содержательных для коллективных пользовательских сессий и участвующих в них приложений.

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

В качестве хранилища данных реализация объектно-реляционного слоя предусматривает использование реляционных СУБД со схемами, основанными на рассмотренных выше паттернах объектно-реляционного отображения. При этом функции по управлению транзакциями, разрешению запросов, контролю целостности данных, управлению версиями и контролю прав доступа распределяются между сервером объектно-реляционного слоя, через который непосредственно взаимодействуют приложения, и реляционной СУБД, выступающей в роли вторичного хранилища данных.

Важнейшими функциями, реализуемыми непосредственно средствами реляционной СУБД, являются операции манипулирования хранимыми объектами и выполнения простых объектных запросов. С этой целью на языке PL/SQL разрабатываются пакеты программ, эмулирующие объектно-ориентированные интерфейсы доступа к данным путем предоставления функциональных средств для создания, модификации, поиска и удаления объектов. Поскольку полная поддержка объектного языка запросов средствами реляционной СУБД представляется проблематичной с учетом разнообразия императивных конструкций языка EXPRESS, пакеты программ выполняют простейшие виды запросов на основе хранимых объектных идентификаторов (PID), объектных типов и навигационных маршрутов в виде графов переходов по ассоциативным связям типизированных объектов. Поддерживая кэширование объектов, посредник в ряде случаев разрешает запросы самостоятельно, а иногда переадресовывает их реляционной СУБД. При этом происходит редукция клиентского запроса, представленного в общей форме, к запросу упрощенного вида, расширяющего множество объектов и выполнимого пакетом программ реляционной СУБД. Результаты затем обрабатываются посредником с целью исключения объектов, полученных в результате интерпретации упрощенного запроса и не удовлетворяющих исходному.

Поскольку выбор стратегии отображения для реализации объектно-реляционного слоя подобной функциональности крайне неоднозначен с учетом разнообразия потенциальных приложений, предполагается реализация и поддержка нескольких альтернативных стратегий, а именно: схемо-независимого, схемо-зависимого и BLOB подходов. Они базируются на тех или иных сочетаниях рассмотренных выше паттернов ОР отображения и используют собственные пакеты программ на PL/SQL для реализации базовой функциональности объектно-реляционного слоя. Хотя все пакеты реализуют семантически эквивалентные наборы операций для манипулирования объектами и их поиска, их внешние интерфейсы не допускают унификацию в силу ограниченных возможностей языка SQL при формировании клиентских запросов со стороны объектно-реляционного посредника для специфических реляционных схем представления объектно-ориентированных моделей данных. Для адаптации посредника к иным объектно-реляционным стратегиям в его архитектуре предусмотрены специальные компоненты-адаптеры, обеспечивающие требуемую виртуализацию хранилищ данных. Каждый адаптер реализуется с учетом специфики конкретного ОР отображения.

В качестве целевой платформы реализации промежуточного объектно-реляционного слоя выбрана СУБД Oracle9.

Схемо-независимая стратегия

 

Разработанная схемо-независимая стратегия состоит в применении обобщенных паттернов AllClasses–OneTable, Attribute–Table, GenericAssociation, GenericSelect, GenericAggregate для отображения схем, классов и атрибутов, а также паттерна представления соответствующих метаданных прикладной модели реляционными таблицами.

Реализованные в среде Oracle9 PL/SQL пакеты обеспечивают выполнение всего базового набора операций с хранимыми объектно-ориентированными данными и запросов к ним. Обобщенная, независимая от конкретных прикладных моделей реализация PL/SQL процедур и функций основана на совместном одновременном использовании данных и метаданных, хранимых в системах таблиц в соответствии с перечисленными паттернами отображения. Приведем описание основных пакетных методов для объектно-реляционного отображения в качестве иллюстрации схемо-независимой стратегии.

Пакет lb_defined_types для работы с метаинформацией о пользовательских типах данных, определенных EXPRESS схемой:

· procedure Register_Defined_Type – регистрация в базе данных пользовательского типа схемы;

· procedure Save_Enum_Type – сохранение метаданных для перечислимого типа;

· procedure Save_Select_Type – сохранение метаданных для селективного типа;

· procedure Save_Aggregate_Type – сохранение метаданных для агрегатного типа;

· function Get_Type – выдача метаинформации о пользовательском типе данных.

Пакет lb_entity предназначен для работы с метаинформацией, относящейся к объектным типам EXPRESS схемы:

· function Register_Entity – регистрация объектного типа схемы;

· procedure Save_Attribute – сохранение метаданных для атрибута, определяемого в регистрируемом объектном типе;

· procedure Save_Inheritance_Relations – сохранение информации о подтипах и супертипах регистрируемого объектного типа;

· function Add_Entity_From_Schema – экспортирование информации об объектном типе из другой схемы;

· function Get_Entity – выдача метаинформации об объектном типе;

· function Get_Attribute – выдача метаинформации об атрибуте, определяемом в объектном типе схемы.

Пакет lb_instance предназначен для работы с данными: занесения данных в базу, а также для извлечения данных из нее на основе запросов:

· function Init_Instance – инициация сохранения объекта;

· procedure Init_Attribute_List – инициация сохранения значений атрибутов объекта;

· procedure Put_Simple_Attribute_{R, I, S, B, L, E} – сохранение значения атрибута вещественного, целочисленного, символьного, бинарного, логического, перечислимого типа, соответственно;

· procedure Put_Association – сохранение значения атрибута ассоциативного типа;

· function Put_Aggregate – инициация сохранения элементов агрегата;

· function Put_Select – инициация сохранения селективного элемента;

· procedure Put_Element_{R, I, S, B, L, E} – сохранение значения элемента агрегатной или селективной конструкции вещественного, целочисленного, символьного, бинарного, логического, перечислимого типа, соответственно;

· procedure Get_Instances_By_ID – выборка объектов по заданным идентификаторам;

· procedure Get_Instances_By_Type – выборка объектов по заданному типу;

· procedure Get_Instances_By_Route – выборка объектов по заданному навигационному маршруту;

· procedure Add_Route_Path – метод формирования навигационного маршрута;

· procedure Get_All_Instances – выборка всех объектов;

· procedure Delete_Instances – удаление объектов по заданным идентификаторам.

До начала работы с прикладными данными соответствующие таблицы метаданных должны быть проинициализированы. С этой целью разработан CASE инструмент, позволяющий автоматически сгенерировать скрипт инициализации соответствующих таблиц на языке PL/SQL по заданной EXPRESS спецификации прикладной модели. Фрагмент данного скрипта для информационной схемы ActorResource представлен ниже.

 

declare l_Sch_ID Schemas.sch_id%TYPE; l_Ent_ID Entities.ent_id%TYPE;begin… lb_defined_types.register_defined_type ('Label','string',l_Sch_ID); lb_defined_types.register_defined_type ('ActorRole','Label',l_Sch_ID); lb_defined_types.register_defined_type ('AddressTypeEnum','enumeration',l_Sch_ID); lb_defined_types.save_enum_type('OFFICE'); lb_defined_types.save_enum_type('HOME'); lb_defined_types.save_enum_type('USERDEFINED'); l_Ent_ID:= lb_entity.register_entity ('Organization',l_Sch_ID,false); lb_entity.save_attribute('Id','integer',1,'','',0,'E'); lb_entity.save_attribute('Name','Label',2,'','',0,'E'); lb_entity.save_attribute ('Description','string',3,'','',1,'E'); lb_entity.save_attribute ('Roles','aggregate',4,'','',0,'E'); lb_entity.save_attribute ('Addresses','aggregate',5,'','',0,'E'); lb_entity.save_attribute ('IsRelatedBy','OrganizationRelationship', 6,'OrganizationRelationship','RelatedOrganizations',0,'I'); lb_entity.save_attribute ('Relates','OrganizationRelationship', 7,'OrganizationRelationship','RelatingOrganization',0,'I'); lb_entity.save_attribute ('Engages','Person',8,'Person','EngagedIn',0,'I'); l_Ent_ID:= lb_entity.register_entity ('Address',l_Sch_ID,false); lb_entity.save_attribute ('Purpose','AddressTypeEnum',1,'','',0,'E'); lb_entity.save_attribute ('UserDefinedPurpose','string',2,'','',1,'E'); lb_entity.save_attribute ('OfPerson','Person',3,'Person','Addresses',0,'I'); lb_entity.save_attribute ('OfOrganization','Organization', 4,'Organization','Addresses',0,'I'); l_Ent_ID:= lb_entity.register_entity ('PostalAddress',l_Sch_ID,false); lb_entity.save_inheritance_relations('Address',false); lb_entity.save_attribute ('AddressLines','aggregate',1,'','',0,'E');…end;

 

При непосредственной работе с данными адаптер схемо-независимой стратегии осуществляет динамическую трансляцию базовых операций манипулирования объектами того или иного типа в соответствующую последовательность вызовов PL/SQL функций и процедур. На следующем примере можно проследить логику генерации подобных последовательностей. Аналогичным образом реализуются операции модификации, удаления и поиска объектов на основе хранимых идентификаторов, объектных типов и маршрутов навигации.

-- Фрагмент исходного файла с данными в формате ISO-10303-21 #10=POSTALADDRESS(.OFFICE., $, ('25, B.Kommunisticheskaya str., Moscow, 109004, Russia'));#11=ORGANIZATION(770901001, 'ISP RAS', $, ('Research','Development','Teaching'), (#10)); -- Фрагмент PL/SQL скрипта для занесения данных в БДDECLARE type Agg_Level_Type is table of Aggregates.agg_type%TYPE index by binary_integer; type Agg_Level_ID is table of Aggregates.agg_id%TYPE index by binary_integer; l_Sch_ID Schemas.sch_id%TYPE; l_Ent_ID Entities.ent_id%TYPE; l_Ins_ID Instances.ins_id%TYPE; l_Atr_ID Attributes.atr_id%TYPE; l_Agg_Type Agg_level_Type; l_Agg_ID Agg_level_ID;… BEGIN … l_Ent_ID:= lb_entity.get_entity('Organization',l_Sch_ID); l_Ins_ID:= lb_instance.init_instance (l_Ent_ID,l_MO_ID,l_Rep_ID,'#11'); lb_instance.init_attribute_list(l_Ent_ID); l_Atr_ID:= lb_entity.get_attribute(1); lb_instance.put_simple_attribute_i (l_Ins_ID,l_Atr_ID,770901001); l_Atr_ID:= lb_entity.get_attribute(2); lb_instance.put_simple_attrib_s (l_Ins_ID,l_Atr_ID,'ISP RAS'); l_Atr_ID:= lb_entity.get_attribute(4); l_Agg_Type(1):= lb_defined_types.get_type ('ActorRole',l_Sch_ID); l_Agg_ID(1):= lb_instance.put_aggregate(l_Agg_Type(1), NULL,NULL,l_Atr_ID,l_Ins_ID,l_MO_ID,NULL,NULL); lb_instance.put_element_s(0,'Research', NULL,l_Agg_ID(1),NULL); lb_instance.put_element_s(1,'Development', NULL,l_Agg_ID(1),NULL); lb_instance.put_element_s(2,'Teaching', NULL,l_Agg_ID(1),NULL); l_Atr_ID:= lb_entity.get_attribute(5); l_Agg_Type(1):= lb_defined_types.get_type ('Address',l_Sch_ID); l_Agg_ID(1):= lb_instance.put_aggregate (l_Agg_Type(1),NULL,NULL,l_Atr_ID,l_Ins_ID, l_MO_ID,NULL,NULL); lb_instance.put_association (l_Ins_ID,l_Atr_ID,'#10',0,l_Agg_ID(1),NULL); l_Ent_ID:= lb_entity.get_entity ('PostalAddress',l_Sch_ID); l_Ins_ID:= lb_instance.init_instance (l_Ent_ID,l_MO_ID,l_Rep_ID,'#10'); lb_instance.init_attribute_list(l_Ent_ID); l_Atr_ID:= lb_entity.get_attribute(1); lb_instance.put_simple_attribute_e (l_Ins_ID,l_Atr_ID,'office'); l_Atr_ID:= lb_entity.get_attribute(3); l_Agg_Type(1):= lb_defined_types.get_type ('Label',l_Sch_ID); l_Agg_ID(1):= lb_instance.put_aggregate(l_Agg_Type(1), NULL,NULL,l_Atr_ID,l_Ins_ID,l_MO_ID,NULL,NULL); lb_instance.put_element_s(0,'25, B.Kommunisticheskaya str.,Moscow, 109004, Russia',NULL,l_Agg_ID(1),NULL);…END;

Схемо-зависимая стратегия

 

Альтернативу рассмотренному способу реализации объектно-реляционного отображения составляет разработанный вариант схемо-зависимой стратегии, основанный на использовании паттернов отображения классов и атрибутов OneInheritancePath–OneTable, Attribute–Column, ClassAssociation, ClassSelect и ClassAggregate. Данный вариант представляет собой попытку оптимизировать реляционную схему для наиболее эффективной работы с данными внутри одной прикладной модели. Подобная постановка задачи возникает довольно часто на практике и представляет интерес для приложений, оперирующих с одной, возможно масштабной, междисциплинарной прикладной моделью. Указанное сочетание паттернов отображения приводит к большому количеству таблиц, требуемых для адекватного представления объектно-ориентированных данных. Однако при этом оно обеспечивает более эффективную реализацию базовых операций манипулирования объектами. Паттерн OneInheritancePath–OneTable использует преимущества сериализованного представления атрибутов конкретных классов и упрощает компоновку наследуемых атрибутов для объектов выбранных типов. Перечисленные паттерны отображения исключают многоуровневую косвенную адресацию при доступе к таблицам атрибутов и обеспечивают высокую эффективность реализации вспомогательных операций сборки значений из таблиц атрибутов при чтении объектов и их рассылку по соответствующим таблицам при записи и модификации объектов.

Реализация адаптера посредника для схемо-зависимой стратегии существенно отличается от реализации схемо-независимой стратегии. Во-первых, реляционная схема для СУБД генерируется соответствующим CASE инструментом для каждой прикладной модели, специфицированной на языке EXPRESS. Ниже представлен фрагмент описания такой схемы на языке SQL для прикладной модели ActorResource, приведенной ниже. Во-вторых, одновременно со схемой генерируются исходные тексты пакета процедур на языке PL/SQL для манипулирования объектами данной прикладной модели. Каждая процедура пакета ориентирована на работу с объектами определенного типа и имеет специфический для него интерфейс. Поддержка подобных хранимых процедур со стороны реляционной СУБД существенно упрощает реализацию адаптера для посредника и позволяет повысить эффективность его работы за счет компиляции соответствующих директив манипулирования объектами в среде самой СУБД. В-третьих, не требуется какая-либо работа с метаданными, поскольку организация таблиц данных следует структурным особенностям прикладной информационной модели и позволяет явно адресоваться к ним при работе.

-- создание таблицы для описателей объектов схемы ActorResourceCREATE TABLE actorresource_instance (PID INTEGER DEFAULT 1 NOT NULL PRIMARY KEY, Title VARCHAR2(128) NOT NULL, Entity VARCHAR2(128) NOT NULL, Model INTEGER NOT NULL, Commentary VARCHAR2(4000), FOREIGN KEY (Model) REFERENCES model(PID) ON DELETE CASCADE); CREATE SEQUENCE sq$actorresource_instance;CREATE UNIQUE INDEX i$actorresource_title_model ON actorresource_instance (Title, Model);CREATE INDEX i$actorresource_entity_model ON actorresource_instance (Entity, Model);CREATE INDEX i$actorresource_model ON actorresource_instance (Model);
-- таблица для объектов типа OrganizationCREATE TABLE actorresource_organization (PID INTEGER DEFAULT 1 NOT NULL PRIMARY KEY, Instance INTEGER NOT NULL, Id_ INTEGER NOT NULL, Name_ VARCHAR2(255) NOT NULL, Description_ VARCHAR2(4000), FOREIGN KEY (Instance) REFERENCES actorresource_instance(PID) ON DELETE CASCADE);CREATE SEQUENCE sq$actorresource_organization;CREATE INDEX i$actorresource_organization ON actorresource_organization (Instance); CREATE TABLE actorresource_organizat_3 (PID INTEGER DEFAULT 1 NOT NULL PRIMARY KEY, Parent INTEGER NOT NULL, Element_Index1 INTEGER, Element_Value VARCHAR2(255), FOREIGN KEY (Parent) REFERENCES  actorresource_organization(PID) ON DELETE CASCADE);CREATE SEQUENCE sq$actorresource_organizat_3;CREATE INDEX i$actorresource_organizat_3 ON actorresource_organizat_3 (Parent); CREATE TABLE actorresource_organizat_4 (PID INTEGER DEFAULT 1 NOT NULL PRIMARY KEY, Parent INTEGER NOT NULL, Element_Index1 INTEGER, Element_Value VARCHAR2(128), FOREIGN KEY (Parent) REFERENCES  actorresource_organization(PID) ON DELETE CASCADE);CREATE SEQUENCE sq$actorresource_organizat_4;CREATE INDEX i$actorresource_organizat_4 ON actorresource_organizat_4 (Parent);…

BLOB стратегия

 

Наконец, третий разработанный вариант реализации адаптера основан на применении BLOB&Text&XML_Encoding паттернов для реляционного представления объектов классов и их атрибутов. Этот вариант воспроизводит упрощенную схему СН стратегии за счет упакованного представления значений атрибутов объекта в виде бинарной или текстовой строки. Сами строки хранятся как элементы записей в таблице объектов BLOB_Instances. Как следствие, таблицы представления простых и сложных атрибутов отсутствуют. Модифицированная таблица Associations хранит ассоциации всех видов и используется для реализации навигационных запросов по ним. Из таблиц метаданных поддерживаются лишь Schemas, Entities, Entities_To_Schemas, Inheritance_Relations, Complex_Entities и Complex_Entities_To_Entities, записи которых воспроизводят отношения наследования классов в прикладной модели и используются для реализации запросов по типам объектов. Соответствующий CASE инструмент позволяет сгенерировать скрипт инициализации таблиц метаданных по спецификации прикладной информационной модели на языке EXPRESS.

Разработанный на языке PL/SQL пакет процедур предоставляет базовый набор операций манипулирования объектами и их поиска по идентификаторам, типам и навигационным маршрутам в рамках BLOB стратегии. Поскольку значения атрибутов объекта представлены в БД единой строкой, функции по упаковке и распаковке строк целиком ложатся на адаптер посредника. В силу этой же причины в рамках BLOB стратегии невозможно выполнение более тонких запросов на основе атрибутных свойств объектов непосредственно средствами СУБД.

 


Рекомендации использования

 

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

Так, приведенный пример реализации схемо-независимой стратегии объектно-реляционного отображения, основанный на фиксированной системе таблиц, может быть рекомендован для использования в приложениях, оперирующих одновременно с несколькими перманентно изменяемыми моделями либо с масштабными промышленными моделями, включающими тысячи классов. Однако эффективность выполнения запросов, а также базовых операций манипулирования объектами при использовании данной стратегии оказывается низкой, поскольку их реализация связана с необходимостью дополнительного анализа таблиц метаданных, а также сборки значений атрибутов объектов из нескольких таблиц при чтении и их обратной рассылки по таблицам при записи.

Частично компенсировать данные недостатки, а также сократить количество необходимых реляционных таблиц позволяет упрощенный вариант реализации схемо-независимой стратегии, основанный на представлении значений атрибутов, упакованных в единую бинарную или текстовую строку (BLOB) и хранимых как элементы записей в таблице объектов. Недостатком данной стратегии является невозможность непосредственной реализации запросов и базовых операций манипулирования объектами средствами самой СУБД. Вся нагрузка здесь ложится на промежуточный слой, выполняющий операции упаковки/распаковки строк со значениями атрибутов. Реализованный вариант BLOB стратегии, описанный в настоящей статье, позволяет разгрузить слой-посредник и выполнить простые запросы по идентификаторам объектов, их типам, а также навигационным маршрутам средствами СУБД, поскольку использует дополнительную систему таблиц для хранения отношений наследования и ассоциативных связей между отдельными объектами.

Наиболее эффективную реализацию запросов и операций манипулирования объектами обеспечивает разработанный вариант схемо-зависимой стратегии, основанный на сериализованном представлении атрибутов конкретных классов. Данная стратегия рекомендуется для использования в приложениях, оперирующих с одной прикладной моделью, включающей несколько сотен классов. Ее недостатками являются большое количество используемых таблиц, критичное для большинства реализаций современных реляционных СУБД, чувствительность к эволюции прикладной модели, а также необходимость применения CASE инструментария для генерации реляционной схемы и процедур, реализующих запросы и операции манипулирования объектами. Подобный инструментарий позволяет существенно упростить сопровождение и администрирование базы данных, эксплуатирующей данную стратегию объектно-реляционного отображения.

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

 


Заключение

 

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

Язык EXPRESS:

· опирается на объектно-ориентированный подход

· использует разбиение на иерархические уровни

Язык EXPRESS может быть использован двумя путями:

1. прямое использование алгоритмов языка EXPRESS; применение программных средств, а также использование оболочки EXPRESS, с помощью которой создается информационная модель

2. моделирование понятий и функциональных (информационных) связей отдельно; проектирование информационной модели включает 3 этапа:

А) информационное моделирование

Б) функциональное моделирование

В) программная реализация

Второй путь является наиболее предпочтительнее для CALS, т.к. есть разделение функциональных обязанностей.

Информационная модель на языке EXPRESS описывается с помощью схемы, которая может включать в свой состав следующие элементы:

1) описание типов

2) описание констант (ввод постоянных)

3) создание правил

4) функции

5) процедуры

Функции и процедуры необходимы для проверки правил, для вычисления каких-то переменных.

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

Язык EXPRESS включил 2 особенности, которых нет у других программных средств:

1) механизм множественного наследования (генетический механизм). С помощью объявлений можно указать список сущностей, которые являются предками этой сущности, от которой она наследует свойства: атрибуты, правила, алгоритмы, постоянные и т.д. EXPRESSследование осуществляется транзитивно (значит выполняются логические операции, в результате которых меняются свойства у взаимодействующих объектов).

2) Использование механизма мутации. При наличии в одной схеме нескольких подтипов определенной сущности считается, что в популяции этой сущности возможны объекты с характерными свойствами.

 



Поделиться:


Последнее изменение этой страницы: 2020-03-02; просмотров: 157; Нарушение авторского права страницы; Мы поможем в написании вашей работы!

infopedia.su Все материалы представленные на сайте исключительно с целью ознакомления читателями и не преследуют коммерческих целей или нарушение авторских прав. Обратная связь - 18.191.174.168 (0.039 с.)