GeoSELECT.ru



Программирование / Реферат: Структурное программирование: предпосылки и назначение; основные критерии оценки качества программы для ЭВМ (Программирование)

Космонавтика
Уфология
Авиация
Административное право
Арбитражный процесс
Архитектура
Астрология
Астрономия
Аудит
Банковское дело
Безопасность жизнедеятельности
Биология
Биржевое дело
Ботаника
Бухгалтерский учет
Валютные отношения
Ветеринария
Военная кафедра
География
Геодезия
Геология
Геополитика
Государство и право
Гражданское право и процесс
Делопроизводство
Деньги и кредит
Естествознание
Журналистика
Зоология
Инвестиции
Иностранные языки
Информатика
Искусство и культура
Исторические личности
История
Кибернетика
Коммуникации и связь
Компьютеры
Косметология
Криминалистика
Криминология
Криптология
Кулинария
Культурология
Литература
Литература : зарубежная
Литература : русская
Логика
Логистика
Маркетинг
Масс-медиа и реклама
Математика
Международное публичное право
Международное частное право
Международные отношения
Менеджмент
Металлургия
Мифология
Москвоведение
Музыка
Муниципальное право
Налоги
Начертательная геометрия
Оккультизм
Педагогика
Полиграфия
Политология
Право
Предпринимательство
Программирование
Психология
Радиоэлектроника
Религия
Риторика
Сельское хозяйство
Социология
Спорт
Статистика
Страхование
Строительство
Схемотехника
Таможенная система
Теория государства и права
Теория организации
Теплотехника
Технология
Товароведение
Транспорт
Трудовое право
Туризм
Уголовное право и процесс
Управление
Физика
Физкультура
Философия
Финансы
Фотография
Химия
Хозяйственное право
Цифровые устройства
Экологическое право
   

Реферат: Структурное программирование: предпосылки и назначение; основные критерии оценки качества программы для ЭВМ (Программирование)



МИНИСТЕРСТВО ОБЩЕГО И ПРОФЕССИОНАЛЬНОГО ОБРАЗОВАНИЯ РФ
АСТРАХАНСКИЙ ГОСУДАРСТВЕННЫЙ ПЕДАГОГИЧЕСКИЙ УНИВЕРСИТЕТ



Структурное программирование: предпосылки и назначение структурного
программирования, основные критерии оценки качества программы для ЭВМ.
Реализация структурного программирования на языке «Е-практикума».


Выполнил:
студент V курса дополнительной
специальности "информатика"
Безниско Евгений.



Астрахань - 1999


Предпосылки и назначение структурного программирования.
Традиционная технология программирования формировалась на заре
вычислительной техники, когда в распоряжении пользователей были
ограниченные ресурсы ЭВМ, а разработчик программ был в то же время и
главным ее пользователем. В этих условиях главное внимание обращалось на
получение эффективных программ в смысле оптимального использования ресурсов
ЭВМ.
В настоящее время, когда сфера применения ЭВМ чрезвычайно расширилась,
разработка и эксплуатация программ осуществляется, как правило, разными
людьми. Поэтому наряду с эффективностью на первый план выдвигаются и другие
важные характеристики программ такие, как понятность, хорошая
документированность, надежность, гибкость, удобство сопровождения и т.п.
Проблема разработки программ, обладающих такими качествами, объясняется
трудоемкостью процесса программирования и связанным с этим быстрым ростом
стоимости программного обеспечения.
Для создания "хорошей" программы появляется необходимость
придерживаться определенных принципов или определенной дисциплины
программирования. Значительный прогресс в области программирования
достигается с использованием так называемого структурного программирования.
Появление новой технологии, или, как еще говорят, дисциплины
программирования, основанной на структурном подходе, связано с именем
известного голландского ученого Э.Дейкстры (1965 г.). В своих работах он
высказал предположение, что оператор GOTO может быть исключен из языков
программирования и что квалификация программиста обратно пропорциональна
числу операторов GOTO в его программах. Такая дисциплина программирования
упрощает и структуризирует программу.
Однако представление о структурном программировании, как о
программировании без использования оператора GOTO, является ошибочным.
Например, Хоор определяет структурное программирование как "систематическое
использование абстракции для управления массой деталей и способ
документирования, который помогает проектировать программу".
Структурное программирование можно толковать как "проектирование,
написание и тестирование программы в соответствии с заранее определенной
дисциплиной".
Структурный подход к программированию как раз и имеет целью снижение
трудоемкости всего процесса создания программного обеспечения от
технического задания на разработку до завершения эксплуатации. Он означает
необходимость единой дисциплины на всех стадиях разработки программы. В
понятие структурного подхода к программированию обычно включают нисходящие
методы разработки программ (принцип «сверху вниз»), собственно структурное
программирование и так называемый сквозной структурный контроль.
Основной целью структурного программирования является уменьшение
трудностей тестирования и доказательства правильности программы. Это
особенно важно при разработке больших программных систем. Опыт применения
методов структурного программирования при разработке ряда сложных
операционных систем показывает, что правильность логической структуры
системы поддается доказательству, а сама программа допускает достаточно
полное тестирование. В результате в готовой программе встречаются только
тривиальные ошибки кодирования, которые легко исправляются.
Структурное программирование улучшает ясность и читабельность программ.
Программы, которые написаны с использованием традиционных методов, особенно
те, которые перегружены операторами GOTO, имеют хаотичную структуру.
Структурированные программы имеют последовательную организацию, поэтому
возможно читать такую программу сверху донизу без перерыва.
Наконец, структурное программирование призвано улучшить эффективность
программ.
Итак, структурное программирование представляет собой некоторые
принципы написания программ в соответствии со строгой дисциплиной и имеет
целью облегчить процесс тестирования, повысить производительность труда
программистов, улучшить ясность и читабельность программы, а также повысить
ее эффективность.

Основные критерии оценки качества программы для ЭВМ.
Известно, что один и тот же алгоритм может быть реализован на ЭВМ
различными способами, т.е. может быть составлено несколько различных
программ, решающих одну и ту же задачу.
Таким образом, нужно иметь некоторые критерии оценки программы, с
помощью которых можно судить насколько одна программа лучше другой. Анализ
и оценка программы носят преимущественно качественный характер.
1. Программа работает и решает поставленную задачу. Понятно, что эта
характеристика программы является самой важной.
В связи с этим каждая программа должна быть устроена так, чтобы можно
было проверить правильность полученных результатов. Такая проверка
проводится в процессе отладки программы, на определенных наборах входных
данных, для которых заранее известен ответ. Но отладка может доказать лишь
наличие ошибок в программе, но не может доказать правильности программы для
всех возможных вычислений, реализуемых с ее помощью. В связи с этим
необходима разработка методов аналитической верификации программы.
Для аналитического доказательства правильности программы требуется,
чтобы программа легко анализировалась. Это означает, что программа должна
быть устроена так, чтобы можно было понять, каким образом с ее помощью
получается данный ответ.
2. Минимальное время, затрачиваемое на тестирование и отладку
программы. Тестирование и отладка программы – необходимый этап в процессе
решения задачи на ЭВМ. Он занимает от трети до половины всего времени
разработки программы, поэтому очень важно уменьшить время, затрачиваемое на
тестирование и отладку.
Тестирование и отладка программы облегчается, если программа просто
анализируется и снабжена необходимыми комментариями, облегчающими ее
понимание. Хорошие комментарии могут ускорить процесс отладки.
Понимание и отладка программы облегчается, если она имеет простую и
ясную структуру, в частности, если ограничено использование операторов
передачи управления (GOTO). Перегруженность программы этими операторами
приводит к хаотической структуре и затрудняет отладку.
Еще один важный принцип – использование мнемонических обозначений для
переменных. Языки программирования представляют здесь вполне достаточные
возможности. Для лучшего понимания программы необходимо использовать
мнемонику, отражающую физический (математический, экономический и т.д.)
смысл переменной (например, SPEED - скорость).
3. Уменьшение затрат на сопровождение. Разработанная и отлаженная
программа предназначена для многократного использования, и ее
эксплуатацией, как правило, занимаются не разработчики, а другие
программисты, входящие в так называемую группу сопровождения.
Программистам, сопровождающим программу, часто приходится продолжать
отладку программы и производить ее модернизацию, в связи с изменением
технического задания, введением новых средств программного обеспечения или
выявлением новых ошибок и недоработок в программе.
Для уменьшения затрат на сопровождение необходимо, чтобы каждый
разработчик учитывал сложность сопровождения. Следует разрабатывать,
отлаживать и оформлять программу с учетом того, что ее будут использовать и
сопровождать другие программисты.
4. Гибкость программы. Разработанная программа обычно находится в
эксплуатации длительное время. За это время могут измениться требования к
решаемой задаче, техническое задание, требования к программе. Появляется
необходимость внести определенные изменения в программу, что в некоторых
случаях бывает трудно сделать, т.к. разработчиком не предусмотрена такая
возможность. "Хорошая" программа должна допускать модификацию.
5. Уменьшение затрат на разработку. Программирование является
коллективным трудом. Состав группы программистов, работающих над решением
данной задачи, может по каким-либо причинам измениться. Поэтому
проектирование и разработка программы должны вестись таким образом, чтобы
было возможно при необходимости передать ее завершение другому
программисту. Несоблюдение этого требования часто приводит к срыву сроков
сдачи программ в эксплуатацию.
6. Простота и эффективность. Программа должна быть просто организована.
Это может проявляться и в структуре программы, и в использовании простых и
наиболее естественных средств языка программирования, и в предпочтении
простых структур данных и т.п.
Эффективность программы считается одной из главных ее характеристик.
Поэтому часто в ущерб другим качествам программы разработчики прибегают к
сложным ухищрениям, чтобы уменьшить объем используемой памяти или сократить
время выполнения программы. Во многих случаях затрачиваемые на это усилия
не оправдывают себя. Разумный подход к повышению эффективности программы
состоит в том, чтобы выявить наиболее "узкие" места и постараться их
улучшить.


Реализация структурного программирования на языке

«Е-практикума».
К основным методам структурного программирования относится, прежде
всего, отказ от бессистемного употребления оператора GOTO и
преимущественное использование других структурированных операторов, методы
нисходящего проектирования разработки программы, идеи пошаговой детализации
и некоторые другие соглашения, касающиесся дисциплины программирования.
Всякая программа, в соответствии с структурным подходом к
программированию, может быть построена только с использованием трех
основных типов блоков.
1. Функциональный блок, который на блок-схеме изображается в виде
прямоугольников с одним входом и одним выходом:
Функциональному блоку в языках программирования соответствуют операторы
ввода и вывода или любой оператор присваивания.
В виде функционального блока может быть изображена любая
последовательность операторов, выполняющихся один за другим, имеющая один
вход и один выход.
2. Условная конструкция. Этот блок включает проверку некоторого
логического условия (P), в зависимости от которого выполняется либо один
(S1), либо другой (S2) операторы:
На языке "Е-практикума":
. если
. . то
. . иначе
. все
3. Блок обобщенного цикла. Этот блок обеспечивает многократное
повторение выполнения оператора S пока выполнено логическое условие P:

На языке "Е-практикума" циклы реализуются 2 способами. Цикл с
параметром:
. нц для от <нач.значение> до
. .
. .
. . ...
. кц
Цикл с предусловием:
. нц пока
. .
. .
. . ...
. кц

При конструировании программы с использованием рассмотренных типов
блоков эти блоки образуют линейную цепочку так, что выход одного блока
подсоединяется ко входу следующего. Таким образом, программа имеет линейную
структуру, причем порядок следования блоков соответствует порядку, в
котором они выполняются.
Такая структура значительно облегчает чтение и понимание программы, а
также упрощает доказательство ее правильности. Так как линейная цепочка
блоков может быть сведена к одному блоку, то любая программа может, в
конечном итоге, рассматриваться как единый функциональный блок с один
входом и одним выходом.
При проектировании и написании программы нужно выполнить обратное
преобразование, то есть этот блок разбить на последовательность подблоков,
затем каждый подблок разбить на последовательность более мелких блоков до
тех пор, пока не будут получены "атомарные" блоки, рассмотренных выше
типов. Такой метод конструирования программы принято называть нисходящим
("сверху вниз").
При нисходящем методе конструирования алгоритма и программы
первоначально рассматривается вся задача в целом. На каждом последующем
этапе задача разбивается на более мелкие подзадачи, каждая подзадача, в
конечном итоге на еще более мелкие подзадачи и так до тех пор, пока не
будут получены такие подзадачи, которые легко кодируются на выбранном языке
программирования. При этом на каждом шаге уточняются все новые и новые
детали ("пошаговая детализация").
В процессе нисходящего проектирования сохраняется строгая дисциплина
программирования, то есть разбиение на подзадачи осуществляется путем
применения только рассмотренных типов конструкций (функциональный блок,
условная конструкция, обобщенный цикл), поэтому, в конечном итоге,
получается хорошо структурированная программа.
На языке "Е-практикума" последовательную детализацию можно реализовать
в виде вспомогательного алгоритма (подпрограммы, процедуры, функции).
...
нач
. ...
. вспомогательный_алгоритм(...)
. ...
кон
алг [] вспомогательный_алгоритм(...)
дано ...
надо
нач
. ...
кон

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

S1

S2

[pic]

S

P






Реферат на тему: Структуры Данных и Абстракции Данных

Хотя термины тип данных (или просто тип), структура данных и
абстрактный тип данных звучат похоже, но имеют они различный смысл. В
языках программирования тип данных переменной обозначает множество
значений, которые может принимать эта переменная. Например, переменная
булевого (логического) типа может принимать только два значения:
значение true (истина) и значение false (ложь) и никакие другие. Набор
базовых типов данных отличается в разных языках: в языке Pascal это
типы целых (integer) и действительных (real) чисел, булев (boolean) тип
и символьный (char) тип. Правила конструирования составных типов данных
(на основе базовых типов) также различаются в разных языках
программирования: Pascal легко и быстро строит такие типы.
Абстрактный тип данных (АТД) – это математическая модель плюс
различные операторы, определённые в рамках этой модели. Мы можем
разрабатывать алгоритм в терминах АТД, но для реализации алгоритма в
конкретном языке программирования необходимо найти способ представления
АТД в терминах типов данных и операторов, поддерживаемых данным языком
программирования. Для представления АТД используются структуры данных,
которые представляют собой набор переменных, возможно, различных типов
данных, объединённых определённым образом.
Базовым строительным блоком структуры данных является ячейка,
которая предназначена для хранения значения определённого базового или
составного типа данных. Структуры данных создаются путём задания имён
совокупностям (агрегатам) ячеек и (необязательно) интерпретации
значения некоторых ячеек как представителей (т.е. указателей) других
ячеек.
В качестве простейшего механизма агрегирования ячеек в Pascal и
большинстве других языков программирования можно применять (одномерный)
массив, т.е. последовательность ячеек определённого типа. Массив также
можно рассматривать как отображение множества индексов (таких как целые
числа 1, 2, …, n) в множество ячеек. Ссылка на ячейку обычно состоит из
имени массива и значения из множества индексов данного массива. В
Pascal множество индексов может быть нечислового типа, например (север,
восток, юг, запад), или интервального типа как (1..10).Значения всех
ячеек массива должны иметь одинаковый тип данных. Объявление
Имя: array[ТипИндекса] of ТипЯчеек;
задаёт имя для последовательности ячеек, тип для элементов множества
индексов и тип содержимого ячеек.
Кстати, Pascal необычайно богат на типы индексов. Многие языки
программирования позволяют использовать в качестве индексов только
множества последовательных целых чисел. Например, чтобы в языке Fortran
в качестве индексов массива можно было использовать буквы, надо всё
равно использовать целые индексы, заменяя «А» на 1, «В» на 2, и т.д.
Другим общим механизмом агрегирования ячеек в языке программирования
является структура записи. Запись (record) можно рассматривать как
ячейку, состоящую из нескольких других ячеек (называемых полями),
значения в которых могут быть разных типов. Записи часто группируются в
массивы; тип данных определяется совокупностью типов полей записи.
Например, в Pascal объявление
var
reclist: array[1..4] of record
data: real;
next: integer;
end
задаёт имя reclist (список записей) 4-элементного массива, значениями
которого являются записи с двумя полями: data (данные) и next
(следующий).
Третий метод агрегирования ячеек, который можно найти в Pascal и
некоторых других языках программирования, - это файл. Файл, как и
одномерный массив, является последовательностью значений определённого
типа. Однако файл не имеет индексов: его элементы доступны только в том
порядке, в каком они были записаны в файл. В отличие от файла, массивы
и записи являются структурами с «произвольным доступом», подразумевая
под этим, что время доступа к компонентам массива и записи не зависит
от значения индекса массива или указателя поля записи. Достоинство
агрегирования с помощью файла (частично компенсирующее описанный
недостаток) заключается в том, что файл не имеет ограничения на
количество составляющих его элементов и это количество может изменяться
во время выполнения программы.

Указатели и курсоры

В дополнение к средствам агрегирования ячеек в языках
программирования можно использовать указатели и курсоры. Указатель
(pointer) – это ячейка, чьё значение указывает на другую ячейку. При
графическом представлении структуры данных в виде схемы тот факт, что
ячейка А является указателем на ячейку В, показывается с помощью
стрелки от ячейки А к ячейке В.
В языке Pascal с помощью следующего объявления можно создать
переменную – указатель prt, указывающую на ячейку определённого типа,
например ТипЯчейки:
var
prt: ТипЯчейки

Постфикс в виде стрелки, направленной вверх, в Pascal используется как
оператор разыменования, т.е. выражение prt обозначает значение (типа
ТипЯчейки) в ячейке, указанной prt.


Курсор (cursor) – это ячейка с целочисленным значением, используемая
для указания на массив. В качестве способа указания курсор работает так
же, как и указатель, но курсор можно использовать и в языках (подобных
Fortran), которые не имеют явного типа указателя. Интерпретировав
целочисленную ячейку как индексное значение для массива, можно
эффективно реализовать указания на ячейки массива. К сожалению, этот
приём подходит только для ячеек массива и не позволяет организовать
указания на ячейки, не являющиеся частью массива.


В схемах структур данных будет рисоваться стрелка из ячейки курсора
к ячейке, на которую указывает курсор. Иногда также будет показываться
целое число в ячейке курсора, напоминая тем самым, что это не настоящий
указатель. Можно заметить, что механизм указателя Pascal разрешает
ячейкам массива только «быть указанными» с помощью курсора, но не быть
истинным указателем. Другие языки программирования, подобные PL/1 или
С, позволяют компонентам массивов быть истинными указателями и,
конечно, «быть указанным» с помощью курсора. В отличие от этих языков,
в языках Fortran и Algol, где нет типа указателя, можно использовать
только курсоры.


Пример1. На рис.1 показана структура данных, состоящая из двух
частей. Она имеет цепочку ячеек, содержащих курсоры для массива reclist
(список записей), определённого выше. Назначение поля next (следующий)
заключается в указании на следующую запись в массиве reclist. Например,
reclist[4].next равно 1, поэтому запись 4 предшествует записи 1.
полагая первой запись 4, в соответствии со значениями поля next получим
следующий порядок записей: 4, 1, 3, 2. Отметим, что значение поля next
в записи 2, равное 0, указывает на то, что нет следующей записи.
Целесообразно принять соглашение, что число 0 будет обозначать нуль-
указатель при использовании курсоров и указателей. Но, чтобы не
возникали проблемы при реализации этого соглашения, необходимо также
условиться, что массивы, на которые указывают курсоры, индексируются
начиная с 1, а не с 0.


header



1


2


3


4


data next


Рис.1Пример структуры данных.


Ячейки в цепочке 1 имеют тип


type


recordtype = record


cursor: integer;


prt recordtype


end


На цепочку можно ссылаться с помощью переменной header (заголовок),
имеющей тип recordtype, header указывает на анонимную запись типа
recordtype 1 .Эта запись имеет 4 значения в поле cursor.
Рассматривается 4 как индекс массива reclist. Эта запись имеет истинный
указатель в поле prt на другую анонимную запись, которая, в свою
очередь, указывает на индекс 4 массива reclist и имеет нуль-указатель в
поле prt.


Абстрактный тип данных «Список».


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


В математике список представляет собой последовательность элементов
определённого типа, который в общем случае будет обозначаться как
elementtype (тип элемента). Список будет часто представляться в виде
последовательности элементов, разделённых запятыми: a1, a2, …, an, где
n > 0 и все ai имеют тип elementtype. Количество элементов n будет
называться длиной списка. Если n > 1 ,то а1 называется первым элементом
списка. В случае n = 0 список будет называться пустым, т.е. не
содержащий элементов.


Важное свойство списка заключается в том, что его элементы можно
линейно упорядочить в соответствии с их позицией в списке. Говорится,
что элемент аi предшествует аi+1 для i = 1, 2, …, n-1 и аi следует за
ai-1 для i = 2, 3, …, n. Говорится также, что элемент аi имеет позицию
i. Кроме того, постулируется существование позиции, следующей за
последним элементом списка. Функция END(L) будет возвращать позицию,
следующую за позицией n в n-элементном списке L. Следует отметить, что
позиция END(L), рассматриваемая как расстояние от начала списка, может
изменяться при уменьшении или увеличении списка, в то время как другие
позиции имеют фиксированное (неизменное) расстояние от начала списка.


Для формирования абстрактного типа данных на основе математического
определения списка нужно задать множество операторов, выполняемых над
объектами типа LIST2 (список). Однако не существует одного множества
операторов, выполняемых над списками, удовлетворяющего сразу все
приложения.


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


Теперь перейдём к непосредственному определению множества операторов
списка. Примем обозначения: L – список объектов типа elementtype, x –
объект этого типа, p – позиция элемента в списке. Следует отметить, что
«позиция» имеет другой тип данных, чья реализация может быть различной
для разных реализаций списков. Обычно позиция понимается как множество
целых положительных чисел, но на практике могут встретиться другие
представления.


INSERT(x, p, L). Этот оператор вставляет объект х в позицию р и далее в
следующую, более высокую позицию. Таким образом, если список L состоит
из элементов а1, a2, …,an, то после выполнения этого оператора он будет
иметь вид а1, a2, …, ap-1, x, ap, …, an. Если р принимает
значение END(L), то будем иметь a1, a2, …, an, x. Если в списке
L нет позиции р, то результат выполнения этого оператора не определён.


LOCATE(x, L). Эта функция возвращает позицию объекта х в списке L.если
в списке объект х встречается несколько раз, то возвращается позиция
первого от начала списка объекта х. Если объекта х нет в списке L, то
возвращается END(L).


RETRIEVE(p, L). Эта функция возвращает элемент, который стоит в позиции
р в списке L. Результат не определён, если p = END(L) или в списке L
нет позиции p. Элементы должны быть одного типа, который в принципе
может возвращать функция. Однако на практике мы всегда можем изменить
эту функцию так, что она будет возвращать указатель на объект типа
elementtype.


DELETE(p, L). этот оператор удаляет элемент в позиции p списка L. Так,
если список L состоит из элементов a1, a2, …, an, то после выполнения
этого оператора он будет иметь вид a1, a2, …, ap-1, ap+1, …, an.
Результат не определён, если в списке L нет позиции p или p = END(L).


NEXT(p, L) и PREVIOUS(p, L). Эти функции возвращают соответственно
следующую и предыдущую позиции от позиции p в списке L.если р –
последняя позиция в списке L, то NEXT(p, L) = END(L). Функция NEXT не
определена, когда p = END(L). Функция PREVIOUS не определена, если p =
1. Обе функции не определены, если в списке L нет позиции p.


MAKENULL(L). Эта функция делает список L пустым и возвращает позицию
END(L).


FIRST(L). Эта функция возвращает первую позицию в списке L. Если список
пустой, то возвращается позиция END(L).


8. PRINTLIST(L). Печатает элементы списка L в порядке их расположения.


Стеки.


Стек – это специальный тип списка, в котором все вставки и удаления
выполняются только на одном конце, называемом вершиной (top). Стеки
также иногда называют «магазинами», а в английской литературе для
обозначения стеков ещё используется аббревиатура LIFO (last-in-first-
out – последний вошел – первый вышел). Интуитивными моделями стека
могут служить колода карт на столе при игре в покер, книги, сложенные в
стопку, или стопка тарелок на полке буфета; во всех этих моделях взять
можно только верхний предмет, а добавить новый объект можно, только
положив его на верхний. Абстрактные типы данных семейства STAK (Стек)
обычно используют следующие пять операторов.


MAKENULL(S). Делает стек S пустым.


TOP(S). Возвращает элемент из вершины стека S. Обычно вершина стека
идентифицируется позицией 1, тогда TOP(S) можно записать в терминах
общих операторов списка как RETRIEVE(FIRST(S), S).


POP(S). Удаляет из вершины стека (выталкивает из стека), в терминах
операторов списка этот оператор можно записать как DELETE(FIRST(S),
S). Иногда этот оператор реализуется в виде функции, возвращающей
удаляемый элемент.


PUSH(x, S). Вставляет элемент x в вершину стека S (заталкивает элемент
в стек). Элемент, ранее находившийся в вершине стека, становится
элементом, следующим за вершиной, и т.д. В терминах общих операторов
списка данный оператор можно записать как INSERT(x, FIRST(S), S).


EMPTY(S). Эта функция возвращает значение true (истина), если стек S
пустой, и значение false (ложь) в противном случае.


Очереди.


Другой специальный тип списка – очередь (queue), где элементы
вставляются с одного конца, называемого задним (rear), а удаляются с
другого, переднего (front). Очереди также называют «списками типа FIFO»
(аббревиатура FIFO расшифровывается как first-in-first-out: первым
вошел – первым вышел). Операторы, выполняемые над очередями, аналогичны
операторам стеков. Существенное отличие между ними состоит в том, что
вставка новых элементов осуществляется в конец списка, а не в начало,
как в стеках. Кроме того, различна устоявшаяся терминология для стеков
и очередей. Для работы с очередями будут использоваться следующие
операторы.


MAKENULL(Q) очищает очередь Q, делая её пустой.


FRONT(Q) – функция, возвращающая первый элемент очереди Q. Можно
реализовать эту функцию с помощью операторов списка как
RETRIEVE(FIRST(Q), Q).


ENQUEUE(x, Q) вставляет элемент x в конец очереди Q. С помощью
операторов списка этот оператор можно выполнить следующим образом:
INSERT(x, END(Q), Q).


DEQUEUE(x, Q) удаляет первый элемент очереди Q. Также реализуем с
помощью операторов списка как DELETE(FIRST(Q), Q).


EMPTY(Q) возвращает значение true тогда и только тогда, когда Q
является пустой очередью.


Отображения.


Отображение – это функция, определённая на множестве элементов
(области определения) одного типа (будет обозначаться domaintype – тип
области определения функции) и принимающая значения из множества
элементов (области значений) другого типа, этот тип будет обозначаться
rangetype – тип области значений (конечно, типы domaintype и rangetype
могут совпадать). Тот факт, что отображение М ставит в соответствие
элемент d типа domaintype из области определения элементу r типа
rangetype из области значений, будет записываться как M(d) = r.


Некоторые отображения, подобные square(i) = i2, легко реализовать с
помощью функций и арифметических выражений языка Pascal. Но для многих
отображений нет очевидных способов реализации, кроме хранения для
каждого d значения M(d). Например, для реализации функции, ставящей в
соответствие работникам их недельную зарплату, требуется хранить
текущий заработок каждого работника.


Рассмотрим операторы, которые можно выполнить над отображением М.
Например, по заданному элементу d типа domaintype мы хотим получить
M(d) или узнать, определено ли M(d) (т.е. узнать, принадлежит ли
элемент d области определения М). Или хотим добавить новые элементы в
текущую область определения М и поставить им в соответствие элементы из
области значений. Очевидна также необходимость иметь оператор, который
изменял бы значение M(d). Кроме того, нужно средство «обнуления»
отображения, переводящее любое отображение в пустое отображение, т.е.
такое, у которого область определения пуста. Всё это можно обобщить в
следующие три команды.


MAKENULL(M). Делает отображение М пустым.


ASSIGN(M, d, r). Делает М(d) равным r независимо от того, как M(d)
определено ранее.


COMPUTE(M, d, r). Возвращает значение true и присваивает переменной r
значение M(d), если последнее определено, и возвращает false в
противном случае.


Структуры данных и алгоритмы для внешней памяти.


В алгоритмах, которые обсуждались до сих пор, предполагалось, что
объем входных данных позволяет обходиться исключительно основной
(оперативной) памятью. Но как быть, если нужно, например, отсортировать
всех государственных служащих по продолжительности их рабочего стажа
или хранить информацию из налоговых деклараций всех граждан страны?
Когда возникает необходимость решать подобные задачи, объём
обрабатываемых данных намного превышает возможности основной памяти. В
большинстве компьютерных систем предусмотрены устройства внешней
памяти, такие как жесткие диски, или запоминающие устройства большой
ёмкости, на которых можно хранить огромные объёмы данных. Однако
характеристики доступа к таким устройствам внешней памяти существенно
отличаются от характеристик доступа к основной памяти. Чтобы повысить
эффективность использования этих устройств был разработан ряд структур
данных и алгоритмов.


В Pascal и некоторых других языках предусмотрен файловый тип данных,
предназначенный для представления данных, хранящихся во вторичной
памяти. Даже если в языке, которым вы пользуетесь, файловый тип данных
не предусмотрен, в операционной системе понятие «внешних» файлов,
несомненно, поддерживается. О каких бы файлах ни говорилось (файлах,
предусмотренных в Pascal, или файлах, поддерживаемых непосредственно
операционной системой), в любом случае придётся действовать в рамках
ограничений, касающихся способов доступа к файлам. Операционная система
делит вторичную память на блоки одинакового размера. Размер блока
зависит от конкретного типа операционной систем и обычно находится в
пределах от 521 до 4096 байт.


Файл можно рассматривать как связный список блоков, хотя чаще всего
операционная система использует древовидную организацию блоков, при
которой блоки, составляющие файл, являются листьями дерева, а каждый
внутренний узел содержит указатель на множество блоков файла. Если,
например, 4 байт достаточно, чтобы хранить адрес блока, а длинна блока
составляет 4096 байт, тогда корневой каталог может содержать указатели
максимум на 1024 блока. Таким образом, файлы, состоящие максимум из
1024 блоков (т.е. примерно четырёх миллионов байт), можно представить
одним корневым блоком и блоками, содержащими сам файл. Файлы, состоящие
из максимум 220 блоков, или 232 байт, можно представить одним корневым
блоком, указывающим на 1024 блока промежуточного уровня, каждый из
которых указывает на 1024 блока-листа, содержащих определённую часть
файла и т.д.


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


Теперь нетрудно понять концепцию, которая лежит в основе правил
чтения файлов в языке Pascal. Каждый файл хранится в виде
определённости блоков; каждый такой блок содержит целое число записей.
(Память будет использоваться нерационально, если хранить части одной и
той же записи в разных блоках.) Указатель считываний всегда указывает
на одну из записей в блоке, который в данный момент находится в буфере.
Когда этот указатель должен переместиться на запись, отсутствующую в
буфере, настало время прочитать следующий блок файла.


Аналогично, процесс записи файла в языке Pascal можно рассматривать
как процесс создания файла в буфере. Когда записи «записываются» в
файл, фактически они перемещаются в буфер для этого файла –
непосредственно вслед за записями, которые уже находятся там. Если
очередная запись не помещается в буфер целиком, содержимое буфера
копируется в свободный блок вторичной памяти, который присоединяется к
концу списка блоков для данного файла. После этого можно считать, что
буфер свободен для помещения в него очередной порции записей.


Стоимость операций со внешней памятью.


Природа устройств вторичной памяти (например, дисководов) такова,
что время, необходимое для поиска блока и чтения его в основную память,
достаточно велико, в сравнении со временем, которое требуется для
относительно простой обработки данных, содержащихся в этом блоке.
Допустим, например, что имеется блок из 1000 целых чисел на диске,
вращающемся со скоростью 1000 об/мин. Время, которое требуется для
позиционирования считывающей головки над дорожкой, содержащей этот блок
(так называемое время установки головок), плюс время, затрачиваемое на
ожидание, пока требуемый блок сделает оборот и окажется под головкой
(время ожидания), может в среднем составлять 100 миллисекунд. Процесс
записи блока в определённое место во вторичной памяти занимает примерно
столько же времени. Однако за те же 100 миллисекунд машина, как
правило, успевает выполнить 100 000 команд. Этого времени более чем
достаточно, чтобы выполнить простую обработку тысячи целых чисел, когда
они находятся в основной памяти (например, их суммирование или
нахождение среди них наибольшего числа). Этого времени может даже
хватить для быстрой сортировки целых чисел.


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


Хранение данных в файлах.


В этом разделе будут рассмотрены структуры данных и алгоритмы для
хранения и поиска информации в файлах, находящихся во внешней памяти.
Файл будет рассматриваться как последовательность записей, причём
каждая запись состоит из одной и той же совокупности полей. Поля могут
иметь либо фиксированную длину (заранее определённое количество байт),
либо переменную. Файлы с записями фиксированной длины широко
используются в системах управления базами данных для хранения данных со
сложной структурой. Файлы с записями переменной длины, как правило,
используются для хранения текстовой информации; в языке Pascal такие
файлы не предусмотрены. В этом разделе будем иметь дело с полями
фиксированной длины; рассмотренные методы после определённой
(несложной) модификации могут использоваться для работы с записями
переменной длины.


Будут рассмотрены следующие операторы для работы с файлами.


INSERT вставляет определённую запись в определённый файл.


DELETE удаляет из определённого файла все записи, содержащие указанные
значения в указанных полях.


MODIFY изменяет все записи в определённом файле, задав указанные
значения определённым полям в тех записях, которые содержат указанные
значениях в других полях.


RETRIEVE отыскивает все записи, содержащие указанные значения в
указанных полях.


Простая организация данных.


Простейшим (и наименее эффективным) способом реализации
перечисленных выше операторов работы с файлами является использование
таких примитивов чтения и записи файлов, которые встречаются, например,
в Pascal. В случае использования подобной «организации» (которая на
самом деле является дезорганизацией) записи могут храниться в любом
порядке. Поиск записи с указанными значениями в определённых полях
осуществляется путём полного просмотра файла и проверки каждой его
записи на наличие в ней заданных значений. Вставку в файл можно
выполнять путём присоединения соответствующей записи к концу файла.


В случае изменения записей необходимо просмотреть файл, проверить
каждую запись и выяснить, соответствует ли она заданным условиям
(значения в указанных полях). Если соответствует, в запись вносятся
требуемые изменения. Принцип действия операции удаления почти тот же,
но когда мы поля которой соответствуют значениям, заданным в операции
удаления, мы должны найти способ удалить её. Один из вариантов –
сдвинуть все последовательные записи в своих блоках на одну позицию
вперёд, а первую запись в каждом последующем блоке переместить на
последнюю позицию предыдущего блока данного файла. Однако такой подход
не годится, если записи являются закрепленными, поскольку указатель на
i-ю запись в файле после выполнения этой операции будет указывать на
(i+1)-ю запись.


Если записи являются закреплёнными, следует воспользоваться каким-то
другим подходом. Мы должны как-то помечать удалённые записи, но не
должны смещать оставшиеся на место удалённых (и не должны вставлять на
их место новые записи). Таким образом выполняется логическое удаление
записи из файла, но её место в файле остаётся незанятым. Это нужно для
того, чтобы в случае появления указателя на удалённую запись мы могли,
во-первых, понять, что указываемая запись уже удалена, и, во-вторых,
предпринять соответствующие меры (например, присвоить этому указателю
значение NIL, чтобы в следующий раз не тратить время на его анализ).
Существуют два способа помечать удалённые записи.


Заменить запись на какое-то значение, которое никогда не может стать
значением «настоящей» записи, и, встретив указатель на какую-либо
запись, считать её удалённой, если она содержит это значение.


Предусмотреть для каждой записи специальный бит удаления; этот бит
содержит 1 в удалённых записях и 0 – в «настоящих» записях.


Ускорение операций с файлами.


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


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


Ещё одним непременным атрибутом быстрого выполнения операций с
файлами является возможность непосредственного доступа к блокам (в
отличие от последовательного перебора всех блоков, содержащих файл).
Многие структуры данных, которые используются для быстрого выполнения
операций с файлами, используют указатели на сами блоки, которые
представляют собой физические адреса этих блоков. К сожалению, на языке
Pascal (и многих других языках программирования) невозможно писать
программы, работающие с данными на уровне физических блоков и их
адресов, – такие операции, как правило, выполняются с помощью команд
файловой системы. Однако будет приведено краткое неформальное описание
принципа действия операторов, в которых используется прямой доступ к
блокам.


Хешированные файлы.


Хеширование – широко распространённый метод обеспечения быстрого
доступа к информации, хранящейся во вторичной памяти. Основная идея
этого метода подобна открытому хешированию. Записи файла распределяются
между так называемыми сегментами, каждый из которых состоит из связного
списка одного или нескольких блоков внешней памяти. Имеется таблица
сегментов, содержащая В указателей, – по одному на каждый сегмент.
Каждый указатель в таблице сегментов представляет собой физический
адрес первого блока связного списка блоков для соответствующего
сегмента.


Сегменты пронумерованы от 0 до В-1. Хеш-функция h отображает каждое
значение ключа в одно из целых чисел от 0 до В-1. Если x – ключ, то
h(x) является номером сегмента, который содержит запись с ключом х
(если такая запись вообще существует). Блоки, составляющие каждый
сегмент, образуют связный список. Таким образом, заголовок i-го блока
содержит указатель на физический адрес (i+1)-го блока. Последний блок
сегмента содержит в своём заголовке NIL-указатель.


Такой способ организации показан на рис.2.



x



Таблица


Сегментов


Рис.2. Сегменты, состоящие из связных блоков.


Если размер таблицы сегментов невелик, её можно хранить в основной
памяти. В противном случае её можно хранить последовательным способом в
отдельных блоках. Если нужно найти запись с ключом х, вычисляется h(x)
и находится блок таблицы сегментов, содержащий указатель на первый блок
сегмента h(x), пока не обнаружится блок, который содержит запись с
ключом х. Если исчерпаны все блоки в связном списке для сегмента h(x),
приходим к выводу, что х не является ключом ни одной из записей.


Такая структура оказывается вполне эффективной, если в выполняемом
операторе указываются значения ключевых полей. Среднее количество
обращения к блокам, требующееся для выполнения оператора, в котором
указан ключ записи, приблизительно равняется среднему количеству блоков
в сегменте, которое равно n/bk, если n – количество записей, блок
содержит b записей, а k соответствует количеству сегментов. Таким
образом, при такой организации данных операторы, использующие значения
ключей, выполняются в среднем в k раз быстрее, чем в случае
неорганизованного файла. К сожалению, ускорения операций, не основанных
на использовании ключей, добиться не удаётся, поскольку при выполнении
подобных операций приходится анализировать практически всё содержимое
сегментов. Единственным универсальным способом ускорения операций, не
основанных на использовании ключей, по-видимому, является применение
вторичных индексов.


Чтобы вставить запись с ключом, значение которого равняется х,
нужно сначала проверить, нет ли в файле записи с таким же значением
ключа. Если такая запись есть, выдаётся сообщение об ошибке, поскольку
мы предполагаем, что ключ уникальным образом идентифицирует каждую
запись. Если записи с ключом х нет, мы вставляем новую запись в первый
же блок цепочки для сегмента h(x), в которыё эту запись удаётся
вставить. Если запись не удаётся вставить ни в какой из существующих
блоков сегмента h(x), файловой системе выдаётся команда найти новый
блок, в который будет помещена эта запись. Этот новый блок затем
добавляется в конец цепочки блоков сегмента h(x).


Чтобы удалить запись с ключом х, нужно сначала найти эту запись, а
затем установить её бит удаления. Ещё одной возможной стратегией
удаления (которой, впрочем, нельзя пользоваться, если мы имеем дело с
закреплёнными записями) является замена удалённой записи на последнюю
запись в цепочке блоков сегмента h(x). Если такое изъятие последней
записи приводит к опустошению последнего блока в сегменте h(x), этот
пустой блок можно затем вернуть файловой системе для повторного
использования.


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

-----------------------
4

2

2. 3
3.

3.4 0

5.6 2

7.8 1

r1 r2 r3

r4 r5 r6

r7 r8






Новинки рефератов ::

Реферат: Оценка достоинств и недостатков высшего образования в Российской Федерации. Перспективы и сложности медицинского образования. Оценка достоинств и недостатков подготовки в ММА им. И.М.Сеченова. (Педагогика)


Реферат: Роль углеводов в жизнедеятельности человека (Биология)


Реферат: Автоматизированная обработка учета складских операций и реализации продукции (Компьютеры)


Реферат: Англо-германские противоречия накануне первой мировой войны (История)


Реферат: Бухгалтерский учет в торговле в Украине (Бухгалтерский учет)


Реферат: Петр Первый, взаимосвязь политических и социально-экономических процессов (История)


Реферат: Картофель (как важная кормовая и техническая культура) (Доклад) (Ботаника)


Реферат: Microsoft Exel (Программирование)


Реферат: Проблема взаимодействия семьи и детского дошкольного учреждения в формировании готовности ребенка к школьному обучению (Педагогика)


Реферат: Организационно экономические резервы повышения эфективности производства продукции в мясном скотоводстве на примере колхоза им. Чапаева Ульяновской обл. (Сельское хозяйство)


Реферат: Основные концепции правопонимания (Теория государства и права)


Реферат: Билеты и ответы по туризму и экскурсиям (Туризм)


Реферат: Формирование многопартийности в Республике Беларусь (Политология)


Реферат: Птолемей I (Искусство и культура)


Реферат: Термодинамические характеристики расплавов на основе железа (Металлургия)


Реферат: Заработная плата (Менеджмент)


Реферат: Здоровье человека и окружающая среда (Безопасность жизнедеятельности)


Реферат: Моделирование процессов переработки пластмасс (Технология)


Реферат: Общая собственность: понятия и виды (Гражданское право и процесс)


Реферат: Ролевые игры на уроках английского языка (Педагогика)



Copyright © GeoRUS, Геологические сайты альтруист