Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:

LINTER от 04.02.15

.pdf
Скачиваний:
44
Добавлен:
21.05.2015
Размер:
2.16 Mб
Скачать

Практическое занятие 11. Основные приемы использования LinAPI 81

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

}else

printf("Table PRAC12_STAT deleted\n");

if (lRet = LINTER_ExecuteDirect(nCurs, "create table PRAC12_STAT (skurs int not null, scount int);", 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

else printf("Table PRAC12_STAT created\n");

for (i = 0; i < MAXKURSNUM; i++)

{

sprintf(szSQL, "insert into PRAC12_STAT values (%d, %d);\n", i + 1, statistic[i]);

printf(szSQL);

if (lRet = LINTER_ExecuteDirect(nCurs, szSQL, 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

}

printf("Cursor closing\n");

if (lRet = LINTER_CloseCursor(nCurs))

processing_error(lRet, 0, nCurs, 0,"Error CloseCursor");

printf("Connection closing\n");

if (lRet = LINTER_CloseConnect(nConn))

processing_error(lRet, nConn, 0, 0, "ERROR CloseConnect");

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

82 Практическое занятие 11. Основные приемы использования LinAPI

printf("LINTER_CloseAPI\n");

LINTER_CloseAPI(); /* Free library resource */

printf("Done.\n"); return 0;

/* End */

}

Скомпилируем этот файл:

gcc prac12_e3.c -I /export/home/lindesk/linter5923/linter/intlib/ -llinapi - lsocket -lm -L /export/home/lindesk/linter5923/linter/intlib/ -o prac12_e3

Задача 5. (Студенты выполняют самостоятельно) Модифицировать данный пример: в таблицу статистики добавляется столбец SGROUPS (количество групп на курсе) и формируется, как наибольшие номер группы у встретившихся студентов данного курса.

11.5. Фиксация и откат изменений

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

Тогда, алгоритм работы приложения будет таким:

 

соединяемся с базой данных в пессимистичном режиме;

 

создаем курсор;

 

выполняем обновление всех студентов – увеличение курса;

 

удаляем студентов с курсом, большим шести;

 

спрашиваем у пользователя подтверждение на фиксацию изменений;

 

фиксируем или откатываем изменения;

 

закрываем курсор и соединение, освобождаем ресурсы библиотеки.

Ниже представлен листинг программы из файла prac12_e4.c.

#include <stdio.h> #include <stdlib.h>

E-mail: market@relex.ru ЗАО НПП «РЕЛЭКС» http://www.relex.ru

Практическое занятие 11. Основные приемы использования LinAPI 83

#include "linapi.h"

 

#if defined(VXWORKS)

/* 05.03.02 */

#include "vxstart.h"

 

#endif

 

#ifdef _BCPP_

 

extern unsigned _stklen = 16383;

 

#endif

 

/* function for error processing for LinAPI */ void processing_error(LONG ret_cod,

WORD con_id, WORD cur_id, WORD stmt_id, char * message)

{

LONG lRet;

LONG apierr = 0, error = 0, syserr = 0;

if ( ret_cod == LINAPI_ERROR )

{

/* getting error codes for required object */ if (lRet = LINTER_Error(con_id, cur_id, stmt_id,

&apierr, &error, &syserr, NULL, NULL)) printf("diagnostic error %ld\n",lRet);

else

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

84 Практическое занятие 11. Основные приемы использования LinAPI

{

printf("ApiErr = %ld, LinErr = %ld, SysErr = %ld\n%s\n",

apierr, error, syserr, message);

if ( (apierr == eLinterError) && error > 2000 && error < 3000)

{

/* getting string and position if syntax error is */ printf("Syntax error : line %d, position %d\n",

(short)syserr, *(short*)((char*)&syserr +2));

}

}

LINTER_CloseAPI(); /* Free library resource */ exit(1);

}

else

printf("Return code = %ld\n",ret_cod);

}

#define MAXSTRLEN 100 #define MAXKURSNUM 6

 

#if defined(VXWORKS)

/* 05.03.02 */

 

 

 

MainStart(apidata, 32*1024, UninitLinterClient)

 

#else

 

 

 

 

 

 

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

 

 

Практическое занятие 11. Основные приемы использования LinAPI 85

int main(void)

#endif

{

WORD

nConn;

/* connection identifier */

WORD

nCurs;

/* cursor identifier

*/

LONG

lRet;

/* return code */

 

LONG realLen = 0;

LONG recCount = 0;

char szMask[MAXSTRLEN] = {0};

printf("\nConnect\n");

if (lRet = LINTER_Connect("SYSTEM", 0, "MANAGER", 0, NULL, mExclusive, &nConn))

processing_error(lRet, nConn, 0, 0, "ERROR Linter_Connect");

printf("Open cursor\n");

if (lRet = LINTER_OpenCursor(nConn, &nCurs, NULL, 0, mExclusive))

processing_error(lRet, nConn, 0, 0, "Error open cursor");

printf("Students updating...");

if (lRet = LINTER_ExecuteDirect(nCurs, "update PRAC12_STUDENT set SKURS = SKURS + 1;", 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

realLen = sizeof(recCount);

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

86 Практическое занятие 11. Основные приемы использования LinAPI

if (lRet = LINTER_GetCursorOption(nCurs, cRowCount, 0, &recCount, &realLen))

processing_error(lRet, 0, nCurs, 0,"Error GetCursorOption");

printf(" Done. %d records updated.\n", recCount);

printf("Students deleting...");

if (lRet = LINTER_ExecuteDirect(nCurs, "delete from PRAC12_STUDENT where SKURS > 6;", 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

realLen = sizeof(recCount);

if (lRet = LINTER_GetCursorOption(nCurs, cRowCount, 0, &recCount, &realLen))

processing_error(lRet, 0, nCurs, 0,"Error GetCursorOption");

printf(" Done. %d records deleted.\n", recCount);

printf("Do you want to commit changes? (Y/N)\n");

gets(szMask);

if (!strcmp(szMask, "Y") || !strcmp(szMask, "y"))

{

printf("Commit...");

if (lRet = LINTER_ExecuteDirect(nCurs, "commit;", 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

Практическое занятие 11. Основные приемы использования LinAPI 87

printf("Done\n"); } else

{

printf("Rollback...");

if (lRet = LINTER_ExecuteDirect(nCurs, "rollback;", 0, NULL,NULL))

processing_error(lRet, 0, nCurs, 0,"Error ExecuteDirect");

printf("Done\n");

}

printf("Cursor closing\n");

if (lRet = LINTER_CloseCursor(nCurs)) processing_error(lRet, 0, nCurs, 0,"Error CloseCursor"); printf("Connection closing\n");

if (lRet = LINTER_CloseConnect(nConn))

processing_error(lRet, nConn, 0, 0, "ERROR CloseConnect");

printf("LINTER_CloseAPI\n");

LINTER_CloseAPI(); /* Free library resource */

printf("Done.\n"); return 0;

/* End */

}

Скомпилируем этот файл:

gcc prac12_e4.c -I /export/home/lindesk/linter5923/linter/intlib/ -llinapi -

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

88 Практическое занятие 11. Основные приемы использования LinAPI

lsocket -lm -L /export/home/lindesk/linter5923/linter/intlib/ -o prac12_e4

Задача 6. (Студенты выполняют самостоятельно) Модифицировать данный пример: в таблицу студентов добавляется столбец SFORM (форма обучения – бакалавр, специалист или магистр). Модифицировать процесс удаления студентов из таблицы по окончанию вуза, учитывая, что время обучения бакалавра – 4 года, специалиста – 5 лет и магистра – 6.

Выпускников заносить в таблицу PRAC12_VYPUSK (ФИО, номер группы, год выпуска, квалификация).

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

89

Практическое занятие 12

Пример разработки приложения для ЛИНТЕР с использованием Qt

Практика 8 часов

Целью данной практики является освоение техники построения графического приложения, работающего с базой данных ЛИНТЕР в среде Unix, при использовании графической библиотеки Qt версии 1.4.x или выше (необходимы общие знания по Qt).

Вданном практике рассматривается синхронная и асинхронная работа с ЛИНТЕР и прием сообщений от ЛИНТЕР через асинхронный запрос. Работа с ЛИНТЕР осуществляется через Call-интерфейс (необходимы знания из предыдущих занятий по Callинтерфейсу). Используются вызовы Call-интерфейса, уже изученные нами, и никакой особой специфики здесь нет. Аналогичным образом можно было бы использовать и LinAPI. Специфичность нашей практики заключается в использовании классов C++, обеспечивающих работу с ЛИНТЕР, в частности, при построении приложений с графическим интерфейсом пользователя на основе библиотеки Qt.

Впримере используется таблица AUTO со следующей структурой:

create table auto(

 

MAKE

char(20),

MODEL

char(20),

BODYTYPE

char(15),

CYLNDERS

integer,

HORSEPWR

integer,

DSPLCMNT

integer,

WEIGHT

integer,

COLOR

char(10),

YEAR

integer,

SERIALNO

char(16),

CHKDATE

integer,

CHKMILE

integer,

PERSONID

integer primary key);

Эта таблица входит в демонстрационную БД СУБД ЛИНТЕР и ее всегда можно создать и загрузить из файлов, входящих в дистрибутив ЛИНТЕР. Для работы примеров в базе данных, после ее старта, должно быть создано событие AUTO_CHANGE:

create event auto_change as delete, update, insert on auto;

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

однострочных

элементах

(однострочный

редактор).

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

Практическое занятие 12

90 Пример разработки приложения для ЛИНТЕР с использованием Qt

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

Сборка приложения осуществляется с помощью команды 'make' в результате выполнения которой создается исполняемый файл 'main'. Перед сборкой должны быть изменены значения переменных LINTER и QTDIR на те, которые соответствуют реальности. Исходные файлы и сценарии сборки для каждого из примеров находятся в подкаталогах step1, step2, step3.

На первом этапе - демонстрируется синхронная работа с ЛИНТЕР. В приложении используется многострочный элемент - список для вывода результата выборки и 4 однострочных редактора, для вывода значений полей каждой записи

После запуска программы список будет заполнен значениями поля AUTO.SERIALNO, а в однострочных редакторах значениями полей MAKE, MODEL,YEAR, SERIALNO текущей записи выборки.

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

На третьем этапе к функциональности реализованной на первых этапах добавляется возможность асинхронного выполнения select-запроса. Программа не ожидает ответа на запрос, а продолжает работать, и только после наступления события о готовности источника данных элементы управления заполняются данными. Внешне программа не отличается от предыдущих, только вместо списка используется таблица для отображения результатов выборки. Для того чтобы была видна работа программы, при изменении состояния источника данных выводятся информационные окна. SQL-запрос к Линтеру написан специально с перемножением таблиц, чтобы была видна пауза между началом выполнения запроса и его концом.

Далее опишем более подробно устройство классов и принципы по которым были написаны приложения на каждом из этапов.

12.1. Первый этап

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

CDataSource - интерфейсный класс источник данных реляционного типа. Его

основные особенности:

 

 

 

 

 

он осуществляет технологию работы с курсором через методы

 

 

getCurRow , setCurRow, rowCount;

 

 

 

 

он может находиться в двух состояниях: активен и неактивен; метод

 

 

state может возвратить Empty/Ready;

 

 

 

 

источник данных активизируется методом retrieve, а деактивизируется

 

 

методом destroy или при возникновении ошибки;

 

 

 

 

табличные значение можно получить вызовом методов

 

 

getNull,

 

 

 

 

 

getField,

 

 

 

 

 

getFieldAsChar

(всегда конвертирует значение в строку);

 

 

 

 

 

 

 

 

E-mail: market@relex.ru

ЗАО НПП «РЕЛЭКС»

http://www.relex.ru

 

 

 

 

 

 

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]