Програма для сканування каталогу на наявність відповідних файлів

Міністерство освіти і науки України

Житомирський державний технологічний університет

Лабораторна робота №2

з курсу «Системне програмування»

м. Житомир 2011 р.

Зміст

1. Розібрати роботу програми find.с

2. Проекспериментувати з функціями findfirst(), findnext(), змінюючи атрибути шуканих файлів

3.Вияснити призначення поля ff_reserved в структурі ffblk

4.Сформувати сьогоднішню дату та час в два байти, відповідно описаним правилам

5.Змінити програму так, щоб вона показувала лише

Контрольні запитання

Висновок

Список використаної літератури

1. Розібрати роботу програми find.с

Підключаємо стандартні бібліотеки:

#include <stdio.h>

#include <stdlib.h>

#include <dir.h>

#include <dos.h>

#include <conio.h>

Оголошуємо прототипи функцій:

void print_info( struct ffblk *find );

char *time_conv( unsigned time, char *char_buf );

char *date_conv( unsigned date, char *char_buf );

Головна функція main() отримує у якості аргументів цілочисельний аргумент argc який містить кількість аргументів у командному рядку та покажчик на масив покажчиків на рядки, де кожен вказує на певний аргумент командного рядка.

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

Оголошуємо змінну find типу ffblk, що є структурою визначеною у бібліотеці <dir.h> і включає наступні поля:

char ff_reserved[ 21 ]; /* зарезервовано для MS-DOS */;

char ff_attrib; /* атрибут з елементу директорія */

unsigned ff_ftime; /* поле часу з елементу директорія */

unsigned ff_fdate; /* поле дати з елементу директорія */

long ff_fsize; /* поле розміру файлу з елементу директорія */

char ff_name[ 13 ]; /*ASCIIZ- рядок з ім’ям і розширенням файлу */

struct ffblk find;

Викличемо функцію findfirst(), задавши їй у якості параметрів другий аргумент із командного рядка, посилання на структуру find та файловий атрибут.

Функція findfirst() повертає нуль у випадку успіху, тобто даний оператор if перевіряє чи виконалась функція із заданими параметрами.

if( !findfirst( argv[1], &find, FA_ARCH ) ) {

Якщо findfirst() виконалася правильно, то інформація буде занесена до структури find. Викличемо функцію print_info(), яка і надрукує структуру у відповідному форматі.

printf( "\n"

"\nFile Name Attr Date Time Size"

"\n------------ ---- ---------- -------- --------" );

print_info( &find );

}

У разі помилки findfirst() поверне -1 і виконається оператор else, який надрукує відповідну інформацію про правильність заповнення командного рядка і завершить виконання програми.

else {

printf( "\nUsage: FIND filename"

"\n Wildcards ""!"" and ""*"" is OK.\n" );

getch();

exit( -1 );

}

Функція findnext() продовжує пошук файлів, який розпочала функція findfirst(). Findnext() також повертає нуль у випадку успішного виконання. Тому за допомогою оператора while продовжуємо пошук до тих пір доки це можливо, тобто доки findnext () не поверне -1.

while( !findnext( &find ) )

print_info( &find );

Завершимо виконання програми.

getch();

exit( 0 );

}

Функція print_info() призначена для друку структури ffblk у відповідному зрозумілому для користувача форматі. Функція нічого не повертає. У якості парметрів приймає покажчик на структуру ffblk.

void print_info( struct ffblk *pfind ) {

Оголосимо два мив мольних масиви, які будуть зберігати відповідно час та дату у форматі рядка.

char timebuf[10], datebuf[12];

Викличемо date_conv() та time_conv() для конвертації полів дати і часу із цілочисельного формату до рядкового.

date_conv( pfind->ff_fdate, datebuf );

time_conv( pfind->ff_ftime, timebuf );

Надрукуємо поля структури ffblk.

printf( "\n%-12s ", pfind->ff_name);

printf( "0x%02X %8s %8s %8ld ",

pfind->ff_attrib, datebuf, timebuf, pfind->ff_fsize );

}

Функція time_conv() конвертує час представлений у вигляді одного беззнакового цілого у рядок. Функція приймає у якості параметрів число t, яке і є часом, та покажчик на рядок *buf, що і буде представленням дати у форматі рядка. Функція повертає покажчик на рядок.

char *time_conv( unsigned t, char *buf ) {

Перетворення відбувається наступним чином. Оголосимо цілочисельні змінні hour, minute, second.

int hour, minute, second;

ff_ftime є 16-бітовою структурою поділеною на бітові області, кожна з яких і відповідає годині, хвилині, та секунді. Числу годин відповідають з 11 по 15 біти, хвилин – з 5 по 10, і секунд поділених на 2 – з 0 по 4 відповідно. Таким чином, для того щоб виокремити кожен із елементів, зсунемо побітово число t (час) на відповідну кількість бітів і скористаємося оператором & (і).

hour = ( t >> 11 ) & 0x1F;

minute = ( t >> 5 ) & 0x3F;

second = ( t & 0x1F ) * 2;

За допомогою функції sprintf() пер направимо змінні hour, minute, second у відповідному форматі до масиву на який вказує buf.

sprintf( buf, "%2.2d:%02.2d:%02.2d", hour, minute, second);

return( buf );

}

Функція date_conv(), подібно функції time_conv(), переводить формат подачі дати із цілого числа в рядок. Дню відповідають з 0 по 4 біти, місяцю – з 5 по 8, а року, починаючи з 1980, – з 9 по 15 біти.

char *date_conv( unsigned d, char *buf ) {

int day, month, year;

day = d & 0x1f;

month = ( d >> 5 ) & 0x0f;

year = ( d >> 9 ) + 1980;

sprintf( buf, "%2.2d.%02.2d.%04.2d", day, month, year);

return( buf );

}

2. Проекспериментувати з функціями findfirst(), findnext(), змінюючи атрибути шуканих файлів

Функція findfirst() у якості одного із параметрів приймає атрибут відкриття файлу. Функція findnext() використовує ті ж атрибути, що визначені у функції findfirst(). Таким чином ми можемо встановити такі атрибути доступу:

    FA_RDONLY (Read-only attribute) – тільки для читання;

    FA_HIDDEN (Hidden file) – скриті файли;

    FA_SYSTEM (System file) – ситемні файли;

    FA_LABEL (Volume label) – мітка тому;

    FA_DIREC (Directory) – директорії;

    FA_ARCH (Archive) - архівні файли.

3. Вияснити призначення поля ff_reserved в структурі ffblk

В структурі ffblk є поле ff_reserved, яке призначене для використання системою MS-DOS. Поле має тип масив розмірністю 21 символ.

4. Сформувати сьогоднішню дату та час в два байти, відповідно описаним правилам

Нехай маємо дату - 31.12.2010

Під дату виділяється 2 байти

|15. . . . . . . . . . .9|8. . . . . .5|4. . . . . . . .0|

|_|_|_|_|_|_|_|_|_|_|_|_|_|_|_|_|

| Years -1980 | Month | Day |

day 31 = 11111

mon 12 = 1100

yaer 2010-1980 = 0011110

date = 0000 0000 0000 0000

    date = year

000000000 0011110

    date << 4

00000 0011110 0000

    date | mon

00000 0011110 1100

    date << 5

0011110 1100 00000

5. date | day

0011110 1100 11111 = 15775

Отже, 31.12.2010 = 15775.

Нехай маємо час - 16:59:35.

Під дату виділяється 2 байти

|15…….....11|10...............5|4…............0|

|_|_|_|_|_|_|_|_|_|_|_|_|_|_|_|_|

| Hours | Minutes | Seconds/2 |

H 16 = 10000

M 59 = 111011

S 35/2 = 10001

time = 00000 000000 00000

1.time = H

00000 000000 10000

2.time << 6

00000 10000 000000

3.time | M

00000 10000 111011

4.time << 5

10000 111011 00000

5.time | S

10000 111011 10001 = 34673

Отже, 16:59:35 = 34673.

програма сканування файл директорія

5. Змінити програму так, щоб вона показувала лише

    директорії;

if( !findfirst( argv[1], &find, FA_ DIREC ) )

    приховані файли;

if( !findfirst( argv[1], &find, FA_ HIDDEN ) )

    системні файли;

if( !findfirst( argv[1], &find, FA_ SYSTEM ) )

    директорії та системні файли.

if( !findfirst( argv[1], &find, FA_ SYSTEM || FA_ DIREC ) )

Контрольні запитання

    Як функція findnext() "дізнається", які файли потрібно шукати?

Функція findnext() шукає файли, параметри яких задані у функції findfirst().

    Чому час зміни файла записується з точністю до двох секунд?

У структурі ffblk виділене поле unsigned ff_ftime для представлення часу. У цьому полі числу секунд поділених на 2 (тому із точністю до двох секунд) виділно 5 бітів – з 0 по 4 відповідно, а це максимум число 31.

    Звідки береться інформація структури ffblk?

Якщо функція findfirst() знаходить файл, то структура ffblk заповнюється інформацією про каталог і файл

    Що таке елемент директорію?

Директорій – це спеціальний файл на диску, який складається з так званих елементів директорію. Розмір елементу в MS-DOS – 32 байта. Розмір кореневого директорія обмежений і залежить від формату диску. Наприклад, для диску 360К байт в кореневому директорії може розміститься тільки 112 елементів. Файл кореневого директорія розташується в фіксованому місці диску. Нижчі в ієрархії директорії називаються субдиректоріями вони можуть розташуватися в будь-якому місці диску і мати число елементів, обмежене лише фізичним об’ємом диску.

Формат елементу директорію MS-DOS

Висновок

Отже, на даній лабораторній роботі було проаналізовано програму для сканування каталогу на наявність відповідних файлів. У ході виконання було розібрано роботу функцій findfirst() та findnext(), а також будову та призначення структури ffblk.

У ході виконання лабораторній роботи, були здобуті навики роботи зі структурою ffblk, а саме з переведенням дати та часу у формат типу unsigned.

Список використаної літератури

    Касаткін А.І. Управление ресурсами. - Минск: Вышейшая школа, 1992.

    Касаткін А.І. Системное программирование. - Минск: Вышейшая школа, 1991.

    Власенко О.В., Данильченко О.М., Северин О.О. Системне прогрмамування. Курс лекцій. Частина 1. (бібліотека ЖІТІ)