Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
СПО_Лекции_2011_01.docx
Скачиваний:
20
Добавлен:
31.07.2019
Размер:
521.09 Кб
Скачать

Удаление элемента из списка.

При удалении элемента из списка необходимо различать три случая:1. Удаление элемента из начала

списка.

      1. Удаление элемента из середины списка.

      2. Удаление из конца списка. Удаление элемента из начала списка.

List := Head; { запомним адрес первого элемента списка } Head := HeadA.List; { теперь Head указывает на второй элемент списка } Dispose(List); { освободим память, занятую переменной ListA } Удаление элемента из середины списка.

Для этого нужно знать адреса удаляемого элемента и элемента, находящегося в списке перед ним. List := Head;

While (List<>nil) and (ListA.Data<>Digit) do begin x := List;

List := ListA.Next; end;

xA.Next := ListA.Next; Dispose(List);

Удаление из конца списка.

Оно производится, когда указатель х показывает на предпоследний элемент списка, а List - на последний.

List := Head; x := Head; While ListA.Next<>nil do begin x := List;

List := ListA.Next; end; xA.Next := nil; Dispose(List);

Лабораторная работа № 3 Цель: Получить практические навыки работы в среде Borland Delphi на языке Assembler.

Реализовать дополнительную процедуру сортировки (метод остается таким же, как и в ЛР1) используя команды assembler.

Теоретическая часть (в популярном для студентов стиле).

Смещение это положение относительно чего-то. смещение это положение, адрес, местонахождение в памяти, которое всегда измеряется в байтах. У каждой переменной и у каждой команды есть смещение. В MS DOS и в Win3.x память была сегментной. Вся память делилась на сегменты по 64 КВ., и адрес (именно адрес, а не смещение) задавалась, форматом сегмент: офсет. Сегмент задаёт порядковый номер сегмента, а офсет задаёт смещение чего-либо в этом сегменте.

Память

Сегмент 1 (64kb)

Сегмент 2 (64kb)

Сегмент 3 (64kb)

Сегмент 4 (64kb)

и так далее

Это применимо к 16-разрядным приложениям.

Таблица выше иллюстрирует общую память, разделенную на сегменты по 64kb. Здесь максимум 65536 сегментов. Теперь возьмем один из сегментов:

Сегмент 1(64kb)

Смещение 1

Смещение 2

Смещение 3

Смещение 4

Смещение 5

и так далее

Вы спросите, зачем нужна эта сегментность. Всему есть разумное объяснение. Дело в том в том, что ДОС 16 битный. И регистры 16 битные. С помощью 16 бит можно адресовать только 65536 байт памяти и 65536 сегментов.

Например:

0030:4012 (Всё шестнадцатиричное)

Это означает: сегмент 30, смещение 4012. Чтобы узнать, что находится в том адресе, вы сначала переходите на сегмент 30, а затем в сегменте смещаетесь на 4012. В предыдущих уроках мы узнали о сегментных и указательных регистрах. Например, сегментные регистры: CS - Сегмент кода DS - Сегмент данных SS - Сегмент стека ES - Дополнительный сегмент FS - Универсальный сегмент GS - Универсальный сегмент

Названия говорят о их функциях: сегмент кода (CS) содержит номер секции, где вылнен текущий код. Сегмент

данных для получения данных из текущего сегмента. На стек указывает сегмент стека (SS) (подробнее о стеке я раскажу позже), ES, FS, GS - универсальные регистры и могут использоваться для любого сегмента (не в Win32). Теперь о Win32

В 16-разрядном программировании, сегменты необходимы. К счастью, эта проблема решена в 32-разрядном Windows (95 и выше). Вы все еще имеете сегменты, но вам не нужно заботиться о них, потому, что они уже не 64kb (как в 16-разрядном), а 4 Гб. Windows вероятно даже "повиснет", если вы попытаетесь изменить один из сегментных регистров. Это называеся плоской моделью памяти (flat). Здесь есть только смещения и они теперь 32-разрядные (в диапазоне от 0 до 4,294,967,295). Каждая ячейка в памяти указывается смещением. Это действительно одно из лучших преимуществ 32-разрядного программирования над 16-разрядным. Так что теперь вы можете забыть о сегментных регистрах и сосредоточиться на других регистрах. Но 4ГБ может и не быть на машине, поэтому эта память называется виртуальной. Мало того, каждый процесс выполняется в своём виртуальном адресном пространстве. А это значит, что никакой другой процесс не сможет получить доступ к вашей памяти. Для этого существуют специальные API функции, но для их использования нужно иметь специальные привилегии в системе. Память 0 -0000FFFF не используется и служит для выявления нулевых указателей, значит, если вы укажете адрес 0000С567, то он будет считаться нулевым, удобно, не правда ли? Любая попытка обратится к этой памяти приводит к ошибке. Память выше 80000000 одна для всех процессов. В этой памяти находится код нулевого кольца, структуры ядра, код планировщика задач, код драйверов, диспетчер ввода вывода, таблица прерываний и т.д. Любая попытка обратиться к памяти ядра приводит к ошибке и к немедленному завершению приложения. Память в диапазоне 00001000 -7FFFFFFF доступна для 3 кольца, т.е. для вашего приложения. С ней вы, что хотите то и делаете, в неё также грузятся Дллки.