Компания ALT Linux
Опубликован: 10.04.2015 | Доступ: свободный | Студентов: 762 / 0 | Длительность: 14:03:00
Специальности: Программист, Преподаватель
Лекция 8:

Работа со строками и записями

< Лекция 7 || Лекция 8: 12 || Лекция 9 >

8.2 Работа с записями

В большинстве случаев при написании программ применяются простые типы данных (числа, строки). Но часто возникает необходимость объединить в одном типе несколько разных типов данных. В Free Pascal для этого применяется структурный тип данныхзапись. Запись состоит из фиксированного числа компонентов, называемоых полями записи. Общий синтаксис объявления записи выглядит так:

type

имя_записи = record

поле1 : тип;

поле2 : тип;

...

полеN : тип;

end;

В программе переменная типа записи объявляется следующим образом:

var имя_переменной : имя_записи;

К каждому из компонентов записи можно получить доступ, используя составное имя. Для этого вначале пишется имя переменой, затем точка, затем имя поля.

Рассмотрим следующий пример. Создадим запись — треугольник с тремя полями — сторонами треугольника. В программе произведен расчёт площади треугольника по формуле Герона.

program Project1;
type
	Triangle= record
	a, b, c : real; //стороны треугольника
	end;
var x : Triangle; //объявление переменой типа записи Triangle
	p, s : real;
begin
	write ( ’ a= ’ );
	readln ( x. a ); //чтение поля а
	write ( ’ b= ’ );
	readln ( x. b ); //чтение поля b
	write ( ’ c= ’ );
	readln ( x. c ); //чтение поля c
	p :=( x. a+x. b+x. c ) / 2; //вычисление полупериметра
//проверка существования треугольника - подкоренное выражение
//для формулы Герона должно быть положительным
	if ( p - x. a ) * ( p - x. b ) * ( p - x. c)>0 then
	begin
//вычисление и вывод площади
		s := sqrt ( p * ( p - x. a ) * ( p - x. b ) * ( p - x. c ) );
		writeln ( ’ Square= ’, s : 7 : 2 );
	end
	else
		writeln ( ’ Not  triangle ’ );
	readln;
end.

Элементами записей могут быть как простые, так и структурные типы. Никаких ограничений по уровню вложенности структур не существует.

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

type
adress = record
	city, street : string; //город, улица
	house, apartment : integer; //дом, квартира
end;
student = record
	surname, name : string; //фамилия, имя
	group : string; //группа
estimation : array [ 1.. 5 ] of integer; //оценки
residence : adress; //прописка
end;

При объявлении такой записи обращение к полям осуществляется так:

var Ivanov : student;
	x : array [ 1.. 100 ] of student;
begin
	Ivanov. group := ’Ф08 ’;
	Ivanov. residense. city := ’Киев ’;
	x [ 1 ]. estimation [ 5 ] : = 3; //у первого студента 5-я оценка =3
...

C использованием ключевого слова With к полям записи можно обращаться без указания префикса каждый раз с названием поля:

with переменная do оператор

Например:

with stud do
begin
	with residence do
	begin
		city := ’Донецк ’;
		street := ’Артёма ’;
		house :=145;
		apartment := 31;
	end;
	surname := ’Иванов ’;
	name:= ’Андрей ’;
	birthday := ’ 01. 11. 1990 ’;
	group := ’Ф07 ’;
	estimation [ 1 ] : = 3; estimation [ 2 ] : = 5; estimation [ 3 ] : = 4;
	estimation [ 4 ] : = 3; estimation [ 5 ] : = 5;
end;
ЗАДАЧА 8.1. Создать базу данных, содержащую сведения о студентах. В программе предусмотреть расчёт среднего балла студента, сортировку по алфавиту, вывод результатов в диалоговое окно и в текстовый файл.
Пример формы для задачи 8.1

Рис. 8.4. Пример формы для задачи 8.1

Создадим новый проект. На форму (рис. 8.4) поместим необходимое количество объектов типа TEdit для ввода исходных данных и объект StringGrid для вывода результатов. Для объекта StringGrid1 зададим свойства ColCount=8 и RowCount=1. Также создадим три кнопки:

  • Запомнить данные — после ввода данных они считываются, и поля ввода очищаются для дальнейшего ввода;
  • Сортировка по алфавиту — после нажатия этой кнопки записи сортируются по алфавиту фамилий;
  • Вывести данные — результаты выводятся в таблицу ниже и записываются в текстовый файл.
unit Unit1;
{$mode objfpc}{$H+}
interface
uses
	Classes, SysUtils, LResources, Forms, Controls, Graphics,
	Dialogs, StdCtrls, Grids;
type
	{ TForm1 }
	TForm1 = class (TForm)
		Button1 : TButton;
		Button2 : TButton;
		Button3 : TButton;
		Edit1 : TEdit;
		Edit10 : TEdit;
		Edit11 : TEdit;
		Edit12 : TEdit;
		Edit2 : TEdit;
		Edit3 : TEdit;
		Edit4 : TEdit;
		Edit5 : TEdit;
		Edit6 : TEdit;
		Edit7 : TEdit;
		Edit8 : TEdit;
		Edit9 : TEdit;
		Label1 : TLabel;
		Label2 : TLabel;
		Label3 : TLabel;
		Label4 : TLabel;
		Label5 : TLabel;
		Label6 : TLabel;
		Label7 : TLabel;
		Label8 : TLabel;
		Label9 : TLabel;
		StringGrid1 : TStringGrid;
		//Процедура-обработчик кнопки "Запомнить данные"
		procedure Button1Click ( Sender : TObject );
		//Процедура-обработчик кнопки "Вывести данные"
		procedure Button2Click ( Sender : TObject );
		//Процедура-обработчик кнопки "Сортировать по алфавиту"
		procedure Button3Click ( Sender : TObject );
		//Процедура инициализации формы
		procedure FormCreate ( Sender : TObject );
	private
		{ private declarations }
	public
		{ public declarations }
	end;
//объявление записи адреса студента
	adress = record
		city, street : string; //город, улица
		house, apartment : integer; //дом, квартира
	end;
//объявление записи сведений о студенте
	student = record
		surname, name : string; //фамилия, имя
		group : string; //группа
		estimation : array [ 1.. 5 ] of integer; //оценки
		residence : adress; //прописка
		s _ ball : real; //средний балл
	end;
var
	Form1 : TForm1;
//массив переменных типа "студент"
	x : array [ 0.. 100 ] of student;
	//переменная для подсчета количества студентов
i : integer;
implementation
{ TForm1 }
//процедура инициализации формы
procedure TForm1. FormCreate ( Sender : TObject );
begin
	i : = 0; //количество студентов вначале равно 0
end;
//процедура работы кнопки "Запомнить данные"
procedure TForm1. Button1Click ( Sender : TObject );
var sum, j : integer;
begin
	//чтение данных с формы
		x [ i ]. surname := Edit1. Text;
		x [ i ]. name:= Edit2. Text;
		x [ i ]. group := Edit3. Text;
		x [ i ]. residence. city := Edit5. Text;
		x [ i ]. residence. street := Edit6. Text;
		x [ i ]. residence. house := StrToInt ( Edit7. Text );
		x [ i ]. residence. apartment := StrToInt ( Edit8. Text );
		x [ i ]. estimation [ 1 ] : = strToInt ( Edit4. Text );
		x [ i ]. estimation [ 2 ] : = strToInt ( Edit9. Text );
		x [ i ]. estimation [ 3 ] : = strToInt ( Edit10. Text );
		x [ i ]. estimation [ 4 ] : = strToInt ( Edit11. Text );
		x [ i ]. estimation [ 5 ] : = strToInt ( Edit12. Text );
	//расчет среднего балла студента
		sum : = 0;
		for j :=1 to 5 do
		sum:=sum +x [ i ]. estimation [ j ];
		x [ i ]. s _ ball :=sum / 5;
		inc ( i );
	//очищение полей ввода для следующих данных
		Edit1. Text := ’ ’; Edit2. Text := ’ ’;
		Edit3. Text := ’ ’; Edit4. Text := ’ ’;
		Edit5. Text := ’ ’; Edit6. Text := ’ ’;
		Edit7. Text := ’ ’; Edit8. Text := ’ ’; Е
		d i t 9. Text := ’ ’; Edit10. Text := ’ ’;
		Edit11. Text := ’ ’; Edit12. Text := ’ ’;
end;
//процедура работы кнопки "Вывести данные"
procedure TForm1. Button2Click ( Sender : TObject );
var f : textfile;
	j : integer;
	s : string;
begin
	//количество строк в таблице StringGrid будет на 1 больше,
	//чем количество студентов (плюс 1 строка для шапки)
		StringGrid1. RowCount:= i +1;
	//вывод шапки таблицы
		StringGrid1.Cells [ 1, 0 ] : = ’Фамилия ’;
		StringGrid1.Cells [ 2, 0 ] : = ’Имя ’;
		StringGrid1.Cells [ 3, 0 ] : = ’Группа ’;
		StringGrid1.Cells [ 4, 0 ] : = ’Город ’;
		StringGrid1.Cells [ 5, 0 ] : = ’Улица ’;
		StringGrid1.Cells [ 6, 0 ] : = ’Дом/кв. ’;
		StringGrid1.Cells [ 7, 0 ] : = ’Средний балл ’;
//вывод сведений о студентах в j-ю строку
	for j :=1 to i do
	begin
		StringGrid1.Cells [ 1, j ] : = x [ j - 1 ]. surname;
		StringGrid1.Cells [ 2, j ] : = x [ j - 1 ]. name;
		StringGrid1.Cells [ 3, j ] : = x [ j - 1 ]. group;
		StringGrid1.Cells [ 4, j ] : = x [ j - 1 ]. residence.city;
		StringGrid1.Cells [ 5, j ] : = x [ j - 1 ]. residence.street;
		s := inttostr ( x [ j - 1 ]. residence. house)+ ’ / ’+
			inttostr ( x [ j - 1 ]. residence. apartment );
		StringGrid1.Cells [ 6, j ] : = s;
		StringGrid1.Cells [ 7, j ] : = floattostr ( x [ j - 1 ]. s _ ball );
	end
	//вывод результатов в текстовый файл
assignfile ( f, ’ g : \ student.txt ’ );
	rewrite ( f );
	for j :=1 to i do
	begin
		writeln ( f, x [ j - 1 ]. surname : 20, x [ j - 1 ]. name : 15,
		x [ j - 1 ]. residence.city : 15, ’, ’, x [ j - 1 ]. residence. street : 15,
		x [ j - 1 ]. residence. house : 4, ’ / ’, x [ j - 1 ]. residence. apartment,
		’  Sr_ball= ’, x [ j - 1 ]. s \ _ball : 4 : 1 );
	end;
	closefile ( f );
end;
//процедура работы кнопки "Сортировать по алфавиту"
procedure TForm1. Button3Click ( Sender : TObject );
var j, k : integer;
	temp : student;
	f : textfile; //временная переменная для сортировки
begin
	for j := 0 to i -1 do
		for k:= j +1 to i -1 do
			if x [ j ]. surname > x [ k ]. surname then
			begin
				temp:=x [ j ];
				x [ j ] : = x [ k ];
				x [ k ] : = temp;
			end;
end;
initialization
{$I unit1.lrs}
end.

При запуске программы и вводе сведений окно формы выглядит, как на рис. 8.5.

Когда запись введена, следует щёлкнуть по кнопке Запомнить данные, при этом поля ввода очищаются для ввода следующей записи. После щелчка по кнопке Вывести данные таблица заполняется введёнными сведениями (рис. 8.6).

Окно формы ввода сведений о студенте

Рис. 8.5. Окно формы ввода сведений о студенте
Окно формы вывода сведений о студентах

Рис. 8.6. Окно формы вывода сведений о студентах
Окно формы после сортировки записей по алфавиту фамилии

Рис. 8.7. Окно формы после сортировки записей по алфавиту фамилии

После щелчка по кнопке Сортировать по алфавиту нужно повторно щелкнуть по кнопке Вывести данные, чтобы увидеть отсортированный список (рисунок 8.7).

< Лекция 7 || Лекция 8: 12 || Лекция 9 >
Юрий Шутиков
Юрий Шутиков

По первому тесту выполнил дважды задания. Результат получается правильный (проверял калькулятором). Пишет, что "Задание не проверено" и предлагает повторить. 
 

Евгений Силуков
Евгений Силуков

Еще в декабре выполнил тест №1, а его все так и не проверили.