Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Программирование / WORD / Лекции по ЯП (часть 2).doc
Скачиваний:
95
Добавлен:
15.04.2015
Размер:
2.24 Mб
Скачать

Линейные списки

В линейном списке каждый элемент связан со следующим и, возможно, с преды­дущим. В первом случае список называется односвязным, во втором — двусвязным. Также применяются термины «однонаправленный» и «двунаправленный». Если последний элемент связать указателем с первым, получится кольцевой список.

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

Над списками можно выполнять следующие операции:

  • начальное формирование списка (создание первого элемента);

  • добавление элемента в конец списка;

  • чтение элемента с заданным ключом;

  • вставка элемента в заданное место списка (до или после элемента с заданным ключом);

  • удаление элемента с заданным ключом;

  • упорядочивание списка по ключу.

При чтении элемент списка не удаляется, в отличие от элемента, выбираемого из стека. Для работы со списком в программе требуется определить ука­затель на его начало. Чтобы упростить добавление новых элементов в конец спи­ска, можно также завести указатель на конец списка.

Рассмотрим программу, которая формирует односвязный спи­сок из пяти элементов, содержащих число и его текстовое представление, а затем выполняет вставку и удаление заданного элемента. В качестве ключа исполь­зуется число.

program List;

const

n = 5;

type

pnode = ^node;

node = record { элемент списка }

d : word;

s : string;

p : pnode;

end;

var

beg : pnode; { указатель на начало списка }

i, key : word;

s : string;

option : word;

const

text: array [1 .. n] of string = ('one', 'two', 'three', 'four', 'five');

{-------------------------------- добавление элемента в конец списка -------------------------}

procedure add(var beg : pnode; d : word; const s : string);

var

p : pnode: { указатель на создаваемый элемент }

t : pnode; { указатель для просмотра списка }

begin

new(p); { создание элемента }

p^.d := d; p^.s := s; { заполнение элемента }

р^.р := nil:

if beg = nil

then beg := p { список был пуст }

else

begin { список не пуст }

t := beg;

while t^.p <> nil do { проход по списку до конца }

t := t^.p;

t^.p := p; { привязка нового элемента к последнему }

end

end;

{-------------------------------- поиск элемента по ключу -------------------------}

function find(beg : pnode; key : word; var p, pp : pnode) : Boolean;

begin

p := beg;

while p <> nil do { 1 }

begin

if p^.d = key { 2 }

then

begin

find := true;

exit

end;

pp := p; { 3 }

p :=p^.р; { 4 }

end;

find := false;

end;

{-------------------------------- вставка элемента -------------------------}

procedure insert(beg : pnode; key, d : word; const s : string);

var

p : pnode; { указатель на создаваемый элемент }

pkey : pnode; { указатель на искомый элемент }

pp : pnode; { указатель на предыдущий элемент }

begin

if not find(beg, key, pkey, pp)

then

begin

writeln(' вставка не выполнена');

exit;

end;

new(p); { 1 }

p^.d := d; p^.s := s; { 2 }

р^.р := pkey^.p; { 3 }

pkey^.p := p; { 4 }

end;

{-------------------------------- удаление элемента -------------------------}

procedure del(var beg : pnode: key : word);

var

p : pnode; { указатель на удаляемый элемент }

pp : pnode; { указатель на предыдущий элемент }

begin

if not find(beg, key, p, pp)

then

begin

writeln(' удаление не выполнено');

exit;

end;

if p = beg then beg := beg^.p { удаление первого элемента }

else pp^.p := р^.р;

dispose(p);

end;

{-------------------------------- вывод списка -------------------------}

procedure print(beg var p : pnode);

var

p : pnode; { указатель для просмотра списка }

begin

p := beg;

while p <> nil do begin { цикл по списку }

writeln(p^.d:3, p^.s); { вывод элемента }

p:=р^.р { переход к следующему элементу списка }

end;

end;

{-------------------------------- главная программа -------------------------}

begin

for i := 1 to 5 do add(beg, i, text[i]);

while true do

begin

writeln('1 - вставка, 2 - удаление, 3 - вывод, 4 - выход');

readln(option);

case option of

1: begin { вставка }

writeln ('Ключ для вставки?');

readln(key);

writeln('Вставляемый элемент?');

readln(i): readln(s);

insert(beg, key, i, s);

end;

2: begin { удаление }

writeln('Ключ для удаления?');

readln(key);

del(beg, key);

end;

3: begin { вывод }

writeln('Вывод списка:');

print(beg);

end;

4: exit; { выход }

end; {case}

writeln;

end;

end.

Функция поиска элемента find возвращает true, если искомый элемент найден, и false в противном случае. Поскольку одного факта отыскания элемента недостаточно, функция также возвращает через список параметров два указателя: на найденный элемент р и на предшествующий ему pp. Последний требуется при удалении элемента из списка, поскольку при этом необходимо связывать преды­дущий и последующий по отношению к удаляемому элементы.

Сначала указатель р устанавливается на начало списка и организуется цикл просмотра списка (оператор 1). Если поле данных очередного элемента совпало с заданным ключом (оператор 2), формируется признак успешного поиска и функция завершается. В противном случае перед переносом указателя на следующий элемент списка (он хранится в поле р текущего элемента, оператор 4) его значение запоминается в переменной рр (оператор 3) для того, чтобы при следующем проходе цикла в ней находился указатель на предыдущий элемент.

Если элемента с заданным ключом в списке нет, цикл завершится естественным образом, поскольку последний элемент списка содержит nil в поле р указателя на следующий элемент.

Вставка элемента выполняется после элемента с заданным ключом (процедура Insert). Если с помощью функции find место вставки определить не удалось, вы­водится сообщение и процедура завершается (в этом случае можно было использовать и другой алгоритм — добавлять элемент к концу списка, это определяется конкретным предназначением программы). Если элемент найден, указатель на него заносится в переменную pkey.

Под новый элемент выделяется место в динамической памяти (оператор 1), и информационные поля элемента заполняются переданными в процедуру значе­ниями (оператор 2). Новый элемент р вставляется между элементами pkey и следующим за ним (его адрес хранится в pkey^. р). Для этого в операторах 3 и 4 уста­навливаются две связи (cм. рис.).

Вставка элемента в список

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

Соседние файлы в папке WORD