Скачиваний:
42
Добавлен:
01.05.2014
Размер:
21.89 Кб
Скачать

Библиотека AP для C++ Библиотека AP для C++ Этот документ описывает библиотеку AP, адаптированную для языка C++. Библиотека AP для С++ содержит базовый набор математических функций и классы-коллекции, которые требуются для работы программ с сайта "Библиотека алгоритмов".

Совместимость Данная библиотека должна быть совместима с любым компилятором С++.

Состав и использование В состав библиотеки входят файлы ap.h и ap.cpp. Для начала работы достаточно подключить файл ap.cpp к проекту.

Описание библиотеки AP Введение

Настройки условной компиляции

Константы

Функции

Класс ap_error

Классы массивов

Базовые подпрограммы линейной алгебры

Класс комплексных чисел

Введение В заголовочном файле ap.h определяется пространство имен ap. Следует учитывать, что имена функций, констант и классов, приведенные дальше, следует предварять префиксом ap::

Настройки условной компиляции AP_ASSERT

Этот символ включает проверку границ массива. Если он определен директивой define, то при каждом обращении к элементам динамического массива проверяется правильность переданного индекса. В случае ошибки генерируется исключение ap_error. Проверка границ массива делает программу более надежной, но замедляет работу.

NO_AP_ASSERT

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

UNSAFE_MEM_COPY

Директива define, определяющая этот символ, отключена. Не включайте её. Библиотека не содержит никакой документации по поводу этого символа.

Константы machineepsilon

Эта константа определяет точность машинных операций, т.е. минимальное число, такое, что 1+machineepsilon≠1 на данной разрядной сетке. Константа может быть взята "с запасом", т.е. реальная точность может быть ещё выше.

maxrealnumber

Эта константа определяет максимальное положительное вещественное число, представимое на данной машине. Константа может быть взята "с запасом", т.е. реальная граница может быть ещё выше.

minrealnumber

Эта константа определяет минимальное положительное вещественное число, представимое на данной машине. Константа может быть взята "с запасом", т.е. реальная граница может быть ещё ниже.

Функции int sign(double x)

Возвращает:

+1, если X>0

-1, если X<0

0, если X=0

double randomreal()

Возвращает случайное вещественное число в полуинтервале [0,1).

int randominteger(int maxv)

Возвращает случайное целое число в полуинтервале [0, maxv).

double round(double x)

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

double trunc(double x)

Отбрасывание дробной части X.

trunc(1.3) = 1

trunc(-1.3)= -1

double pi()

Возвращает константу π

double sqr(double x)

Возвращает x2

double maxreal(double m1, double m2)

Возвращает максимум из двух вещественных чисел.

double minreal(double m1, double m2)

Возвращает минимум из двух вещественных чисел.

int maxint(int m1, int m2)

Возвращает максимум из двух целых чисел.

int minint(int m1, int m2)

Возвращает минимум из двух целых чисел.

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

Классы массивов Работа с массивами Здесь мы рассмотрим общие принципы работы с классами-массивами, после чего будут рассмотрены сами классы и их методы.

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

Работа с массивом начинается с создания массива. Следует различать создание экземпляра класса-массива и выделение памяти под массив. При создании экземпляра класса может использоваться конструктор без параметров, создающий массив без элементов, а могут использоваться конструкторы копий и присваивания, копирующие один массив в другой. В случае создания массива конструктором по умолчанию он не содержит элементов и попытка обратиться к ним может привести к краху программы. Если при копировании массив-источник не имеет выделенной памяти под элементы, то массив-копия тоже не будет содержать элементов. Если массив-источник имеет выделенную под элементы память, то массив-копия выделяет тот же объем памяти и копирует в неё элементы. Т.е. при копировании получаются два полностью независимых массива с одинаковым содержимым.

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

Для обращения к элементам массива используется перегруженный operator(). Т.е. код, обращающийся к элементу массива a с индексами i, j, k будет выглядеть как a(i,j,k). Ниже приведен пример вычисления массива факториалов, иллюстрирующий работу с массивами.

integer_1d_array factarr(int n) { integer_1d_array result; result.setbounds(1,n); result(1) = 1; for(int i=2; i<=n; i++) result(i) = result(i-1)*i; return result; } Класс template_1d_array Это класс-шаблон динамического одномерного массива с переменными верхней и нижней границами. На основе этого класса получены следующие классы:

typedef template_1d_array<int> integer_1d_array; typedef template_1d_array<double> real_1d_array; typedef template_1d_array<bool> boolean_1d_array; typedef template_1d_array<complex> complex_1d_array; Функции-члены класса template_1d_array()

Конструктор. Создание пустого массива.

~template_1d_array()

Деструктор. При вызове освобождается выделенная под массив память

template_1d_array(const template_1d_array &rhs)

Конструктор копий массива. При этом выделяется отдельная область памяти, в которую копируется содержимое массива-источника.

const template_1d_array& operator=(const template_1d_array &rhs)

Присваивание массива. При этом содержимое массива-приемника удаляется и освобождается выделенная под него память, затем заново выделяется отдельная область памяти, в которую копируется содержимое массива-источника.

T& operator()(int i)

Обращение к элементу массива с номером i

void setbounds(int iLow, int iHigh)

Выделение памяти под массив. При этом старое содержимое массива удаляется и освобождается выделенная под него память, затем заново выделяется отдельная область памяти размера iHigh-iLow+1 элементов.

Нумерация элементов в новом массива начинается с iLow и заканчивается iHigh. Содержимое нового массива не определено.

void setcontent(int iLow, int iHigh, const T *pContent)

Метод аналогичен методу setbounds() за тем исключением, что после выделения памяти в неё копируется содержимое массива pContent[].

T* getcontent()

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

int getlowbound()

int gethighbound()

Методы используются для получения информации о нижней и верхней границах массива.

raw_vector<T> getvector(int iStart, int iEnd)

Метод используется базовыми подпрограммами линейной алгебры для получения доступа к внутренней памяти массива. Метод возвращает объект, содержащий в себе указатель на часть вектора (начиная с элемента с индексом iStart и заканчивая индексом iEnd). Если iEnd<iStart, то считается, что задан пустой вектор.

const_raw_vector<T> getvector(int iStart, int iEnd) const

Метод используется базовыми подпрограммами линейной алгебры для получения доступа к внутренней памяти массива. Метод возвращает объект, содержащий в себе указатель на часть вектора (начиная с элемента с индексом iStart и заканчивая индексом iEnd). Если iEnd<iStart, то считается, что задан пустой вектор. Возвращенный объект позволяет получать доступ только для чтения.

Класс template_2d_array Это класс-шаблон динамического двухмерного массива с переменными верхней и нижней границами. На основе этого класса получены следующие классы:

typedef template_2d_array<int> integer_2d_array; typedef template_2d_array<double> real_2d_array; typedef template_2d_array<bool> boolean_2d_array; typedef template_2d_array<complex> complex_2d_array; Функции-члены класса template_2d_array()

Конструктор. Создание пустого массива.

~template_2d_array()

Деструктор. При вызове освобождается выделенная под массив память

template_2d_array(const template_2d_array &rhs)

Конструктор копий массива. При этом выделяется отдельная область памяти, в которую копируется содержимое массива-источника

const template_2d_array& operator=(const template_2d_array &rhs)

Присваивание массива. При этом содержимое массива-приемника удаляется и освобождается выделенная под него память, затем заново выделяется отдельная область памяти, в которую копируется содержимое источника.

T& operator()(int i1, int i2)

Обращение к элементу массива с индексом [i1,i2]

void setbounds(int iLow1, int iHigh1, int iLow2, int iHigh2)

Выделение памяти под массив. При этом старое содержимое массива удаляется и освобождается выделенная под него память, затем заново выделяется отдельная область памяти размером (iHigh1-iLow1+1)*(iHigh2-iLow2+1) элементов.

Нумерация элементов в новом массива по первой размерности начинается с iLow1 и заканчивается iHigh1, аналогично для второй размерности.

Содержимое нового массива не определено.

void setcontent(int iLow1, int iHigh1, int iLow2, int iHigh2, const T *pContent)

Метод аналогичен методу setbounds() за тем исключением, что после выделения памяти в неё копируется содержимое массива pContent[].

Массив pContent содержит двухмерный массив, записанный построчно, т.е. первым идет элемент [iLow1, iLow2], затем [iLow1, iLow2+1] и т.д.

T* getcontent()

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

int getlowbound(int iBoundNum)

int gethighbound(int iBoundNum)

Методы используются для получения информации о нижней и верхней границах массива по размерности с переданным номером.

raw_vector getcolumn(int iColumn, int iRowStart, int iRowEnd)

const_raw_vector getcolumn(int iColumn, int iRowStart, int iRowEnd) const

Методы используются базовыми подпрограммами линейной алгебры для получения доступа к внутренней памяти массива. Методы возвращают объект, содержащий в себе указатель на часть столбца iColumn (начиная со строки iRowStart и заканчивая строкой iRowEnd).

Параметр iColumn должен быть допустимым номером столбца (т.е. находиться в пределах выделенной под массив памяти). Если iRowEnd<iRowStart, то считается, что задан пустой столбец.

raw_vector getrow(int iRow, int iColumnStart, int iColumnEnd)

const_raw_vector getrow(int iRow, int iColumnStart, int iColumnEnd) const

Методы используются базовыми подпрограммами линейной алгебры для получения доступа к внутренней памяти массива. Методы возвращают объект, содержащий в себе указатель на часть строки iRow (начиная со столбца iColumnStart и заканчивая столбцом iColumnEnd).

Параметр iRow должен быть допустимым номером строки (т.е. находиться в пределах выделенной под массив памяти). Если iColumnEnd<iColumnStart, то считается, что задана пустая строка.

Базовые подпрограммы линейной алгебры Базовые подпрограммы линейной алгебры библиотеки AP по своей функциональности близки к Level 1 BLAS, позволяя осуществлять простейшие операции с векторами, а также со строками и столбцами матриц.

Работа с подпрограммами осуществляется следующим образом. Сначала необходимо получить объект типа raw_vector или const_raw_vector, указывающий на обрабатываемую часть матрицы или массива при помощи методов getcolumn/getrow (для матрицы) или getvector (для массива). Объект содержит в себе указатель на начало строки (или столбца), число элементов в обрабатываемой строке и интервал между двумя соседними элементами. При использовании стандартной схемы хранения матриц в памяти (т.е. при хранении по строкам) интервал между элементами одной строки равен 1, а интервал между элементами одного столбца равен числу столбцов.

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

template<class T> T vdotproduct(const_raw_vector<T> v1, const_raw_vector<T> v2)

Подпрограмма вычисляет скалярное произведение переданных векторов.

template<class T> void vmove(raw_vector<T> vdst, const_raw_vector<T> vsrc)

template<class T> void vmoveneg(raw_vector<T> vdst, const_raw_vector<T> vsrc)

template<class T, class T2> void vmove(raw_vector<T> vdst, const_raw_vector<T> vsrc, T2 alpha)

Это семейство подпрограмм служит для различных видов копирования содержимого одного вектора на место другого вектора: простого копирования, копирования с умножением на -1, копирования с умножением на число.

template<class T> void vadd(raw_vector<T> vdst, const_raw_vector<T> vsrc)

template<class T, class T2> void vadd(raw_vector<T> vdst, const_raw_vector<T> vsrc, T2 alpha)

Это семейство подпрограмм служит для различных видов добавления одного вектора к другому: простого добавления или добавления с умножением на число.

template<class T> void vsub(raw_vector<T> vdst, const_raw_vector<T> vsrc)

template<class T, class T2> void vsub(raw_vector<T> vdst, const_raw_vector<T> vsrc, T2 alpha)

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

template<class T, class T2> void vmul(raw_vector<T> vdst, T2 alpha)

Эта подпрограмма служит для умножения вектора на число с сохранением результата в том же месте.

Класс комплексных чисел Библиотека AP содержит класс ap::complex, который позволяет осуществлять операции с комплексными числами. Доступ к действительной и мнимой частям комплексного числа осуществляется через открытые (public) поля x и y. Поддерживаются арифметические операции, как со встроенными типами данных, путем перегрузки операций сложения, вычитания, умножения и деления. Сложение, вычитание и умножение осуществляются обычным способом (т.е. по определению, которое можно найти в любом учебнике алгебры), операция деления осуществляется с использованием т.н. "безопасного" алгоритма, который никогда не приводит к переполнению при вычислении промежуточных результатов. Также библиотека включает в себя несколько функций, осуществляющих элементарные операции с комплексными числами.

const double abscomplex(const complex &z)

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

const complex conj(const complex &z)

Функция возвращает комплексное число, сопряженное своему аргументу.

const complex csqr(const complex &z)

Функция возвращает квадрат аргумента.

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