11833

Модули. Многофайловые проекты

Лабораторная работа

Информатика, кибернетика и программирование

Лабораторная работа №13. Модули. Многофайловые проекты 1 Цель и порядок работы Цель работы ознакомиться с возможностью работы с многофайловыми проектами в среде разработки Visual Studio и научиться применять полученные знания при создании собственных модулей. Порядок...

Русский

2013-04-12

227 KB

47 чел.


Лабораторная работа №13. Модули. Многофайловые проекты

1 Цель и порядок работы

Цель работы – ознакомиться с возможностью работы с многофайловыми проектами в среде разработки Visual Studio и научиться применять полученные знания при создании собственных модулей.

Порядок выполнения работы:

  •  ознакомиться с описанием лабораторной работы;
  •  получить задание у преподавателя, согласно своему варианту;
  •  написать программу и отладить ее на ЭВМ;
  •  оформить отчет.

2 Краткая теория

2.1 Особенности работы с многофайловыми проектами

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

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

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

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

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

Исходные тексты совокупности функций для решения какой-либо подзадачи, как правило, размещаются в отдельном модуле (файле). Такой файл называют исходным (sources). Обычно он имеет расширение .c или .cpp Прототипы всех функций исходного файла выносят в отдельный так называемый заголовочный файл (header file), для него принято использовать расширение .h или .hpp.

Таким образом, заголовочный файл xxx.h содержит интерфейс для некоторого набора функций, а исходный файл ххх.срр содержит реализацию этого набора. Если некоторая функция из указанного набора вызывается из какого-то другого исходного модуля ууу.срр, то обязаны включить в этот модуль заголовочный файл xxx.h с помощью директивы #include. Негласное правило стиля программирования на C++ требует включения этого же заголовочного файла (с помощью #include) и в исходный файл ххх.срр.

Хотя использование глобальных переменных считается дурным тоном в программировании, иногда возникают ситуации когда невозможно избежать их использования. В многофайловом проекте возможны два «вида глобальности переменных». Если некоторая глобальная переменная glvar1 объявлена в файле ххх.срр с модификатором static, то она видима от точки определения до конца этого файла, то есть области видимости ограничена файлом. Если же другая глобальная переменная glvar2 объявлена в файле ххх.срр без модификатора static, то она может быть видимой в пределах всего проекта. Правда, для того, чтобы она оказалась видимой в другом файле, необходимо в этом файле ее объявление с модификатором extern (рекомендуется это объявление поместить в файл xxx.h).

В заголовочном файле принято размещать:

  •  определения типов, задаваемых пользователем, констант, шаблонов;
  •  объявления (прототипы) функций;
  •  объявления внешних глобальных переменных (с модификатором extern);
  •  пространства имен.

При подобно использовании заголовочных файлов возникает проблема их повторного включения. Проблема может возникнуть при иерархическом проектировании структур данных, когда в некоторый заголовочный файл ууу.h включается при помощи директивы include другой заголовочный файл xxx.h (например, для использования типов, определенных в этом файле) и где-нибудь еще, например в основной программе.

Для решения этой проблемы рекомендуется использовать так называемые стражи включения. Данный способ состоит в следующем: чтобы предотвратить заголовочных файлов, содержимое каждого .h файла должно находиться между директивами условной компиляции #ifndef #endif, а внутри устанавливаться признак включения при помощи директивы #define.

#ifndef FILENAME_H

#define FILENAME_H

/* содержимое заголовочного файла */

#endif

2.2 Добавление новых файлов в проект в среде разработки Visual Studio 2008

Для добавления новых файлов в проект новых файлов в проект необходимо вызвать контекстное меню над именем проекта в «Проводнике решений» (Solution Explorer) (см. рисунок 13.1) и далее выбрать раскрывающийся пункт «Добавить» (Add). В нем нас на текущий момент будут интересовать два подпункта “New Item …” (Новый объект) и “Exiting Item …” (Существующий объект). Первый создает новый файл, одного из предложенных типов, а второй добавляет к проекту уже существующий. Диалоговое окно с вариантами доступных для создания файлов представлено на рисунке 13.2.

Результат добавления файлов в проект представлен на рисунке 13.3. Теперь достаточно воспользоваться директивой #include для использования функций, констант и типов данных, располагающихся в новых файлах.

Рисунок 13.1 – Добавление нового файла в проект

Рисунок 13.2 – Диалоговое окно выбора типа добавляемого файла

Рисунок 13.3 – Результат добавления новых файлов в проект

2.3 Пример работы с многофайловыми проектами

Разберем работу с многофайловыми проектами на примере задачи из предыдущей лабораторной работы.

Пример 13.1 Нахождение расстояния между точками

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

Алгоритм решения:

  •  Задать или ввести с клавиатуры координаты точек.
  •  Для пары точек определить расстояние по одной из формул:
    •  для декартовой системы:
    •  для полярной системы:
  •  Вывести результаты на экран.

Определение требуемых типов данных.

Для удобства хранения информации о координате точки в одной из систем координат удобно воспользоваться структурами, хранящими пары координат одной и той же точки. Для простоты назовем их cartesian для декартовой, и polar для полярной системы координат.

Разбиение на подзадачи.

В результате описанного алгоритма и задания определяем спецификации нужных нам функций. Для удобства использования воспользуемся перегрузкой:

1. double len(cartesian a, cartesian b) – принимает две координаты в декартовой системе и возвращает длину.

2. double len(polar a, polar b) – принимает две координаты в полярной системе и возвращает длину.

3. double len(cartesian a, polar b) – принимает первую координату в декартовой системе, а вторую в полярной, и возвращает длину.

4. double len(polar a, cartesian b) – принимает первую координату в полярной системе, а вторую в декартовой, и возвращает длину.

5. polar cartesian_to_polar(cartesian x) – переводит координаты точки из в декартовой системы в полярную.

6. cartesian polar_to_cartesian(polar x) – переводит координаты точки из в полярной системы в декартову.

Разбиение на модули.

Исходя из логики работы программы, а также, возможного дальнейшего применения полученных результатов естественным будет разбить ее на два файла: PRG-Lab13.cpp – содержащий точку входа в программу, а также основные вызовы функции и взаимодействие с пользователем; points.cpp – содержащий реализации вышеперечисленных функций; а также points.h – содержащий интерфейсы данных функций и типы данных, и служащий для подключения возможностей работы с точками в различных системах координат.

Файл points.h

// points.h : Содержит определения основных типов и прототипы функций.

//

#ifndef POINTS_H

#define POINTS_H

//Объявления типов

//Точка в декартовой системе координат

struct cartesian{

  double x, y;

};

//Точка в полярной системе координат

struct polar{

  double r, pi;

};

//Прототипы функций

double len(polar a, polar b);

double len(cartesian a, cartesian b);

double len(cartesian a, polar b);

double len(polar a, cartesian b);

polar cartesian_to_polar(cartesian x);

cartesian polar_to_cartesian(polar x);

#endif

Файл points.cpp

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

// points.cpp : Содержит определения функций

//

#include "stdafx.h"

#include "points.h"

#include <math.h>

double len(polar a, polar b)

{

  return sqrt( pow(a.r, 2) + pow(b.r, 2) - 2*a.r*b.r*cos(a.pi - b.pi) );

}

double len(cartesian a, cartesian b)

{

  return sqrt( pow(a.x - b.x, 2) + pow(a.y - b.y, 2) );

}

double len(cartesian a, polar b)

{

  cartesian c;

  c = polar_to_cartesian(b);

  return len(a, c);

}

double len(polar a, cartesian b)

{

  return len(b, a);

}

polar cartesian_to_polar(cartesian x)

{

  polar y;

  y.r = sqrt(pow(x.x, 2) + pow(x.x, 2));

  if (y.r == 0)

      y.pi = 0;

  else

      y.pi = asin(x.y/y.r);

  return y;

}

cartesian polar_to_cartesian(polar x)

{

  cartesian y;

  y.x = x.r*cos(x.pi);

  y.y = x.r*sin(x.pi);

  return y;

}

Файл PRG-Lab13.cpp

// PRG-Lab13.cpp : Основной файл проекта, содержит точку входа в программу

//

#include "stdafx.h"

#include <iostream>

#include "points.h"

using namespace std;

void main(int argc, char* argv[])

{

  setlocale(LC_ALL, "Russian");

  const double PI = 3.14159;

  cartesian a = {3, 0},

            b = {1, 1};

  polar c = {1.41, PI/4},

        d = {3.1, 0.95};

  //вызовы перегруженных функций

  //для точек в различных системах координат

  //декартова - декартова

  cout << len(a, b) << endl;

  //полярная - полярная

  cout << len(c, d) << endl;

  //декартова - полярная

  cout << len(a, d) << endl;

  //полярная - декартова

  cout << len(c, b) << endl;

}

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

3 Контрольные вопросы

  1.  В чем отличие многофайлового проекта?
  2.  Как добавить новый файл в проект?
  3.  Что такое модуль?
  4.  Для чего нужно разбиение на модули?
  5.  Для чего применяются заголовочные файлы?
  6.  Что размещается в заголовочных файлах, а что выносится в тело модуля?

4 Задание

  1.  Создать новый проект.
  2.  Добавить к проекту файл исходным кодом (.h и .cpp).
  3.  Вынести в отдельный модуль функции и типы данных.
  4.  Написать программу в соответствии с вариантом задания из пункта 5, задав начальные значения при объявлении переменных.
  5.  Отладить и протестировать программу.
  6.  Создать новый проект.
  7.  Подключить модуль из предыдущего задания в новый проект.
  8.  Написать программу в соответствии с вариантом задания из пункта 5, введя начальные значения переменных с клавиатуры.
  9.  Отладить и протестировать программу.
  10.  Оформить отчёт.

5 Варианты заданий

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

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

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

Два параметра типа int.

Два параметра типа long.

Два параметра типа float.

Два параметра типа double.

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

Одномерный массив типа int размерностью N.

Одномерный массив типа float размерностью N.

Одномерный массив типа double размерностью N.

Определить функцию, возвращающую минимальное из нескольких чисел. Выполнить перегрузку функции для следующих типов параметров:

Три параметра типа int.

Четыре параметра типа int.

Три параметра типа float.

Два параметра типа double.

Определить функцию, возвращающую количество недель с начала года. Выполнить перегрузку функции для следующих типов параметров:

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

Определить функцию, возвращающую количество недель до конца года. Выполнить перегрузку функции для следующих типов параметров:

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

Определить функцию, возвращающую НОД нескольких чисел. Выполнить перегрузку функции для следующих типов параметров:

Два параметра типа int.

Два параметра типа long.

Два параметра типа float.

Два параметра типа double.

Определить функцию, возвращающую количество минут до окончания суток. Выполнить перегрузку функции для следующих типов параметров:

Структура «время» (часы, минуты, секунды).

Три целочисленных параметра: часы, минуты, секунды.

Два целочисленных параметра: часы, минуты.

Определить функцию, возвращающую предыдущую минуту. Выполнить перегрузку функции для следующих типов параметров:

Структура «время» (часы, минуты, секунды).

Три целочисленных параметра: часы, минуты, секунды.

Два целочисленных параметра: часы, минуты.

Определить функцию, находящую сумму элементов массива. Выполнить перегрузку функции для следующих типов параметров:

Одномерный массив типа int размерностью N.

Одномерный массив типа float размерностью N.

Одномерный массив типа double размерностью N.

Определить функцию, проверяющую верна ли дата. Выполнить перегрузку функции для следующих типов параметров:

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

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

Два параметра типа структура «точка» (координаты x, y).

Четыре параметра типа float.

Четыре параметра типа double.

Определить функцию, возвращающую следующую минуту. Выполнить перегрузку функции для следующих типов параметров:

Структура «время» (часы, минуты, секунды).

Три целочисленных параметра: часы, минуты, секунды.

Два целочисленных параметра: часы, минуты.

Определить функцию, находящую произведение ненулевых элементов массива. Выполнить перегрузку функции для следующих типов параметров:

Одномерный массив типа int размерностью N.

Одномерный массив типа float размерностью N.

Одномерный массив типа double размерностью N.

Определить функцию, возвращающую прошедшее время в минутах (считать, что разница между передаваемыми значениями не превышает 24 часа). Выполнить перегрузку функции для следующих типов параметров:

Два параметра типа структура «время» (часы, минуты, секунды).

Шесть целочисленных параметра: часы, минуты, секунды.

Четыре целочисленных параметра: часы, минуты.

Определить функцию, возвращающую минимальное из нескольких чисел. Выполнить перегрузку функции для следующих типов параметров:

Два параметра типа int.

Три параметра типа int.

Два параметра типа float.

Три параметра типа float.

Три параметра типа double.

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

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

Определить функцию, возвращающую n-ю степень числа x. Выполнить перегрузку функции для следующих типов параметров:

Два параметра: x и n – оба типа int.

Два параметра: x и n – оба типа float.

Два параметра: x – типа float, и n – типа int.

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

Одномерный массив типа int размерностью N.

Одномерный массив типа float размерностью N.

Одномерный массив типа double размерностью N.

Определить функцию, возвращающую количество минут с начала суток. Выполнить перегрузку функции для следующих типов параметров:

Структура «время» (часы, минуты, секунды).

Три целочисленных параметра: часы, минуты, секунды.

Два целочисленных параметра: часы, минуты.

Определить функцию, возвращающую среднеарифметическое нескольких чисел. Выполнить перегрузку функции для следующих типов параметров:

Два параметра типа int.

Три параметра типа int.

Три параметра типа float.

Два параметра типа double.

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

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

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

Одномерный массив типа int размерностью N.

Одномерный массив типа float размерностью N.

Одномерный массив типа double размерностью N.

Определить функцию, возвращающую максимальное из нескольких чисел. Выполнить перегрузку функции для следующих типов параметров:

Два параметра типа int.

Три параметра типа int.

Два параметра типа float.

Три параметра типа double.

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

Структура «дата» (год, месяц, день).

Три целочисленных параметра: год, месяц, день.

Два целочисленных параметра: месяц, день (считать передаваемые числа датой текущего года).

6 Содержание отчета

  1.  Титульный лист.
  2.  Наименование и цель работы.
  3.  Краткое теоретическое описание.
  4.  Задание на лабораторную работу.
  5.  Схема алгоритма.
  6.  Листинг программы.
  7.  Результаты выполнения программы.


 

А также другие работы, которые могут Вас заинтересовать

22828. ВИМІРЮВАННЯ НАПРУЖЕННОСТІ МАГНІТНОГО ПОЛЯ ВЗДОВЖ ОСІ СОЛЕНОЇДА ІНДУКЦІЙНАМ МЕТОДОМ 141 KB
  ВИМІРЮВАННЯ НАПРУЖЕННОСТІ МАГНІТНОГО ПОЛЯ ВЗДОВЖ ОСІ СОЛЕНОЇДА ІНДУКЦІЙНАМ МЕТОДОМ Явище електромагнітної індукції полягає у виникненні е. Напруженість магнітного поля в будьякій точці А що лежить на осі ОО соленоїда чисельно дорівнює алгебраїчній сумі напруженостей магнітних полів створених у точці А всіма витками спрямована вздовж осі за правилом свердлика 3 Де n число витків за одиницю довжини соленоїда І величина струму; кути що утворює радіусвектор проведений з точки А до крайніх витків соленоїда мал....
22829. ЯВИЩЕ ГІСТЕРЕЗИСУ В ФЕРОМАГНЕТИКУ 115 KB
  ЯВИЩЕ ГІСТЕРЕЗИСУ В ФЕРОМАГНЕТИКУ Особливий клас магнетиків становлять феромагнетики речовини здатні мати намагнічення у відсутності зовнішнього магнітного поля.21 наведена залежність модуля вектора намагнічення від напруженості зовнішнього поля для феромагнетика з попереднім магнітним полем рівним нулеві основна або нульова крива намагнічення . При деякому значенні H намагнічення досягає насичення оскільки вектор магнітної індукції та вектора намагнічення звязані співвідношенням то при досягненні вектор стає функцією від:...
22830. ВИЗНАЧЕННЯ КОНЦЕНТРАЦІЇ НОСІЇВ ЗАРЯДУ В НАПІВПРОВІДНИКАХ З ЕФЕКТУ ХОЛЛА 71.5 KB
  ВИЗНАЧЕННЯ КОНЦЕНТРАЦІЇ НОСІЇВ ЗАРЯДУ В НАПІВПРОВІДНИКАХ З ЕФЕКТУ ХОЛЛА В основу вимірювання концентрації електронів покладено явище Холла яке полягає у виникненні поперечної різниці потенціалів при проходженні струму по провіднику напівпровіднику який знаходиться в магнітному полі перпендикулярному до лінії струму. Ефект Холла в електронній теорії пояснюється так. Введемо сталу Холла 7 Тоді 8 Отже згідно з формулою 8 вимірявши силу струму I у...
22831. ДВОПРОВІДНА ЛІНІЯ 95.5 KB
  В таких системах активний опір ємність і індуктивність розподілені рівномірно вздовж лінії. Як правило в двопровідних лініях умова квазістаціонарності виконується щодо відстані між провідниками а сила струму I лінійна густина заряду q і напруга між провідниками U суттєво змінюються вздовж лінії. Застосовуючи до нескінченно малої ділянки двопровідної лінії закон збереження електричного заряду і електромагнітної Індукції нехтуючи активним опором провідників можна отримати такі співвідношення: 1 2 Тут L С ...
22832. Ефект Пельтьє 70.5 KB
  Ефект Пельтьє. Дійсно експериментально така закономірність відома як ефект Пельтьє спостерігається. Встановлено що при проходженні електричного струму через контакт двох провідників напівпровідників виділяється чи поглинається в залежності від напрямку струму деяка кількість теплоти Qn пропорційна величині струму I та часу його протікання t: Qn=It 1 де  коефіцієнт Пельтьє. Ефект Пельтьє тим значніший чим більше відрізняються положення рівнів Фермі у напівпровідниках.
22833. РОЗШИРЕННЯ ШКАЛИ МІКРОАМПЕРМЕГРА ТА ВОЛЬТМЕТРА 73 KB
  Сила струму I обчислюється за формулою: 1 де Ca ціна поділки шкали мікроамперметра в амперах на поділку А под n відхилення стрілки у поділках шкали. Ціну поділки шкали мікроамперметра в одиницях напруги Cu можна обчислити за відомим внутрішнім опором мікроамперметра Rr та ціною поділки в одиницях сили струму Ca за формулою Cu=CaRr 2 При використанні мікроамперметра необхідно звертати увагу на такі характеристики як верхня та нижня межі значень вимірювання величин...
22834. РЕОСТАТ І ПОДІЛЬНИК НАПРУГИ 139.5 KB
  РЕОСТАТ І ПОДІЛЬНИК НАПРУГИ Реостат і подільник напруги це прилади що застосовуються для регулювання сили струму і напруги в електричних схемах. Спад напруги на опорінавантаженні а на реостаті напруга на опорінавантаженні змінюватиметься від до . Подільником напруги може правити реостат з трьома клемами який підключається до електричного кола так як зображено на мал. Переміщуючи точку вздовж подільника напруги можна одержати будьяку напругу від до 0.
22835. МЕТОД КОМПЕНСАЦІЇ В ЕЛЕКТРИЧНИХ ВИМІРЮВАННЯХ 232 KB
  МЕТОД КОМПЕНСАЦІЇ В ЕЛЕКТРИЧНИХ ВИМІРЮВАННЯХ Вимірювання електрорушійної сили джерела струму методом компенсації. джерела струму дорівнює різниці потенціалів на полюсах розімкненого елемента. Вимірювання термоелектрорушійної сили диференціальної термопари за допомогою потенціометра постійного струму. Принцип роботи потенціометра постійного струму такий.
22836. ЗАЛЕЖНІСТЬ ОПОРІВ МЕТАЛІВ ТА НАПІВПРОВІДНИКІВ ВІД ТЕМПЕРАТУРИ 76 KB
  ЗАЛЕЖНІСТЬ ОПОРІВ МЕТАЛІВ ТА НАПІВПРОВІДНИКІВ ВІД ТЕМПЕРАТУРИ При підвищенні температури металу його опір електричному струму зростає. Температурний коефіцієнт характеризує відносну зміну опору при зміні температури на один градус:. 1 Величина не є постійною вона залежить від температури. Для багатьох металів ця залежність може бути описана таким виразом: 2 де опір при температурі опір при температурі яку прийнято за точку початку відліку температури; постійні величини які залежать від роду металу і вони...