![]() |
Пишу прогу на С.
в неё входит создание маленькой БД. использую arrays of records and files как я не крутила так и не нашла способа не задавать явное колличество записей в БД. т.е. что бы файл растягивался как резиновый по мере поплнения бд или удаления из неё записей. подумала о том чтобы за ранее обозначить бОльшее число записей в array, допустим сразу 1000, и пополнять по мере необходимости. но тогда в памяти и будет занимать место для всей 1000 даже если там всего 10 записей. слышала есть спец. функция, которая ограничивает занятость памяти для array на колличество заполненных ентри. т.е. даже если арей на 1000, а занято всего 10, то память будет занимать только под 10. подскажите что за функция? желательно с синтаксисом, плз. [s]Исправлено: Ginger, 16:25 25-01-2004[/s] |
нашла, кажется, выход с безразмерными structures:
Код:
#include <stdio.h> а это что такое - 1,f); ? [s]Исправлено: Ginger, 18:24 25-01-2004[/s] |
Ginger
Причём тут безразмерные структуры не ясно. Обычный RANDOM access к файлу. Если всё хранится в файле, то вполне подойдёт связка fseek/ftell. Если надо запись надо удалить, то просто помечаешь её как удалённую (т.к. сдвигать все последующие слишком долго). И изредка сжимаешь БД сдвигая все записи на места удалённых. Так работают некоторые dbm-базы. Правда, там ещё поле-ключ поддерживается, но это детали. Там просто балансируемое при доступе дерево с ключами в начале файла в простейшем случае сойдёт. На счёт fwrite: Цитата:
RANDOM д.б. написано маленькими буквами, а безразмерных структур вообще не существует. |
говоря о размере я имела ввиду функцию - sizeof()
о fwrite() спросила, потому что не знаю синтаксиса, а не потому что считаю, что она как-то на размер файла влияет и это всё не главное. мой главный вопрос в первом посте: Цитата:
Цитата:
|
Какое имеет отношение sizeof к безразмерности не ясно.
Начнём с простого вопроса: хранится вся информация в файле или в памяти? В оригинальном посте это не очевидно (мне во всяком случае). А так же, храниться должно всё именно как массив (т.е. в программе присутствует доступ по индексу) или как угодно/удобно? Если в файле, то см. предыдущий мой ответ. Т.е. надо курить доументацию поблочному вводу выводу, а так же ф-иям установки/чтения позиции в файле. Весь необходимый арсенал ф-ий в этом случае: fopen, fclose, fwrite, freed, fseek, ftell. Далее, если мы пользуем C++ и всё хранится в памяти, то вполне подойдёт stdd:vector/std::deque (в случае массива) или std::list (если присутствует только последовательный доступ к элементам). Если мы пользуем чистый C, то есть ф-ии malloc/free/realloc. В частности, интересна ф-ия realloc, которая изменяет размер памяти выделенной с помощью malloc. Ну и от связных списков и более нетривиальных структур, реализованных ручками тоже никто отказываться не собирается. В общем, если чуть поточнее будут поставлены условия, то любой из этих способов можно будет рассмотреть поподробнее. |
ivank
вобщем прога вычесляет коллории. при этом она хранит данные в нескольких файлах, которые представляют собой отдельные категории - мясо, хлеб, овощи, фрукты. в каждой категории хранятся данные (имя продукта и его каллорийность) неограниченного чилса продуктов. т.к. нужно чтобы юзер мог и добавить новый продукт в конкретную категорию и удалить, и править. при этом имя продукта - чар каллорийность - инт потому что каллорийность будет использоваться в вычеслениях. так вот я и думаю, как это дело организовать по умному. Цитата:
я тут порылась и придумала приблизительно такой выход: для каждой категории заводим отдельный файл имя продукта и его каллорийность записываем через строку, т.е: белый хлеб 226 чёрный хлеб 214 ... когда нам нужно мы находим (юзер будет вводить имя продукта, а прога находить его каллорийность и вычеслять что там надо) каллорийность переводим его из чар в инт (т.к. я не знаю как хранить каллорийность как инт): callor=(int)value_of_caloricity (вроде так можно) и делаем вычесления. конечно был ещё "выход": создать struct { char name[40]; int caloricity;} для каждой категории.. при этом записей должно быть какое-то колличество, т.е. arrays of records вот я с самого начала и спросила - есть ли функция ограничения занятости памяти до заполненного коллическва ентриес в аррей... но это ещё ладно, а вот как это дело запихать в файл и как потом от туда так же считать... я не знаю... [s]Исправлено: Ginger, 0:38 26-01-2004[/s] |
ну обьясните хотя бы как struct Vegetables { char name[40]; int caloricity;} записать в файл, а потом из файла преобразовать в struct при чтении? чтобы int caloricity можно было пользоваться при вычеслении.
|
Ginger
Если сойдёт бинарный вид, то fwrite(&data, sizeof(data), 1, out); В общем, неправильный, но наименее требовательный к знаниям способ. Данные хранятся в файле data.dat Добавление записи. Код:
out = fopen("data.fat", "a"); Код:
int n, i, desc; |
ivank
вобщем в твоём примере я мало чего поняла... но нашла хорошую книжку, где чуть ли не каждая буковка расталковывается, там прочитала о нужных мне функциях и вот что получилось (без malloc(), fseek(), ftell() etc...) Код:
#include <stdio.h> |
Ginger
В общем-то добавление и чтение сделаны как я и сказал :) А удаления нет вообще. Кстати, ещё более простой метод удаления, я о нём забыл в предыдущий раз сказать, хотя хотел: открываем файл с данными для чтения, и временный для записи. Затем в цикле копируем все записи кроме той, которую надо удалить, из исходного файла во временный. Закрываем оба файла, и таким же маром копируем всё из временного в файл с данными. Дёшево и сердито. Добавлено: Цитата:
|
ivank
Цитата:
Цитата:
нужно попробовать. Цитата:
отркыла файл, посмотрела - там и записан 1. значит читает не верно. короче покрутила, повертела не поняла, как сделать (примера ни одного не нашла) и сделал через float. :gigi: надо твой вариант попробовать! только выглядит как в object oriented... а как редактирование осуществить? Добавлено: вобщем не Цитата:
[s]Исправлено: Ginger, 2:22 29-01-2004[/s] |
[rus]Ya prosto ochepyatalsya, kogda kopiroval. Vernee nedoudalyal nemnogo. Koroche, kak vsegada oblazhalsya.
P.S. Na schet ne ponyala - est' u menya takoi porok, zaumnyami slovami pol'zovat'sya :([/rus] |
всё равно меня на мысль правильную наталкнул. ;)
сталкнулась с ещё одно проблемой: я сделала меню выбора всех каталогов по принципу меню вверху: Код:
while(!done) Код:
void add_del_dis(void) veget = fopen("G:/Studies/Project/programm/vegetables.dat", "w"); загвоздка в том что я хочу сначала из главного меню передать в функцию add_del_dis() название каталога соответствующее выбору юзера. т.е. когда юзер на главном меню выбирает Bread and Macarony, то в названии оно и пишеться: Bread and Macarony Catalogue - на самом деле это просто передача стрингового значения в функицю add_del_dis() при выборе каталога на главном меню, что-то типа: add_del_dis(char *catalogue_name) потом я хочу чтобы таким же образом в функиции write_info() и read_info(), тоже стринговым значением и тоже с главного меню передавался путь и имя открываемого файла. т.е. когда юзер выбирает Meat products в фукнцию write_info(), если она вызывается передавался: G:/Studies/Project/programm/meat.dat а не G:/Studies/Project/programm/bread_and_mac.dat только нам никто никогда не обьяснял как обозначать и передавать параметры из функции в функцию. как это сделать? [s]Исправлено: Ginger, 17:29 29-01-2004[/s] |
Код:
switch(c) Код:
void add_del_dis(char *datafile) Код:
void write_info(char *datafile) |
ivank
Цитата:
Цитата:
я же написала, что не знаю как их прописать вначале проги. у меня они прописаны как: #include <stdio.h> #include <conio.h> #include <ctype.h> #include <stdlib.h> #define FALSE 0 #define TRUE !FALSE void add_del_dis(void); *// но тут в скобках что-то другое должно быть void write_info(void); void read_info(void); struct caloricity_table { char name[30]; int caloricity; }tab; char c; int done=FALSE; // и к тому же стринги, которые передаю, тоже где-то как-то прописать надо. я не знаю синтаксиса. оказалось ничего сложного :) void add_del_dis(char *datafile); void write_info(char *datafile); void read_info(char *datafile); пасиб, ivank [s]Исправлено: Ginger, 21:39 29-01-2004[/s] |
ivank
написала функцию удаления записей: Код:
void delete_info(void) //function deleting form file и тут же выводит на экран перечень всего, что в каталоге (я что-то так и не поняла почему она не возвращается на главное меню) и то, что я просила удалить там же среди перечня. |
Сорри, забыл про жту тему.
Цитата:
Цитата:
Я бы посоветовал почитать книжку по C. Например, классику в виде Кернигана и Ритчи... |
ivank
кажись работает: Код:
void delete_info(void) //function deleting form file млин, эта scanf() какая-то тупая - если 2 раза подряд вызывать функцию, где эта присутствует, то scanf() игнорирует предидущие инпут ф-ии (типа gets() ) и проскакивает с имени (сринг) сразу на каллорийность(интеджер).. как с ней бороться? конечно, я взяла книжку. :) иначе как бы я сначала спрашивала ничего не зная, потом сама написала, что надо? просто сходила в библиотеку и взяла книгу: полтора дня ушло на чтение и 20 минут (с тестами) на написание того, что нужно просто там так и было написано - удаление из файла мы здесь рассматривать не будем, потому что это долго и сложно. |
Теперь всё вроде в порядке.
А на счёт нерабочести scanf - скорее всего твои косяки (т.к. тысячи людей её пользуются и ничего...). Маленьки пример, где scanf ведёт себя не так покажи - может смогу помочь, а может и нет, по тому как сам нередко нарывался на непонятные проблемы с ней. |
ivank
может и мои косяки, но я сомневаюсь... вот например функция добавления записи в файл - первый раз вызываеш её, чтобы добавить запись - всё проходит на ура. тут же хочешь добавить ешё одну запись - она проскакивает ввод имени и сразу просит ввести каллорийность. Код:
void write_info(void) { |
Внимательно читаем документацию на ф-ии scanf и gets. В частности, scanf после считывания числа, оставляет символа перехода на новую строку во входном потоке, если его специально не попросить его убрать. А ф-ия gets, как известно считывает до первого '\n', поэтому наткнувшись на этот символ, не убранный ещё scanf'ом она сразу завершает свою работу.
Должно быть так: scanf("%d\n",&veg.caloricity ); Если, число записано на отдельной строке, то его можно считать в строку (т.е. массив char), а затем sscanf'ом или atoi перевести в число. |
Цитата:
Цитата:
предложение понятно, функции почти понятны (sscanf - не совсем понятно как работает: sscanf (input_string, "%as %as", &str_arg1, &str_arg2); - но так же можно просто scanf("%s", &string) ), но к чему это? и спасибо за помощь и обьяснения :) [s]Исправлено: Ginger, 18:39 3-02-2004[/s] |
sscanf работает так же как и scanf, за тем исключением, что считывает не со стандартного потока ввода, а из строчки, которая передаётся первым параметром. К чему это было сказано: если не дружишь со scanf'ом, но дружишь с gets'ом. То считывай всё gets'ом в отдельные буферы, а затем уже из них sscanf'ом вычленяй данные как хочешь.
Т.е. вместо (псевдокод) Код:
sscanf("%d", &calor); Код:
char buf[256+1]; |
Цитата:
Код:
#include <stdio.h> Цитата:
может тему переименовать? - Создание бд с помощью structure и file в С к нам, конечно, никто не прикапывается, но тему мы не по названию обсуждаем. потом кому-нить пригодиться, в поиске найдёт... [s]Исправлено: Ginger, 15:08 4-02-2004[/s] |
Время: 11:28. |
Время: 11:28.
© OSzone.net 2001-