|
Web-доступ к базам данных
Базы данных выполняют функцию систематизации
знаний. На основе этой систематизации могут создаваться новые знания. Так или
иначе, любая база данных служит человеку именно для описания происшедших в
прошлом событий и на основе знания этих событий помогает принять то или иное
решение на будущее. База знаний может быть построена как мультимедийный
справочник или как набор текстов и файлов другого формата, проиндексированных по
определенным признакам в базе данных. База данных – это, прежде всего,
хранилище объектов данных, т.е. набора возможных понятий или событий,
описываемых базой данных, с возможностью поиска этих объектов по признакам.
Неотъемлемой чертой базы данных является возможность связывания объектов между
собой. Базой данных можно считать не только таблицы, индексирующие файлы со
знаниями разных форматов, но и сами эти файлы, потому, что они являются не
типизированными хранилищами знаний в такой базе данных. Итак, в базах знаний
мы накапливаем опыт прошлого. Потом человек может сам принять решение на основе
этого опыта (типичный случай с мультимедийным справочником) или поставить задачу
перед базой данных по поиску решения согласно сложившейся ситуации (найти закон,
поясняющий правило оформления таможенной декларации и т.п.). Так происходит в
программах справочного характера. Как частный случай баз данных, можно
рассматривать различные структурированные файлы, например, словари для
переводчиков, форматы файлов RTF, DOC, книги Microsoft Excel, файлы с письмами
для почтовых Internet-программ и т.д., жизненно важные функции баз данных, в
которых реализуются за счет внутренних функций программ работающих с ними. Базы
данных могут применяться как вспомогательное средство, позволяющее реализовать
какую-то полезную функцию. Например, хранение настроек программы, Internet-
адресов для рассылки рекламы и т.д. Для
построения информационных систем применяются базы данных, созданные вокруг ядра
базы данных. Работа с базой данных происходит, как правило, в
многопользовательском режиме, т.е. программа должна быть сетевой. В связи с
этим, необходимо обеспечить разделение прав доступа различным пользователям к
данным, правильность завершения транзакций, т.е. ссылочную целостность,
ограничения и другие правила, реализуемые через встроенные средства сервера базы
данных. К тому же, должна быть обеспечена приемлемая производительность
информационной системы. В центре всей информационной системы стоит сервер базы
данных. Он обеспечивает низкоуровневый доступ к таблицам базы данных, в которых
и хранится информация об объектах базы данных. Ядром информационной системы в
простейшем случае могут выступать несколько функций, реализованных в программе
программистом. В современном мире чаще всего применяется сервер приложений для
реализации ядра информационной системы. В распределенной вычислительной системе
сервер приложений берет на себя функцию распределения нагрузки между серверами,
которые в общем случае могут работать под разными операционными системами, или
находится в разных географически местах. Сервер приложений – это мостик между
программами-клиентами и одним или несколькими серверами базы данных. За счет
сервера приложений можно снизить нагрузку на приложения пользователя и
реализовать сложные правила объектной модели базы данных, которые трудно или
нерационально реализовывать на стороне сервера базы данных. В результате, сервер
приложений снижает трафик между сервером базы данных и компьютером клиента,
повышая общую производительность информационной системы. Исходя из сказанного
ранее, на приложение пользователя остается только реализация интерфейса. Такая
структура информационной системы называется многозвенной, а приложение
пользователя – тонким клиентом. Надо отметить, что в общем случае серверы
приложений могут посылать команды друг другу, и взаимодействовать, таким
образом, самым рациональным способом с географически удаленными серверами баз
данных. Например, для получения отчета с большим количеством вычисляемых полей,
нет необходимости делать несколько запросов к удаленной базе данных через
Internet, если это может сделать сервер приложений, находящийся в
непосредственной близости от сервера базы данных. Он и пошлет в ответ готовый
отчет. Таким образом, только информационная система, построенная по принципу
многозвенности, может удовлетворять наиболее полным образом условиям наивысшей
производительности при полной коммуникабельности и распределенности вычислений.
Система, построенная из нескольких отдельных модулей, выполняющих ряд
определенных задач, к тому же, может быть проще модифицируемой. Первой и самой важной функцией базы данных, является
функция хранения информации. Информация должна хранится упорядоченно для более
быстрого и понятного пользователю доступа к ней. Упорядоченность информации в
базе данных, помимо удобств доступа, может привести к значительному сокращению
аппаратных ресурсов, необходимых для ее обслуживания. Упорядоченность
достигается путем нормализации. Здесь мы вплотную подошли ко второй функции
базы данных – ввод информации. Какую информацию будет вводить пользователь?
Хорошая база данных построена из главного документа, справочников, из которых
пользователь вводит информацию и нескольких полей для ручного ввода, например,
текстов назначения платежа в платежных поручениях и суммы. База данных должна
заполняться средствами, наиболее полно автоматизирующими этот процесс. При этом
плохим тоном являются: ввод одной и той же информации в
нескольких местах; Одной из основных функций базы данных является
автоматизация. Под автоматизацией, как правило, понимают автоматическое создание
выходных документов и пересчет данных, например печать накладной, счета фактуры
и протокола согласования цен в складской программе для исходящей накладной.
Далее, нужно вспомнить о системах принятия решений. Информационная система
должна позволять создавать статистические отчеты в реальном режиме времени о
состоянии описываемого в базе данных процесса. Эта функция удобна для
руководителей подразделений, которые могут прогнозировать поведение описываемой
системы на основе статистических данных, полученных из базы данных.
Собственно, описанные выше функции информационной системы являются
"джентльменским набором", которого достаточно в большинстве случаев. Из
дополнительных функций необходимо упомянуть возможность поиска по нескольким
взаимосвязанным характеристикам. В единой информационной системе необходима
возможность идентификации пользователя с целью ограничения доступа пользователя
к определенным частям базы данных и введения информации о создателе документа и
лиц, редактировавших его. Это придаст пользователям ощущение ответственности за
выполняемые действия. Хорошая информационная система должна легко расширяться
при необходимости добавления в нее новых возможностей. Расширяемость
подразумевает элементы объектной ориентированности, встроенные в базу данных.
Настраивая эти объекты, возможно вносить незначительные изменения в структуру
базы данных, что продляет срок морального устаревания всей информационной
системы. Одним из факторов расширяемости является возможность сочленять
разнородные базы данных в единый комплекс. Такая возможность сейчас реализуется
через дополнительные модули, которые по своей сути являются серверами
приложений, или правильное построение базы данных по классическим реляционным
законам. Последний случай затрудняется тем, что некоторые серверы базы данных не
могут выполнить один SQL запрос к разным базам данных, тем более находящимся в
географической удаленности друг от друга. Описанные выше функции в
разных реализациях информационных систем имеют специфические черты,
ориентированные на конкретное прикладное применения. Любую
структуру данных можно преобразовать в простую двумерную таблицу. Такое
представление является наиболее удобным и для пользователя, и для машины, -
подавляющее большинство современных информационных систем работает именно с
такими таблицами. Базы данных, которые состоят из двумерных таблиц, называются
реляционными. Основная идея реляционного подхода состоит в том, чтобы
представить произвольную структуру данных в виде простой двумерной таблицы или,
как говорят, нормализовать структуру. Из всех систем баз данных реляционные
относятся к самым распространенным в мире. Эти системы способны разрешить многие
из тех проблем, которые усложняли работу с нереляционными продуктами прежних
поколений. Программисты и администраторы таких баз данных были вынуждены
тщательно изучать, как структурирована информация и как она представлена в базе
данных, что значительно усложнило разработку этих приложений и модификацию самих
программ. Реляционные системы способны работать на более высоком уровне. Все
операции с данными реализуются программой, называемой DBMS (система управления
базой данных Обращаться к ней можно только с помощью операторов языка высокого
уровня. Хотя некоторые продукты по-прежнему поддерживают работу в терминах своих
собственных языков, язык SQL (Standard Query Language) стал тем технологическим
стандартом, на базе которого созданы все, более или менее известные, реляционные
продукты. Язык для взаимодействия с БД SQL появился в середине 70-х и был
разработан в рамках проекта экспериментальной реляционной СУБД System R.
Исходное название языка SEQUEL (Structered English Query Language) только
частично отражает суть этого языка. Конечно, язык был ориентирован главным
образом на удобную и понятную пользователям формулировку запросов к реляционной
БД, но на самом деле уже являлся полным языком БД, содержащим помимо операторов
формулирования запросов и манипулирования БД средства определения и
манипулирования схемой БД. В языке отсутствовали средства синхронизации доступа
к объектам БД со стороны параллельно выполняемых транзакций: с самого начала
предполагалось, что необходимую синхронизацию неявно выполняет СУБД. Все значения данных состоят из простых типов данных. В SQL
отсутствуют массивы, указатели, векторы и другие сложные типы данных. Все
данные в реляционной базе данных изображаются в форме двумерных таблиц (на языке
математики – "отношений"). Каждая таблица содержит некоторое число строк (в том
числе 0), называемых "картежами" и один или несколько столбцов, называемых
"атрибутами". Все строки в таблице имеют одну и ту же последовательность
столбцов, в которых записаны различные значения, однако наборы значений в
столбцах отличаются. После того как данные введены в БД, можно сравнивать значения в
различных столбцах (в том числе и для разных таблиц) или объединять строки, в
которых найдено совпадение. Это позволяет соотносить между собой строки и
производить очень сложные операции обработки над всеми данными, находящимися в
базе. Все операции определяются только логикой, а не положением строки в
таблице. Например, можно запросить все строки, со значением 2 и не возможно
запросить первую или, третью или пятую строку. Строки в реляционной базе данных
расположены в произвольном порядке. Он не обязательно соответствует тому
порядку, в котором они были занесены или в котором хранятся на диске.
Поскольку невозможно определить строку по ее положению (порядку в таблице),
необходимо иметь один или несколько столбцов с уникальным значением для
идентификации каждой строки. Эти столбцы называются первичными ключами
таблицы. Иван И. Иванов Константин В. Волков Одним из преимуществ реляционного подхода к
построению БД – отсутствие необходимости заботится о таких деталях, как способы
представления данных или их физическое размещение в самой базе. Старые
иерархические и сетевые базы данных, в которых приходилось иметь дело с
подобными вопросами реализации, имели громоздкую структуру и были сложными в
управлении. Увеличение объема и
структурной сложности хранимых данных, расширение круга пользователей
информационных систем привели к широкому распространению наиболее удобных и
сравнительно простых для понимания реляционных (табличных) СУБД. Для обеспечения
одновременного доступа к данным множества пользователей, нередко расположенных
достаточно далеко друг от друга и от места хранения баз данных, созданы сетевые
мультипользовательские версии СУБД. В них тем или иным путем решаются
специфические проблемы параллельных процессов, целостности (правильности) и
безопасности данных, а также санкционирования доступа. SQL стал
унифицированным средством общения и стандартным языком манипулирования с базами
данных, обладающим средствами для реализации перечисленных выше возможностей.
После появления на рынке двух пионерских СУБД – SQL/DS (1981 год) и DB2 (1983
год) – он приобрел статус стандарта де-факто для профессиональных реляционных
СУБД. В 1987 году SQL стал официальным международным стандартом языка баз
данных, а в 1992 году вышла вторая версия этого стандарта. Важной
отличительной чертой SQL является его независимость от компьютерной среды
(операционной системы и архитектуры). Такой язык назвали SQL – это аббревиатура
структурированного языка запросов (Structured Query Language). SQL является
инструментом, предназначенным для обработки и чтения информации, содержащейся в
компьютерной базе данных. При создании языка запросов нового поколения
разработчики старались сделать его простым и легким в освоении инструментом для
обращения к БД. В итоге SQL стал слабо структурированным языком, особенно по
сравнению с такими языками, как С или Pascal, и в то же время достаточно мощным
и относительно легким для изучения. Одним из наиболее важных шагов на пути к признанию SQL на рынке
стало появление стандартов на этот язык. Обычно при упоминании стандарта SQL
имеют в виду официальный стандарт, утвержденный Американским институтом
национальных стандартов (American National Standards Institute — ANSI) и
Международной организацией по стандартам (International Standards Organization—
ISO). Однако существуют и другие важные стандарты SQL, включая SQL,
реализованный в системе DB2 компании IBM, и стандарт X/OPEN для SQL в среде
UNIX. Работа над официальным стандартом SQL началась в 1982 году, когда ANSI
поставил перед своим комитетом ХЗН2 задачу по созданию стандарта языка
реляционных баз данных. Вначале в комитете обсуждались достоинства различных
предложенных языков. Однако поскольку к тому времени SQL уже стал фактическим
стандартом, комитет ХЗН2 остановил свой выбор на нем и занялся стандартизацией
SQL. Разработанный в результате стандарт в большой степени был основан на
диалекте SQL системы DB2, хотя и содержал в себе ряд существенных отличий от
этого диалекта. После нескольких доработок, в 1986 году стандарт был официально
утвержден как стандарт ANSI номер Х3.135, а в 1987 году — в качестве стандарта
ISO. Затем стандарт ANSI/ISO был принят правительством США как федеральный
стандарт США по обработке информации (FIPS — Federal Information Processing
Standard). Этот стандарт, незначительно пересмотренный в 1989 году, обычно
называют стандартом "SQL-89", или "SQLI". Когда в данном реферате я упоминаю
"стандарт ANSI/ISO", то подразумеваю SQLI, который в настоящее время лежит в
основе большинства коммерческих продуктов. Многие из членов комитетов по
стандартизации ANSI и ISO представляли фирмы-поставщики различных СУБД, в каждой
из которых был реализован собственный вариант SQL. Как и диалекты человеческого
языка, диалекты SQL были в основном похожи друг на друга, однако несовместимы в
деталях. Во многих случаях комитет просто обошел существующие различия и не
стандартизировал некоторые части языка, определив, что они реализуются по
усмотрению разработчика. Этот подход позволил объявить большое число реализаций
SQL совместимыми со стандартом, однако сделал сам стандарт относительно
слабым. Чтобы заполнить эти пробелы, комитет ANSI продолжил свою работу и
создал проект нового, более жесткого стандарта SQL2. В отличие от стандарта 1989
года, проект SQL2 предусматривал возможности, выходящие за рамки таковых, уже
существующих в реальных коммерческих продуктах. А для следующего за ним
стандарта SQL3 были предложены еще более глубокие изменения. В результате
предложенные стандарты SQL2 и SQL3 оказались более противоречивыми, чем исходный
стандарт. Стандарт SQL2 прошел процесс утверждения в ANSI и был окончательно
принят в октябре 1992 года. В то время, как первый стандарт 1986 года занимает
не более ста страниц, официальный стандарт SQL2 содержит около
шестисот. Вопреки стандарту SQL2, во всех существующих на сегодняшний день
коммерческих продуктах поддерживаются собственныедиалекты SQL. Более того,
поставщики СУБД включают в свои продукты новые возможности и расширяют
собственные диалекты SQL, чем еще больше отдаляют их от стандарта. Однако ядро
SQL стандартизировано довольно жестко. Там, где это можно было сделать, не
ущемляя интересы клиентов, поставщики СУБД привели свои продукты в соответствие
со стандартом SQL-89, то же самое постепенно произошло и с SQL2. Хотя стандарт
ANSI/ISO наиболее широко распространен, он не является единственным стандартом
SQL. Европейская группа поставщиков X/OPEN также приняла SQL в качестве одного
из своих стандартов для "среды переносимых приложений" на основе UNIX. Стандарты
группы X/OPEN играют важную роль на европейском компьютерном рынке, где основной
проблемой является переносимость приложений между компьютерными системами
различных производителей. К несчастью, стандарт X/OPEN отличается от стандарта
ANSI/ISO. Кроме того, компания IBM включила SQL в свою спецификацию Systems
Application Architecture (архитектура прикладных систем) и пообещала, что все ее
продукты, очевидно, будут переведены на этот диалект SQL. Хотя данная
спецификация и не оправдала надежд на унификацию линии продуктов компании IBM,
движение в сторону унификации SQL в IBM продолжается. Система DB2 остается
основной СУБД компании IBM для мэйнфреймов. Однако компания выпустила реализацию
DB2 и для OS/2 (собственной операционной системы для персональных компьютеров),
и для линии серверов и рабочих станций RS/6000, работающих под управлением UNIX.
Таким образом, диалект DB2 языка SQL является мощным стандартом де-факто. В
технологии баз данных существует важная область, которую не затрагивают
официальные стандарты. Это способность к взаимодействию с другими базами данных
— методы, с помощью которых различные БД могут обмениваться информацией (как
правило, по сети). В 1989 году несколько поставщиков сформировали консорциум SQL
Access Group специально для решения этой проблемы. В 1991 году консорциум
опубликовал спецификацию RDA (Remote Database Access — удаленный доступ к базам
данных). Эта спецификация тесно связана с протоколами OSI, которые не смогли
завоевать широкого признания, поэтому она оказывает на рынок незначительное
влияние. Прозрачность взаимодействия между различными базами данных остается
иллюзорной мечтой. Тем не менее, второй стандарт от SQL Access Group имеет на
рынке больший вес. В результате настойчивых требований компании Microsoft,
консорциум SQL Access Group включил в стандарт SQL интерфейс вызовов функций.
Полученная спецификация CLI (Call Level Interface), основанная на разработках
компании Microsoft, увидела свет в 1992 году. В этом же году была опубликована
собственная спецификация ODBC (Open Database Connectivity — взаимодействие с
открытыми базами данных) компании Microsoft, основанная на стандарте CLI.
Благодаря рыночной силе Microsoft и благословению, полученному "открытым
стандартом" от SQL Access Group, ODBC оказался стандартом де-факто для
интерфейсов доступа к базам данных на персональных компьютерах. Весной 1993 года
компании Apple и Microsoft объявили о соглашении относительно поддержки ODBC в
MacOS и Windows, что закрепило за этой спецификацией статус стандарта в обеих
популярных средах с графическим пользовательским интерфейсом. Появление
стандарта SQL вызвало довольно много восторженных заявлений о переносимости SQL
и использующих его приложений. На самом деле пробелы в стандарте SQL-89 и
различия между существующими диалектами SQL достаточно значительны, и при
переводе приложения под другую СУБД его всегда приходится модифицировать. Эти
отличия, большинство из которых устранено в стандарте SQL2, включают в
себя: Коды ошибок. В стандарте SQL-89 не определены коды, которые возвращают
операторы SQL при возникновении ошибок, и в каждой из коммерческих реализаций
используется собственный набор таких кодов. В стандарте SQL2 определены
стандартные коды ошибок. Типы данных. В стандарте SQL-89 определен минимальный
набор типов данных,однако, в нем отсутствуют некоторые из наиболее
распространенных и полезных типов, например символьные строки переменной длины,
дата и время, а также денежные единицы. В стандарте SQL2 упомянуты эти типы
данных, однако, отсутствуют "новые" типы данных, такие как графические и
мультимедийные объекты. Системные таблицы. В стандарте SQL-89 умалчивается о
системных таблицах, в которых содержится информация о структуре самой базы
данных. Поэтому каждый поставщик создавал собственные системные таблицы, и их
структура отличается даже в четырех реализациях SQL компании IBM. Системные
таблицы стандартизированы в SQL2. Интерактивный SQL. В стандарте определен
только программный SQL, используемый прикладной программой, но не интерактивный
SQL. Например, оператор select, предназначенный для выполнения запросов к базе
данных в интерактивном режиме, в стандарте отсутствует. Программный интерфейс.
В первом стандарте определен абстрактный способ использования SQL в программах,
написанных на таких языках программирования, как COBOL, FORTRAN и другие. Этот
способ не используется ни в одном коммерческом продукте, а в существующих
программных интерфейсах имеются значительные отличия. В стандарте SQL2 определен
интерфейс встроенного SQL для популярных языков программирования, но не
интерфейс вызова функций. Динамический SQL. В стандарте SQL-89 не описаны
элементы SQL, необходимые для разработки приложений общего назначения, таких как
генераторы отчетов и программы создания и выполнения запросов. Однако эти
элементы, известные под названием динамический SQL, имеются почти во всех СУБД и
в различных системах значительно отличаются. В SQL2 входит стандарт
динамического SQL. Семантические отличия. Поскольку некоторые элементы
определены в стандартах как зависящие от реализации, может возникнуть ситуация,
когда в результате выполнения одного и того же запроса в двух совместимых СУБД
будут получены два различных набора результатов. Отличия результатов обусловлены
различиями в обработке значений null, разными агрегатными функциями и
несовпадением процедур удаления повторяющихся строк. Последовательность
сравнения. В стандарте SQL-89 не упоминаются последовательности сравнения
(сортировки) символов, хранящихся в базе данных. Результаты запроса с
сортировкой будут отличаться при выполнении этого запроса на персональном
компьютере (с кодировкой ASCII) и на мэйнфрейме (с кодировкой EBCDIC). Стандарт
SQL2 позволяет программе или пользователю указывать требуемую последовательность
сортировки. Структура базы данных. В стандарте SQL-89 определен SQL, который
используется уже после того, как база данных была открыта и подготовлена к
работе. Детали наименования баз данных и первоначального подключения к ним
сильно отличаются и несовместимы. Стандарт SQL2 в некоторой степени унифицирует
этот процесс, но не может полностью ликвидировать все отличия. Вопреки
перечисленным различиям, в начале 90-х годов стали появляться коммерческие
программы, реализующие переносимость приложений между различными СУБД. Однако в
таких программах для каждой из поддерживаемых СУБД требуется специальный
конвертер, который генерирует код в соответствии с определенным диалектом SQL,
выполняет преобразование типов данных, транслирует коды ошибок и т.д.
"Прозрачная" переносимость между различными СУБД, использующими SQL, является
основной целью стандарта SQL2 и протокола ODBC. Однако повсеместный,
"прозрачный" и унифицированный доступ к базам данных SQL остается делом
будущего. Всемирная
Паутина недаром так быстро завоевала широкую популярность среди пользователей
Internet, в мире бизнеса, науки, политики и т. д. Основные достижения Web – это
простота опубликования информации в сети, удобство и сравнительная
унифицированность доступа к документам, наличие на сегодняшний день достаточно
развитых средств поиска. Однако в целом способы представления, хранения и поиска
информации в WWW относятся к категории информационно-поисковых систем (ИПС).
Хотя хранилища данных в узлах Web иногда называют базами данных, этот термин в
данном случае можно использовать только в самом широком смысле. Исторически ИПС
применялись для хранения слабоструктурированной и редко изменяемой информации.
Базы данных в узком смысле – это хранилища структурированной, изменяемой
информации, причем информация в базе данных должна всегда находиться в
согласованном состоянии. С равным успехом можно хвалить и ругать Web. Можно
хвалить Всемирную Паутину за то, что, не выходя из дома, вы можете побывать в
любой точке земного шара и посмотреть, что же там происходит. Можно ругать Web
за то, что трудно найти действительно актуальную информацию (обычно она
устаревшая), за то, что хранилища информации содержат очень много "мусора",
опубликованного непонятно из каких соображений. Но в любом случае интерфейс
действительно удобен. Ситуация с базами данных кардинально отличается. Именно
базы данных содержат основные знания человечества. В конце двадцатого века с
появлением технологии баз данных мы накопили больше информации, чем за всю
предыдущую историю. Вся беда в том, что доступ к базам данных (даже к тем,
которые содержат полностью открытую информацию) ограничен. Чтобы получить
интересующую его информацию, пользователь должен иметь физический доступ к
соответствующей СУБД, быть в курсе модели данных, знать схему базы данных и,
наконец, уметь пользоваться соответствующим языком запросов. Что касается языка
запросов, то проблему частично решает протокол ODBC, позволяющий направлять
ограниченный набор операторов SQL (с промежуточной обработкой соответствующим
драйвером ODBC) к произвольному серверу баз данных. Но это только частичное
решение, поскольку оно никак не помогает пользователю понять схему базы данных
(даже в терминах SQL) и, конечно, не способствует созданию унифицированного
интерфейса конечного пользователя (нельзя же заставить всех работать в строчном
режиме на языке SQL). Итак, мы имеем удобные средства разработки
распределенных в Internet гипермедийных документов, простые, удобные, развитые и
унифицированные интерфейсы для доступа к информации WWW. Кроме того, мы имеем
большое количество ценных баз данных, управляемых разнородными СУБД, а также
желание сделать эти базы доступными всем людям (в случае публичных баз данных)
или членам территориально-распределенной корпорации (в случае корпоративных баз
данных). Возникает естественное желание скрестить эти две технологии и
обеспечить доступ к базам данных в интерфейсе Web. Еще два года назад
существовали только идеи такого скрещивания и не очень тщательно разработанные
подходы к реализации. На сегодняшний день такие механизмы уже существуют и
используются. SQL является инструментом,
предназначенным для обработки и чтения данных, содержащихся в компьютерной базе
данных. SQL (структурированный язык запросов) как следует из названия, является
языком программирования, который применяется для организации взаимодействия
пользователя с базой данных. На самом деле SQL работает только с базами данных
одного определенного типа, называемых реляционными. На рисунке 2.1
изображена схема работы SQL. Согласно этой схеме, в вычислительной системе
имеется база данных, в которой хранится важная информация. Если БД относится к
сфере бизнеса, то в ней может храниться информация о материальных ценностях,
выпускаемой продукции, объемах продаж и зарплате. В базе данных на персональном
компьютере может храниться информация о выписанных чеках, телефонах и адресах
или информация, извлеченная из более крупной вычислительной системы.
Компьютерная программа, которая управляет базой данных, называется системой
управления базой данных, или СУБД. Если пользователю необходимо прочитать
данные из базы данных, он запрашивает их у SQL с помощью СУБД. SQL обрабатывает
запрос, находит требуемые данные и посылает их пользователю. Процесс
запрашивания данных и получения результата называется запросом к базе данных:
отсюда и название — структурированный язык запросов. Однако это название не
совсем соответствует действительности. Cегодня SQL представляет собой нечто
гораздо большее, чем простой инструмент создания запросов, хотя именно для этого
он и был первоначально предназначен. Несмотря на то, что чтение данных по-
прежнему остается одной из наиболее важных функций SQL, сейчас этот язык
используется для реализации всех функциональных возможностей, которые СУБД
предоставляет пользователю, а именно: Организация данных. SQL дает
пользователю возможность изменять структуру представления данных, а также
устанавливать отношения между элементами базы данных. Чтение данных. SQL дает
пользователю или приложению возможность читать из базы данных содержащиеся в ней
данные и пользоваться ими. Обработка данных. SQL дает пользователю или
приложению возможность изменять базу данных, т.е. добавлять в нее новые данные,
а также удалять или обновлять уже имеющиеся в ней данные. Управление доступом.
С помощью SQL можно ограничить возможности пользователя по чтению и изменению
данных и защитить их от несанкционированного доступа. Совместное использование
данных. SQL координирует совместное использование данных пользователями,
работающими параллельно, чтобы они не мешали друг другу. Целостность данных.
SQL позволяет обеспечить целостность базы данных, защищая ее от разрушения из-за
несогласованных изменений или отказа системы. Основными объектами
реляционной базы данных являются: Прямоугольная таблица,
состоящая из СТРОК и СТОЛБЦОВ. Задать таблицу – значит указать, из каких
столбцов она состоит. Запись, состоящая из полей – столбцов. В
каждом поле содержится его значение, либо значение NULL – "пусто". Строк в
таблице может быть сколько угодно. Физический порядок их расположения друг
относительно друга неопределен. В реляционной базе данных
информация организована в виде таблиц, разделённых на строки и столбцы, на
пересечении которых содержатся значения данных. Используемые в языке SQL для
запросов сочетания ключей (CREATE TABLE my_table – создание таблицы с названием
my_table) получили название "предложение". Таблицы создаются в SLQ с помощью
предложения CREATE TABLE. Предложение CREAT TABLE специфицирует имя базовой
таблицы, которая должна быть создана, имена ее столбцов и типы данных для этих
столбцов. CREAT TABLE – выполняемое предложение. Если SQL-серверу дать запрос
CREATE TABLE, система построит таблицу, которая сначала будет пустой: она будет
содержать только строку заголовков столбцов, но не будет еще содержать никаких
строк с данными. Информация в таблицу вставляется при помощи предложения команды
INSERT Все запросы на получение практически любых
данных из одной или нескольких таблиц выполняются с помощью единственного
предложения SELECT. звездочка (*) для обозначения
"все" – употребляется в обычном для программирования смысле, т.е. "все случаи,
удовлетворяющие определению"; квадратные скобки ([]) – означают, что
конструкции, заключенные в эти скобки, являются необязательными (т.е. могут быть
опущены); фигурные скобки ({}) – означают, что конструкции,
заключенные в эти скобки, должны рассматриваться как целые синтаксические
единицы, т.е. они позволяют уточнить порядок разбора синтаксических конструкций,
заменяя обычные скобки, используемые в синтаксисе SQL; многоточие
(…) – указывает на то, что непосредственно предшествующая ему синтаксическая
единица факультативно может повторяться один или более раз; прямая
черта () – означает наличие выбора из двух или более возможностей. Например,
обозначение ASCDESC указывает, можно выбрать один из терминов ASC или DESC;
когда же один из элементов выбора заключен в квадратные скобки, то это означает,
что он выбирается по умолчанию (так, [ASC]DESC означает, что отсутствие всей
этой конструкции будет восприниматься как выбор ASC); запятая (,) –
используется для разделения элементов списков; пробелы ( ) – могут
вводиться для повышения наглядности между любыми синтаксическими конструкциями
предложений SQL; жирные прописные латинские буквы и символы –
используются для написания конструкций языка SQL и должны (если это специально
не оговорено) записываться в точности так, как показано-……..;
строчные буквы используются для написания конструкций, которые
должны заменяться конкретными значениями, выбранными пользователем, причем для
определенности отдельные слова этих конструкций связываются между собой символом
подчеркивания (_); термины "таблица" и "столбец" заменяют (с целью
сокращения текста синтаксических конструкций) термины "имя_таблицы",
"имя_столбца", …, соответственно; термин "таблица" - используется
для обобщения таких видов таблиц, как базовая_таблица, представление или
псевдоним; здесь псевдоним служит для временного (на момент выполнения запроса)
переименования и (или) создания рабочей копии базовой_таблицы (представления).
[ORDER BY {[таблица.]столбец номер_элемента_SELECT}
[[ASC] DESC] и позволяет объединить (UNION) а затем упорядочить (ORDER BY) результаты
выбора данных, полученных с помощью нескольких "подзапросов". При этом
упорядочение можно производить в порядке возрастания – ASC (ASCending) или
убывания DESC (DESCending), а по умолчанию принимается ASC. SELECT (выбрать) данные из указанных столбцов и (если
необходимо) выполнить перед выводом их преобразование в соответствии с
указанными выражениями и (или) функциями (где) строки из указанных таблиц
должны удовлетворять указанному перечню условий отбора строк (группируя по) указанному перечню столбцов с тем, чтобы получить для каждой
группы единственное агрегированное значение, используя во фразе SELECT SQL-
функции SUM (сумма), COUNT (количество), MIN (минимальное значение), MAX
(максимальное значение) или AVG (среднее значение) и имеет формат FROM {базовая_таблица представление} [псевдоним] Элемент_SELECT – это одна из следующих
конструкций: [таблица.]столбец (выражение) константа
переменная а синтаксис SQL_функций – одна из
следующих конструкций: COUNT(*) WHERE [NOT] WHERE_условие [[ANDOR][NOT] WHERE_условие]… значение { = <> <
<= > >= } { значение ( подзапрос ) } значение [NOT] IN { ( константа [,константа]… )
( подзапрос ) } EXISTS ( подзапрос ) Кроме
традиционных операторов сравнения (= <> < <= > >=) в
WHERE фразе используются условия BETWEEN (между), LIKE (похоже на), IN
(принадлежит), IS NULL (не определено) и EXISTS (существует), которые могут
предваряться оператором NOT (не). Критерий отбора строк формируется из одного
или нескольких условий, соединенных логическими операторами: когда
должно удовлетворяться одно из разделяемых с помощью OR условий; когда или должно удовлетворяться первое условие или не должно удовлетворяться
второе, причем существует приоритет AND над OR (сначала выполняются все
операции AND и только после этого операции OR). Для получения желаемого
результата WHERE условия должны быть введены в правильном порядке, который можно
организовать введением скобок. При обработке условия числа сравниваются
алгебраически – отрицательные числа считаются меньшими, чем положительные,
независимо от их абсолютной величины. Строки символов сравниваются в
соответствии с их представлением в коде, используемом в конкретной СУБД,
например, в коде ASCII. Если сравниваются две строки символов, имеющих разные
длины, более короткая строка дополняется справа пробелами для того, чтобы они
имели одинаковую длину перед осуществлением сравнения. GROUP BY [таблица.]столбец [,[таблица.]столбец] …
[HAVING фраза] GROUP BY инициирует перекомпоновку формируемой таблицы по
группам, каждая из которых имеет одинаковое значение в столб-цах, включенных в
перечень GROUP BY. Далее к этим группам применяются агрегирующие функции,
указанные во фразе SELECT, что приводит к замене всех значений группы на
единственное значение (сумма, количество и т.п.). HAVING
[NOT] HAVING_условие [[ANDOR][NOT] HAVING_условие]… значение { =
<> < <= > >= } { значение ( подзапрос ) {значение_2 SQL_функция_2}
AND {значение_3 SQL_функция_3} {значение SQL_функция} IS
[NOT] NULL 2.2.2. Запросы с использованием
единственной таблицы. SELECT
Название, Статус, Адрес При необходимости получения полной информации о поставщиках, можно
было бы дать запрос или использовать его более короткую нотацию:
Здесь "звездочка" (*) служит кратким
обозначением всех имен полей в таблице, указанной во фразе FROM. При этом
порядок вывода полей соответствует порядку, в котором эти поля определялись при
создании таблицы. дает результат, показанный на рис. 2.2,б. б) Статус Крупа Кооператив Рыба ОГУРЕЧИК Исключение
дубликатов В предыдущем примере был выдан правильный, но не совсем удачный
перечень основных продуктов: из него не были исключены дубликаты. Для исключения
дубликатов и одновременного упорядочения перечня необходимо дополнить запрос
ключевым словом DISTINCT (различный, различные), как показано в следующем
примере: Из синтаксиса фразы SELECT видно, что в
ней может содержаться не только перечень столбцов таблицы или символ *, но и
выражения. Например, если нужно получить значение калорийности всех продуктов,
то можно учесть, что при окислении 1 г углеводов или белков в организме
освобождается в среднем 4.1 ккал, а при окислении 1 г жиров – 9.3 ккал, и выдать
запрос: результат которого приведен на рис. 2.3,а. Судак Масло Кофе Майонез Фраза SELECT
может включать не только выражения, но и отдельные числовые или текстовые
константы. Следует отметить, что текстовые константы должны заключаться в
апострофы ('). На рис. 2.3,б приведен результат запроса: А что
произойдет, если какой-либо член выражения не определен, т.е. имеет значение
NULL и каким образом появилось такое значение? Если при загрузке строк таблицы
в какой-либо из вводимых строк отсутствует значение для какого-либо столбца, то
СУБД введет в такое поле NULL-значение. NULL-значение "придумано" для того,
чтобы представить единым образом "неизвестные значения" для любых типов данных.
Действительно, так как при вводе данных в столбец или их изменении СУБД
запрещает ввод значений не соответствующих описанию данных этого столбца, то,
например, нельзя использовать пробел для отсутствующего значения числа. Нельзя
для этих целей использовать и ноль: нет месяца или дня недели равного нулю, да и
для чисел ноль не может рассматриваться как неизвестное значение в одном месте и
как известное – в другом. При выводе же NULL-значения на экран или печатающее
устройство его код воспроизводится каким-либо специально заданным символом или
набором символов: например, пробелом (если его нельзя перепутать с текстовым
значением пробела) или сочетанием –0-. С помощью специальной команды можно
установить в СУБД один из режимов представления NULL-значений при выполнении
числовых расчетов: запрет или разрешение замены NULL-значения нулем. В первом
случае любое арифметическое выражение, содержащее неопределенный операнд, будет
также иметь неопределенное значение. Во втором случае результат вычислений будет
иметь численное значение (если это значение попадает в диапазон представления
соответствующего типа данных). и разных "настройках" СУБД
могут быть получены разные результаты: -0- 170 С помощью BETWEEN … AND … (находится в интервале от … до …) можно
отобрать строки, в которых значение какого-либо столбца находятся в заданном
диапазоне. Например, выдать перечень продуктов, в которых значение содержания
белка находится в диапазоне от 10 до 50: Можно задать и
NOT BETWEEN (не принадлежит диапазону между), например: WHERE Белки NOT BETWEEN 10 AND 50 189. 825. BETWEEN
особенно удобен при работе с данными, задаваемыми интервалами, начало и конец
которых расположен в разных столбцах. Для примера воспользуемся таблицей
"минимальных окладов" (табл. 2.4), величина которых непосредственно связана со
студенческой стипендией. В этой таблице для текущего значения минимального
оклада установлена запредельная дата окончания 9 сентября 9999 года. Начало Если, например, потребовалось
узнать, какие изменения минимальных окладов производились в 1993/94 учебном
году, то можно выдать запрос и получить
результат: Миноклад 01-07-1994 Отметим, что при формировании
запросов значения дат следует заключать в апострофы, чтобы СУБД не путала их с
выражениями и не пыталась вычитать из 31 значение 8, а затем 1994. Для
выявления всех значений минимальных окладов, которые существовали в 1993/94
учебном году, можно сформировать запрос OR Конец BETWEEN '1-9-1993' AND '31-
8-1994' Наконец, для получения минимального оклада на 15-
5-1994: WHERE '15-05-
1994' BETWEEN Начало AND Конец WHERE Основа IN (Яйца Крупа Овощи); Блюдо Овощи Салат
витаминный Морковь с рисом 3 Яйца Каша рисовая Крупа Помидоры с луком Рассмотренная форма IN является в
действительности просто краткой записью последовательности отдельных сравнений,
соединенных операторами OR. Предыдущее предложение эквивалентно
такому: Выдать перечень салатов Результат: Обычная форма "имя_столбца LIKE
текстовая_константа" для столбца текстового типа позволяет отыскать все значения
указанного столбца, соответствующие образцу, заданному "текстовой_константой".
Символы этой константы интерпретируются следующим образом: символ % (процент) –
заменяет любую последовательность из N символов (где N может быть нулем), Следовательно, в приведенном примере
SELECT будет осуществлять выборку записей из таблицы Блюда, для которых значение
в столбце Блюдо начинается сочетанием 'Салат' и содержит любую
последовательность из нуля или более символов, следующих за сочетанием 'Салат'.
Если бы среди блюд были "Луковый салат", "Фруктовый салат" и т.п., то они не
были бы найдены. Для их отыскания надо изменить фразу WHERE: или при отсутствии различий между малыми и большими буквами
(такую настройку допускают некоторые СУБД): Вовлечение неопределенного значения
(NULL-значения) Если при загрузке данных не введено значение в какое-либо поле
таблицы, то СУБД поместит в него NULL-значение. Аналогичное значение можно
ввести в поле таблицы, выполняя операцию изменения данных. Так, при отсутствии
сведений о наличии у поставщиков судака и моркови в столбцы Цена и К_во
соответствующих строк таблицы Поставки вводится NULL и там будет храниться код
NULL-значения, а не 0, 0. Или пробел. (Отметим, что в распечатке таблицы
Поставки в этих местах расположен пробел, установленный в СУБД для представления
NULL-значения при выводе на печать). В этом случае для выявления названий
продуктов, отсутствующих в кладовой, шеф-повар может дать запрос WHERE К_во IS
NULL; Естественно, что для
выявления продуктов, существующих в кладовой, следует дать запрос столбец IS NULL и столбец IS NOT NULL связано с тем, что ничто – и
даже само NULL-значение – не считается равным другому NULL-значению. (Несмотря
на это, два неопределенных значения рассматриваются, однако, как дубликаты друг
друга при исключении дубликатов, и предложение SELECT DISTINCT даст в результате
не более одного NULL-значения.) Простейший
вариант этой фразы – упорядочение строк результата по значению одного из
столбцов с указанием порядка сортировки или без такого указания. (По умолчанию
строки будут сортироваться в порядке возрастания значений в указанном
столбце.) Например, выдать перечень продуктов и содержание в них основных
веществ в порядке убывания содержания белка SELECT Продукт,
Белки, Жиры, Углев 190. 189. 167. 127. 127. 106. При включении в список ORDER
BY нескольких столбцов СУБД сортирует строки результата по значениям первого
столбца списка пока не появится несколько строк с одинаковыми значениями данных
в этом столбце. Такие строки сортируются по значениям следующего столбца из
списка ORDER BY и т.д. Крупа Яблоки печеные 3 Молоко Мясо Кроме того, в список ORDER BY можно включать не только
имя столбца, а его порядковую позицию в перечне SELECT. Благодаря этому возможно
упорядочение результатов на основе вычисляемых столбцов, не имеющих
имен. позволит получить
список продуктов, показанный на рис.2.3,в – переупорядоченный по возрастанию
значений калорийности список рис.2.3,а. В SQL существует ряд специальных стандартных функций (SQL-функций).
Кроме специального случая COUNT(*) каждая из этих функций оперирует
совокупностью значений столбца некоторой таблицы и создает единственное
значение, определяемое так: самое большое
значение в столбце, Для функций SUM
и AVG рассматриваемый столбец должен содержать числовые значения. Следует
отметить, что здесь столбец – это столбец виртуальной таблицы, в которой могут
содержаться данные не только из столбца базовой таблицы, но и данные, полученные
путем функционального преобразования и (или) связывания символами арифметических
операций значений из одного или нескольких столбцов. При этом выражение,
определяющее столбец такой таблицы, может быть сколь угодно сложным, но не
должно содержать SQL-функций (вложенность SQL-функций не допускается). Однако из
SQL-функций можно составлять любые выражения. Аргументу всех функций, кроме
COUNT(*), может предшествовать ключевое слово DISTINCT (различный), указывающее,
что избыточные дублирующие значения должны быть исключены перед тем, как будет
применяться функция. Специальная же функция COUNT(*) служит для подсчета всех
без исключения строк в таблице (включая дубликаты). Если не используется фраза GROUP BY, то в
перечень элементов_SELECT можно включать лишь SQL-функции или выражения,
содержащие такие функции. Другими словами, нельзя иметь в списке столбцы, не
являющихся аргументами SQL-функций. Результат: COUNT(К_во) Если бы для вывода в результат еще
и номера продукта был сформирован запрос то было бы получено
сообщение об ошибке. Это связано с тем, что SQL-функция создает единственное
значение из множества значений столбца-аргумента, а для "свободного" столбца
должно быть выдано все множество его значений. Без специального указания (оно
задается фразой GROUP BY) SQL не будет выяснять, одинаковы значения этого
множества (как в данном примере, где ПР=10) или различны (как было бы при
отсутствии WHERE фразы). Поэтому подобный запрос отвергается системой. WHERE ПР = 10; Отметим также, что в столбце-
аргументе перед применением любой функции, кроме COUNT(*), исключаются все
неопределенные значения. Если оказывается, что аргумент – пустое множество,
функция COUNT принимает значение 0, а остальные – NULL. Например, для
получения суммы цен, средней цены, количества поставляемых продуктов и
количества разных цен продуктов, проданных коопторгом УРОЖАЙ (ПС=5), а также для
получения количества продуктов, которые могут поставляться этим коопторгом,
можно дать запрос и получить В другом примере, где надо узнать "Сколько
поставлено моркови и сколько поставщиков ее поставляют?": Наконец, попробуем получить сумму
массы поставленного лука с его средней ценой ("Сапоги с яичницей"): Мы показали, как можно вычислить массу определенного продукта,
поставляемого поставщиками. Предположим, что теперь требуется вычислить общую
массу каждого из продуктов, поставляемых в настоящее время поставщиками. Это
можно легко сделать с помощью предложения 9 11 12 15 1 3 5 6 8 7 2 4 13 14 16 17 10 Фраза
GROUP BY (группировать по) инициирует перекомпоновку указанной во FROM таблицы
по группам, каждая из которых имеет одинаковые значения в столбце, указанном в
GROUP BY. В рассматриваемом примере строки таблицы Поставки группируются так,
что в одной группе содержатся все строки для продукта с ПР = 1, в другой – для
продукта с ПР = 2 и т.д. (см. рис. 2.5,б). Далее к каждой группе применяется
фраза SELECT. Каждое выражение в этой фразе должно принимать единственное
значение для группы, т.е. оно может быть либо значением столбца, указанного в
GROUP BY, либо арифметическим выражением, включающим это значение, либо
константой, либо одной из SQL-функций, которая оперирует всеми значениями
столбца в группе и сводит эти значения к единственному значению (например, к
сумме). Отметим, что фраза GROUP BY не предполагает ORDER BY. Чтобы
гарантировать упорядочение по ПР результата рассматриваемого примера (рис.
2.5,в) следует дать запрос Наконец, отметим, что строки таблицы можно группировать по
любой комбинации ее столбцов. Так, по запросу можно узнать коды и количество порций
блюд, заказанных отдыхающими пансионата (32 человека) на каждую из трапез
следующего дня: 1 Использование фразы
HAVING Фраза HAVING (рис. 2.3) играет такую же роль для групп, что и фраза
WHERE для строк: она используется для исключения групп, точно так же, как WHERE
используется для исключения строк. Эта фраза включается в предложение лишь при
наличии фразы GROUP BY, а выражение в HAVING должно принимать единственное
значение для группы. HAVING COUNT(*) 2; О средствах одновременной работы
с множеством таблиц Затрагивая вопросы проектирования баз данных, мы выяснили,
что базы данных – это множество взаимосвязанных сущностей или отношений (таблиц)
в терминологии реляционных СУБД. При проектировании стремятся создавать таблицы,
в каждой из которых содержалась бы информация об одном и только об одном типе
сущностей. Это облегчает модификацию базы данных и поддержание ее целостности.
Но такой подход тяжело усваивается начинающими проектантами, которые пытаются
привязать проект к будущим приложениям и так организовать таблицы, чтобы в
каждой из них хранилось все необходимое для реализации возможных запросов.
Типичен вопрос: как же получить сведения о том, где купить продукты для
приготовления того или иного блюда и определить его калорийность и стоимость,
если нужные данные "рассыпаны" по семи различным таблицам? Не лучше ли иметь
одну большую таблицу, содержащую все сведения базы данных ПАНСИОН ? Даже при
отсутствии средств одновременного доступа ко многим таблицам нежелателен проект,
в котором информация о многих типах сущностей перемешана в одной таблице. SQL же
обладает великолепным механизмом для одновременной или последовательной
обработки данных из нескольких взаимосвязанных таблиц. В нем реализованы
возможности "соединять" или "объединять" несколько таблиц и так называемые
"вложенные подзапросы". Например, чтобы получить перечень поставщиков продуктов,
необходимых для приготовления Сырников, возможен запрос WHERE Продукты.ПР = Состав.ПР AND Цена IS NOT NULL; 1.8 Ферма ТУЛЬСКИЙ Он получен следующим образом: СУБД
последовательно формирует строки декартова произведения таблиц, перечисленных во
фразе FROM, проверяет, удовлетворяют ли данные сформированной строки условиям
фразы WHERE, и если удовлетворяют, то включает в ответ на запрос те ее поля,
которые перечислены во фразе SELECT. Следует подчеркнуть, что в SELECT и WHERE
(во избежание двусмысленности) ссылки на все (*) или отдельные столбцы могут (а
иногда и должны) уточняться именем соответствующей таблицы, например,
Поставки.ПС, Поставщики.ПС, Меню.*, Состав.БЛ, Блюда.* и т.п. Очевидно, что с
помощью соединения несложно сформировать запрос на обработку данных из
нескольких таблиц. Кроме того, в такой запрос можно включить любые части
предложения SELECT, рассмотренные в главе 2 (выражения с использованием функций,
группирование с отбором указанных групп и упорядочением полученного результата).
Следовательно, соединения позволяют обрабатывать множество взаимосвязанных
таблиц как единую таблицу, в которой перемешана информация о многих типах
сущностей. Поэтому начинающий проектант базы данных может спокойно создавать
маленькие нормализованные таблицы, так как он всегда может получить из них любую
"большую" таблицу. Кроме механизма соединений в SQL есть механизм вложенных
подзапросов, позволяющий объединить несколько простых запросов в едином
предложении SELECT. Иными словами, вложенный подзапрос – это уже знакомый нам
подзапрос (с небольшими огра-ничениями), который вложен в WHERE фразу другого
вложенного подзапроса или WHERE фразу основного запроса. Для иллюстрации
вложенного подзапроса вернемся к предыдущему примеру и попробуем получить
перечень тех поставщиков продуктов для Сырников, которые поставляют нужные
продукты за минимальную цену. WHERE Продукты.ПР =
Состав.ПР AND Цена = ( SELECT
MIN(Цена) Название ПОРТОС Творог Здесь с помощью подзапроса,
размещенного в трех последних строках запроса, описывается процесс определения
минимальной цены каждого продукта для Сырников и поиск поставщика, предлагающего
этот продукт за такую цену. Так как декартово произведение n таблиц – это таблица,
содержащая все возможные строки r, такие, что r является сцеплением какой-либо
строки из первой таблицы, строки из второй таблицы, … и строки из n-й таблицы (а
мы уже научились выделять с помощью SELECT любое подмножество реляционной
таблицы), то осталось лишь выяснить, можно ли с помощью SELECT получить
декартово произведение. Для получения декартова произведения нескольких таблиц
надо указать во фразе FROM перечень перемножаемых таблиц, а во фразе SELECT –
все их столбцы. FROM Вид_блюд,
Трапезы; В З С Г SELECT Меню.*, Трапезы.*,
Вид_блюд.*, Блюда.* Если из декартова произведения
убрать ненужные строки и столбцы, то можно получить актуальные таблицы,
соответствующие любому из соединений. Завтрак Салат
летний Завтрак Салат
мясной Завтрак Салат
витаминный 4 Завтрак Суп молочный Завтрак З 1 Завтрак Салат
витаминный Завтрак Салат
рыбный Завтрак Паштет из
рыбы Завтрак Мясо с
гарниром 3 Очевидно, что отбор актуальных
строк обеспечивается вводом в запрос WHERE фразы, в которой устанавливается
соответствие между: кодами видов блюд (В) в таблицах Меню и Вид_блюд (Меню.В =
Вид_блюд.В), SELECT Меню.*, Трапезы.*,
Вид_блюд.*, Блюда.* позволит
получить эквисоединение таблиц Меню, Трапезы, Вид_блюд и Блюда: Завтрак Мясо с
гарниром Завтрак Омлет с
луком Яйца Фрукты Естественное
соединение таблиц Легко заметить, что в эквисоединение таблиц вошли дубликаты
столбцов, по которым проводилось соединение (Т, В и БЛ). Для исключения этих
дубликатов можно создать естественное соединение тех же таблиц: WHERE Меню.Т = Трапезы.Т 1 4 250. Яйца Фрукты Молочный напиток Для исключения всех столбцов, по которым проводится соединение таблиц,
надо создать композицию WHERE Меню.Т = Трапезы.Т Салат витаминный ACE="Times New
Roman"> Ужин 180. Напиток Ужин В базе
данных ПАНСИОН трудно подобрать несложный пример, иллюстрирующий тета-соединение
таблиц. Поэтому сконструируем такой надуманный запрос: позволяющий выбрать из
полученного декартова произведения таблиц Вид_блюд и Трапезы лишь те строки, в
которых значение трапезы "меньше" (по алфавиту) значения вида блюда: Закуска Суп Суп Напиток Соединение
таблиц с дополнительным условием При формировании соединения создается рабочая
таблица, к которой применимы все операции: отбор нужных строк соединения (WHERE
фраза), упорядочение получаемого результата (ORDER BY фраза) и агрегатирование
данных (SQL-функции и GROUP BY фраза). Например, для получения перечня блюд,
предлагаемых в меню на завтрак, можно сформировать запрос на основе
композиции: WHERE Меню.Т = Трапезы.Т Закуска 200. 100. Соединение таблицы со своей
копией В ряде приложений возникает необходимость одновременной обработки
данных какой-либо таблицы и одной или нескольких ее копий, создаваемых на время
выполнения запроса. Например, при создании списков студентов (таблица
Студенты) возможен повторный ввод данных о каком-либо студенте с присвоением ему
второго номера зачетной книжки. Для выявления таких ошибок можно соединить
таблицу Студенты с ее временной копией, установив в WHERE фразе равенство
значений всех одноименных столбцов этих таблиц кроме столбцов с номером зачетной
книжки (для последних надо установить условие неравенства значений). Временную
копию таблицы можно сформировать, указав имя псевдонима за именем таблицы во
фразе FROM. Так, с помощью фразы В качестве примера
соединения таблицы с ней самой сформируем запрос на вывод таких пар блюд таблицы
Блюда, в которых совпадает основа, а название первого блюда пары меньше (по
алфавиту), чем номер второго блюда пары. Для этого можно создать запрос с одной
копией таблицы Блюда (Копия): AND Блюдо <
Копия.Блюдо; FROM Блюда Первая, Блюда Вторая Вторая.Блюдо Помидоры с луком Вложенный подзапрос – это подзапрос, заключенный в
круглые скобки и вложенный в WHERE (HAVING) фразу предложения SELECT или других
предложений, использующих WHERE фразу. Вложенный подзапрос может содержать в
своей WHERE (HAVING) фразе другой вложенный подзапрос и т.д. Нетрудно
догадаться, что вложенный подзапрос создан для того, чтобы при отборе строк
таблицы, сформированной основным запросом, можно было использовать данные из
других таблиц (например, при отборе блюд для меню использовать данные о наличии
продуктов в кладовой пансионата). Существуют простые и коррелированные
вложенные подзапросы. Они включаются в WHERE (HAVING) фразу с помощью условий
IN, EXISTS или одного из условий сравнения ( = < < <= = ). Простые
вложенные подзапросы обрабатываютя системой "снизу вверх". Первым обрабатывается
вложенный подзапрос самого нижнего уровня. Множество значений, полученное в
результате его выполнения, используется при реализации подзапроса более высокого
уровня и т.д. Запросы с коррелированными вложенными подзапросами
обрабатываются системой в обратном порядке. Сначала выбирается первая строка
рабочей таблицы, сформированной основным запросом, и из нее выбираются значения
тех столбцов, которые используются во вложенном подзапросе (вложенных
подзапросах). Если эти значения удовлетворяют условиям вложенного подзапроса, то
выбранная строка включается в результат. Затем выбирается вторая строка и т.д.,
пока в результат не будут включены все строки, удовлетворяющие вложенному
подзапросу (последовательности вложенных подзапросов). Следует отметить, что
SQL обладает большой избыточностью в том смысле, что он часто предоставляет
несколько различных способов формулировки одного и того же запроса. Поэтому во
многих примерах данной главы будут использованы уже знакомые нам по предыдущей
главе концептуальные формулировки запросов. И несмотря на то, что часть из них
успешнее реализуется с помощью соединений, здесь все же будут приведены их
варианты с использованием вложенных подзапросов. Это связано с необходимостью
детального знакомства с созданием и принципом выполнения вложенных подзапросов,
так как существует немало задач (особенно на удаление и изменение данных),
которые не могут быть реализованы другим способом. Кроме того, разные
формулировки одного и того же запроса требуют для своего выполнения различных
ресурсов памяти и могут значительно отличаться по времени реализации в разных
СУБД. Простые вложенные подзапросы
используются для представления множества значений, исследование которых должно
осуществляться в каком-либо предикате IN, что иллюстрируется в следующем
примере: выдать название и статус поставщиков продукта с номером 11, т.е.
помидоров. FROM Поставки СЫТНЫЙ При
обработке полного запроса система выполняет прежде всего вложенный подзапрос.
Этот подзапрос выдает множество номеров поставщиков, которые поставляют продукт
с кодом ПР = 11, а именно множество (1, 5, 6, 8). Поэтому первоначальный запрос
эквивалентен такому простому запросу: Подзапрос с несколькими уровнями
вложенности можно проиллюстрировать на том же примере. Пусть требуется узнать не
поставщиков продукта 11, как это делалось в предыдущих запросах, а поставщиков
помидоров, являющихся продуктом с номером 11. Для этого можно дать
запрос ( SELECT ПР В данном случае результатом самого внутреннего
подзапроса является только одно значение (11). Как уже было показано выше,
подзапрос следующего уровня в свою очередь дает в результате множество (1, 5, 6,
8). Последний, самый внешний SELECT, вычисляет приведенный выше окончательный
результат. Вообще допускается любая глубина вложенности подзапросов. FROM Поставщики, Поставки, Продукты При
выполнении этого компактного запроса система должна одновременно обрабатывать
данные из трех таблиц, тогда как в предыдущем примере эти таблицы обрабатываются
поочередно. Естественно, что для их реализации тебуются различные ресурсы памяти
и времени, однако этого невозможно ощутить при работе с ограниченным объемом
данных в иллюстративной базе ПАНСИОН. Выдать номера поставщиков, которые
поставляют хотя бы один продукт, поставляемый поставщиком 6. FROM Поставки Отметим, что ссылка на Поставки во
вложенном подзапросе означает не то же самое, что ссылка на Поставки во внешнем
запросе. В действительности, два имени Поставки обозначают различные значения.
Чтобы этот факт стал явным, полезно использовать псевдонимы, например, X и
Y: Здесь X и Y – произвольные
псевдонимы таблицы Поставки, определяемые во фразе FROM и используемые как явные
уточнители во фразах SELECT и WHERE. Напомним, что псевдонимы определены лишь в
пределах одного запроса. Выдать номера поставщиков, находящихся в том же
городе, что и поставщик с номером 6.
WHERE ПС = 6 ); В подобных запросах можно использовать и
другие операторы сравнения (<, <=, <, = или ), однако, если вложенный
подзапрос возвращает более одного значения и не используется оператор IN, будет
возникать ошибка. SELECT Название,
Статус Такой подзапрос отличается от обычного тем, что
вложенный подзапрос не может быть обработан прежде, чем будет обрабатываться
внешний подзапрос. Это связано с тем, что вложенный подзапрос зависит от
значения Поставщики.ПС а оно изменяется по мере того, как система проверяет
различные строки таблицы Поставщики. Следовательно, с концептуальной точки
зрения обработка осуществляется следующим образом: Система проверяет первую
строку таблицы Поставщики. Предположим, что это строка поставщика с номером 1.
Тогда значение Поставщики.ПС будет в данный момент имеет значение, равное 1, и
система обрабатывает внутренний запрос получая в результате множество (9, 11, 12, 15). Теперь система
может завершить обработку для поставщика с номером 1. Выборка значений Название
и Статус для ПС=1 (СЫТНЫЙ и рынок) будет проведена тогда и только тогда, когда
ПР=11 будет принадлежать этому множеству, что, очевидно, справедливо. Далее
система будет повторять обработку такого рода для следующего поставщика и т.д.
до тех пор, пока не будут рассмотрены все строки таблицы Поставщики. Подобные
подзапросы называются коррелированными, так как их результат зависит от
значений, определенных во внешнем подзапросе. Обработка коррелированного
подзапроса, следовательно, должна повторяться для каждого значения извлекаемого
из внешнего подзапроса, а не выполняться раз и навсегда. Рассмотрим пример
использования одной и той же таблицы во внешнем подзапросе и коррелированном
вложенном подзапросе. SELECT DISTINCT X.ПР Действие этого запроса можно пояснить
следующим образом: "Поочередно для каждой строки таблицы Поставки, скажем X,
выделить значение номера продукта (ПР), если и только если это значение не
входит в некоторую строку, скажем, Y, той же таблицы, а значение столбца номер
поставщика (ПС) в строке Y не равно его значению в строке X". Запросы, использующие EXISTS Квантор EXISTS
(существует) – понятие, заимствованное из формальной логики. В языке SQL
предикат с квантором существования представляется выражением EXISTS (SELECT *
FROM …). Такое выражение считается истинным только тогда, когда результат
вычисления "SELECT * FROM …" является непустым множеством, т.е. когда существует
какая-либо запись в таблице, указанной во фразе FROM подзапроса, которая
удовлетворяет условию WHERE подзапроса. (Практически этот подзапрос всегда будет
коррелированным множеством.) FROM Поставщики Название КОРЮШКА Система последовательно выбирает строки
таблицы Поставщики, выделяет из них значения столбцов Название и ПС, а затем
проверяет, является ли истинным условие существования, т.е. су-ществует ли в
таблице Поставки хотя бы одна строка со значением ПР=11 и значением ПС, равным
значению ПС, выбранному из таблицы Поставщики. Если условие выполняется, то
полученное значение столбца Название включается в результат. Предположим, что
первые значения полей Название и ПС равны, соответственно, 'СЫТНЫЙ' и 1. Так как
в таблице Поставки есть строка с ПР=11 и ПС=1, то значение 'СЫТНЫЙ' должно быть
включено в результат. Хотя этот первый пример только показывает иной способ
формулировки запроса для задачи, решаемой и другими путями (с помощью оператора
IN или соединения), EXISTS представляет собой одну из наиболее важных
возможностей SQL. Фактически любой запрос, который выражается через IN, может
быть альтернативным образом сформулирован также с помощью EXISTS. Однако
обратное высказывание несправедливо. SELECT Название, Статус AND ПР
= 11 ); Теперь, после знакомства с различными формулировками вложенных
подзапросов и псевдонимами легче понять текст и алгоритм реализации запроса на
получение тех поставщиков продуктов для Сырников, которые поставляют эти
продукты за минимальную цену: WHERE Продукты.ПР =
Состав.ПР AND Цена = ( SELECT
MIN(Цена) Естественно, что
это коррелированный подзапрос: здесь сначала определяется минимальная цена
продукта, входящего в состав Сырников, и только затем выясняется его
поставщик. На этом примере мы закончим знакомство с вложенными подзапросами,
предложив попробовать свои силы в составлении ряда запросов, с помощью механизма
таких подзапросов: Выдать блюда, продукты для которых
поставляются агрофирмой ЛЕТО.
| |