Чтение онлайн

ЖАНРЫ

Основы программирования в Linux
Шрифт:

 cdc_entry cdc_found;

 cdt_entry cdt_found;

 int track_no = 1;

 int first_time = 1;

 char *search_string = "";

 do {

cdc_found = search_cdc_entry(search_string, &first_time);

if (cdc_found.catalog[0]) {

cd_entries_found++;

track_no = 1;

do {

cdt_found = get_cdt_entry(cdc_found.catalog, track_no);

if (cdt_found.catalog[0]) {

track_entries_found++;

track_no++;

}

} while (cdt_found.catalog[0]);

}

 } while (cdc_found.catalog[0]);

 printf("Found %d CDs, with a total of %d tracks\n",

cd_entries_found, track_entries_found);

 (void)get_confirm("Press return");

}

20. Теперь

у вас есть утилита
display_cdc
для вывода элемента каталога:

static void display_cdc(const cdc_entry *cdc_to_show) {

 printf("Catalog: %s\n", cdc_to_show->catalog);

 printf("\ttitle: %s\n", cdc_to_show->title);

 printf("\ttype: %s\n", cdc_to_show->type);

 printf("\tartist: %s\n", cdc_to_show->artist);

}

и утилита

display_cdt
для отображения элемента-дорожки:

static void display_cdt(const cdt_entry *cdt_to_show) {

 printf("%d: %s\n", cdt_to_show->track_no,

cdt_to_show->track_txt);

}

21. Служебная функция

strip_return
удаляет завершающий строку символ перевода строки. Помните о том, что Linux, как и UNIX, использует один символ перевода строки для обозначения конца строки.

static void strip_return(char *string_to_strip) {

 int len;

 len = strlen(string_to_strip);

 if (string_to_strip[len - 1] == '\n')

 string_to_strip[len - 1] = '\0';

}

22. Функция

command_mode
предназначена для синтаксического анализа аргументов командной строки. Функция
getopt
— хороший способ убедиться в том, что ваша программа принимает аргументы, соответствующие стандартным соглашениям, принятым в системе Linux.

static int command_mode(int argc, char *argv[]) {

 int c;

 int result = EXIT_SUCCESS;

 char *prog_name = argv[0];

 /* Эти внешние переменные используются функцией getopt */

 extern char *optarg;

 extern optind, opterr, optopt;

 while ((c = getopt(argc, argv, ":i")) != -1) {

switch(c) {

case 'i':

if (!database_initialize(1)) {

result = EXIT_FAILURE;

fprintf(stderr, "Failed to initialize database\n");

}

break;

case ':':

case '?':

default:

fprintf(stderr, "Usage: %s [-i]\n", prog_name);

result = EXIT_FAILURE;

break;

} /* switch */

 } /* while */

 return(result);

}

Упражнение 7.16.
Файл cd_access.c

Теперь переходите к функциям доступа к базе данных dbm.

1. Как обычно, начните с нескольких файлов

#include
. Далее примените директивы
#define
для задания файлов, которые будут использоваться для хранения данных:

#define _XOPEN_SOURCE

#include <unistd.h>

#include <stdlib.h>

#include <stdio.h>

#include <fcntl.h>

#include <string.h>

#include <ndbm.h>

/* В некоторых дистрибутивах файл в предыдущей строке может быть придется заменить на gdbm-ndbm.h */

#include "cd_data.h"

#define CDC_FILE_BASE "cdc_data"

#define CDT_FILE_BASE "cdt_data"

#define CDC_FILE_DIR "cdc_data.dir"

#define CDC_FILE_PAG "cdc_data.pag"

#define CDT_FILE_DIR "cdt_data.dir"

#define CDT_FILE_PAG "cdt_data.pag"

2. Используйте эти две переменные области действия файла для отслеживания текущей базы данных:

static DBM *cdc_dbm_ptr = NULL;

static DBM *cdt_dbm_ptr = NULL;

3. По умолчанию функция

database_initialize
открывает существующую базу данных, но передав ненулевой (т.е. true) параметр
new_database
, вы можете заставить ее создать новую (пустую) базу данных, при этом существующая база данных удаляется. Если база данных успешно инициализирована, также инициализированы и два ее указателя, указывающие на то, что база данных открыта.

int database_initialize(const int new_database) {

 int open_mode = O_CREAT | O_RDWR;

 /* Если открыта какая-либо имеющаяся база данных, закрывает ее */

 if (cdc_dbm_ptr) dbm_close(cdc_dbm_ptr);

 if (cdt_dbm_ptr) dbm_close(cdt_dbm_ptr);

 if (new_database) {

/* Удаляет старые файлы */

(void)unlink(CDC_FILE_PAG);

(void)unlink(CDC_FILE_DIR);

Поделиться с друзьями: