Поиск:

Ответ в темуСоздание новой темы Создание опроса
> SQLite и С++, Давайте поработаем здесь 
:(
    Опции темы
Coocky
Дата 3.8.2005, 16:22 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Итак, скачал я 213 Кб файлов с расширением *.С и *.H
а так же билиотеку sqllite3.dll
Что делать дальше?


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
sergejzr
Дата 3.8.2005, 20:20 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Un salsero
Group Icon


Профиль
Группа: Админ
Сообщений: 13285
Регистрация: 10.2.2004
Где: Германия г .Ганновер

Репутация: 2
Всего: 360



 Да, много воды утекло с тех пор, как я последний раз брался за SQLite. Сорри, я конечно не мог знать, что они конкретно поменяли интерфейс. Но это не беда, конечно smile Я потихоньку начал разбираться заного. Всё уже работает, но только пока одним способом и пока без дллsmile

К сожалению в прошлый раз я дал тебе ссылочку на старую версию. Чтобы не топтаться на месте переходим на новейшую: http://www.sqlite.org/sqlite-source-3_2_2.zip

вот helloword'ик (работает, но будет ещё меняться)

Код

#include <stdio.h>
#include <windows.h>
/*
Конечно SQLite можно загрузить из DLL или же Link Lib(?).
*/
#define  SQLITEDLL
#ifdef SQLITEDLL
#define SQLITE_OK      0  /* Successful result */
#define SQLITE_ERROR    1  /* SQL error or missing database */
#define SQLITE_ROW     100 /* sqlite_step() has another row ready */
#define SQLITE_DONE    101 /* sqlite_step() has finished executing */
#define SQLITE_DENY   1   /* Abort the SQL statement with an error */
#define SQLITE_IGNORE 2   /* Don't allow access, but don't generate an error */
typedef int (*sqlite3_callback)(void*,int,char**, char**);
typedef struct sqlite3 sqlite3;
typedef struct sqlite3_stmt sqlite3_stmt;
typedef int (*sqlite3_close_dll)(sqlite3 *);
typedef int (__cdecl * sqlite3_open_dll)(
  const char *filename,   /* Database filename (UTF-8) */
  sqlite3 **ppDb          /* OUT: SQLite db handle */
);
typedef int (__cdecl * sqlite3_exec_dll)(
  sqlite3*,                     /* An open database */
  const char *sql,              /* SQL to be executed */
  sqlite3_callback,             /* Callback function */
  void *,                       /* 1st argument to callback function */
  char **errmsg                 /* Error msg written here */
);
typedef int (__cdecl * sqlite3_errcode_dll)(sqlite3 *db);
typedef const char * (__cdecl * sqlite3_errmsg_dll)(sqlite3*);
typedef int (__cdecl * sqlite3_prepare_dll)(
  sqlite3 *db,            /* Database handle */
  const char *zSql,       /* SQL statement, UTF-8 encoded */
  int nBytes,             /* Length of zSql in bytes. */
  sqlite3_stmt **ppStmt,  /* OUT: Statement handle */
  const char **pzTail     /* OUT: Pointer to unused portion of zSql */
);
typedef int (__cdecl * sqlite3_column_count_dll)(sqlite3_stmt *pStmt);
typedef const void * (__cdecl * sqlite3_column_name_dll)(sqlite3_stmt*,int);
typedef const char * (__cdecl * sqlite3_column_decltype_dll)(sqlite3_stmt*,int);
typedef const unsigned char * (__cdecl * sqlite3_column_text_dll)(sqlite3_stmt*,int);
typedef int (__cdecl * sqlite3_step_dll)(sqlite3_stmt*);
typedef int (__cdecl * sqlite3_finalize_dll)(sqlite3_stmt *pStmt);
sqlite3_exec_dll sqlite3_exec;
sqlite3_open_dll sqlite3_open;
sqlite3_close_dll    sqlite3_close;
sqlite3_errmsg_dll    sqlite3_errmsg;
sqlite3_prepare_dll   sqlite3_prepare;
sqlite3_step_dll   sqlite3_step;
sqlite3_finalize_dll sqlite3_finalize;
sqlite3_column_count_dll sqlite3_column_count;
sqlite3_column_name_dll sqlite3_column_name;
sqlite3_column_decltype_dll sqlite3_column_decltype;
sqlite3_column_text_dll sqlite3_column_text;
HINSTANCE    SQLiteDLL;
#else
#include "sqlite3.h"
#endif
int myCallback(void*,int,char**, char**);
int init();
void finish();
int main(int argc, char **argv){
  sqlite3 *db;
  char *zErrMsg = 0;
  int rc;
   if( init()!=0)
 {
 printf("DLL's not loaded ");
 return 1;
 }
  
  /*arg1: Моя_БД.db есть файл БД. Если его несуществует, SQLite сама его создаст.
   arg2: указатель на структуру нашего БД
../Где_Угодно/Моя_БД.db
*/
  //  argv[1]="test.bd";
 rc = sqlite3_open(argv[1], &db);
 /*
 Если было невозможно создать файлик то rc содержит номер ошибки.
*/
 if( rc ){
    fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(db));
    sqlite3_close(db);
    return 1;
  }
 /*
Всё в порядке, можно работать!
В SQLite существует два способа получения результата БД.
1. exec/callback
        Функция callback будет вызываться SQLite с каждой строкой от ResultSet'a
        Чем то похоже Listener на в JAVA
2. compile/step/finalize
        в этом случае можно забирать ResultSet строка за строкой
Пока не разобрался с новой версией :)
*/
 /*
рассмотрим sqlite_exec
arg1: Наша БД
arg2: SQL statement
arg3: Функция callback (Здесь нам не нужна).
arg4: Аргумент, который будет передаваться в функцию callback. Например вектор, в который мы хотим добавлять результат,
  или вообще указатель на какой нибудь обьект (Здесь нам не нужeн).
arg5: Сообщение об ошибке. Его можно распечатать, но только если возвращённый rc!=0.
Накидаем табличку
*/
 rc=sqlite3_exec(db, "create table test(num integer,name string);", NULL, NULL, &zErrMsg);
 /*
if(rc!=SQLITE_OK)
{
 printf( "%s\n", zErrMsg);
    return 1;
}
*/
 /*
 При импорте данных очень важно начать с begin иначе БД будет тормозить.
   Обьяснение: Все записи будут осуществляться в памяти и только с commit писаться на диск.
   После этого SQLite проверит, всё ли записано правильно.
   Без beginn'a это будет осуществляться при каждой записи
*/
 rc=sqlite3_exec(db, "begin;", NULL, NULL, &zErrMsg);
 for(int i=0;i<10;i++)
     {
 /*
 На оригинальность не претендую :)
*/
 char sqlText[200];
 sprintf(sqlText,"insert into test values(%i,'name_%i');",i,i);
 rc=sqlite3_exec(db, sqlText, NULL, NULL, &zErrMsg);
  if(rc!=SQLITE_OK)
      {
      printf( "%s\n", zErrMsg);
    break;
      }
    }
/*
Добавлено!!!!!!
Commit - база перенимает все изменения
*/
rc=sqlite3_exec(db, "commit;", NULL, NULL, &zErrMsg);
 /*
произведём селект способом 1:
*/
 printf("\n exec/callback \n");
 rc=sqlite3_exec(db, "select * from test;", myCallback, NULL, &zErrMsg);
 if(rc!=SQLITE_OK)
  {
      printf( "%s\n", zErrMsg);
      return 1;
     }
  
/*
  произведём SELECT способом 2:
*/
 const char *sqlText="select * from test;";
 /*
 ppVm непрозрачная структура виртуальной машины. В некоторых случаях я исползьовал несколько таких машин,
   чтобы опрашивать несколько разных ResultSet'ов.
*/
 sqlite3_stmt *ppVm;
 /*
 рассмотрим sqlite_compile
 arg1:Наша БД
 arg2:SQL statement
 arg3:Длинна запроса. Если -1, то SQLite сама определит её
 arg4:Виртуальная машинка.
 arg5:Остаток запроса после выполнения.
*/
 rc=sqlite3_prepare(db, sqlText,-1,&ppVm,&sqlText);
 if(rc!=SQLITE_OK)
       {
       printf( "%s\n", zErrMsg);
     return 1;
      }
    printf("\n compile/step/finalize \n");
    /*sqlite3_column_count отдаст количество колонок в результате*/
    int colcount=sqlite3_column_count(ppVm);
  /*
    рассмотрим sqlite_step
    arg1:Виртуальная машинка
    Она везде передаётся. Таким образом можно выполнять кучу запросов одновременно.
 */
 
 while(sqlite3_step(ppVm)!=SQLITE_DONE)
 {
  for(int i=0;i<colcount;i++)
  {
  
  printf("%s(%s)=%s |",sqlite3_column_name(ppVm,i), sqlite3_column_decltype(ppVm,i), sqlite3_column_text(ppVm,i));
  }
 printf("\n");
 }
 /*destructor Виртуальной машины*/
 rc=sqlite3_finalize(ppVm);
 
  sqlite3_close(db);
  finish();
  return 0;
}
static int myCallback(void *NotUsed, int columns, char **pazValue, char **pazColName){
 int i;
 for(i=0; i<columns; i++){
     /*
     Раньше был отступ в columns для типов данных..Сейчас вроде по другому, но как, пока не разобрался..
     */
  printf("%s(%s)=%s |",pazColName[i],pazColName[i+columns],pazValue[i] ? pazValue[i]:"NULL");
 }
 printf("\n");
finish();
 return 0;
}
int init()
{
#ifdef SQLITEDLL
 HINSTANCE    SQLiteDLL = LoadLibrary("sqlite3.dll");
     sqlite3_exec=((sqlite3_exec_dll) GetProcAddress (SQLiteDLL, "sqlite3_exec"));
     sqlite3_open=((sqlite3_open_dll) GetProcAddress (SQLiteDLL, "sqlite3_open"));
     sqlite3_step=((sqlite3_step_dll) GetProcAddress (SQLiteDLL, "sqlite3_step"));
     sqlite3_finalize=((sqlite3_finalize_dll) GetProcAddress (SQLiteDLL, "sqlite3_finalize"));
     sqlite3_close=((sqlite3_close_dll) GetProcAddress (SQLiteDLL, "sqlite3_close"));
    sqlite3_errmsg=((sqlite3_errmsg_dll) GetProcAddress (SQLiteDLL, "sqlite3_errmsg"));
    sqlite3_prepare=((sqlite3_prepare_dll) GetProcAddress (SQLiteDLL, "sqlite3_prepare"));
    sqlite3_column_count=((sqlite3_column_count_dll) GetProcAddress (SQLiteDLL, "sqlite3_column_count"));
    sqlite3_column_name=((sqlite3_column_name_dll) GetProcAddress (SQLiteDLL, "sqlite3_column_name"));
    sqlite3_column_decltype=((sqlite3_column_decltype_dll) GetProcAddress (SQLiteDLL, "sqlite3_column_decltype"));
    sqlite3_column_text=((sqlite3_column_text_dll) GetProcAddress (SQLiteDLL, "sqlite3_column_text"));
    
     return SQLiteDLL==NULL ? 1:0;
#endif
 return 0;
}
void finish()
{
#ifdef SQLITEDLL
 if(SQLiteDLL!=NULL)
 FreeLibrary(SQLiteDLL);
#endif
}

Теперь создаём проект. Самый быстрый способ:
Рспаковываем архив в папку.
Берём вышеописанный пример и сохраняем его в той же папке как (например) sergej.cpp
Открываем sergej.cpp в VC6 и просим закомпилить.
Компилер ругается на unresolved symbols. Открываем таб FileView и правой кнопкой мыши в меню выбираем "add Files to Project". Забираем ВСЕ *.с файлы кроме: shell.c и tclsqlite.c.

Компилируем.
Всё!

Как альтернативу, подсоеденю готовый проект. 

Присоединённый файл ( Кол-во скачиваний: 102 )
Присоединённый файл  sqlite_sergej.z.zip 656,63 Kb


--------------------
PM WWW IM ICQ Skype GTalk Jabber AOL YIM MSN   Вверх
sergejzr
Дата 3.8.2005, 20:55 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Un salsero
Group Icon


Профиль
Группа: Админ
Сообщений: 13285
Регистрация: 10.2.2004
Где: Германия г .Ганновер

Репутация: 2
Всего: 360



Вот тоже неплохой пример работы. Там так же описываются примеры работы вторым способом - через sqlite3_prepare() sqlite3_step()
http://www.adp-gmbh.ch/sqlite/bind_insert.html
Таким образом можно забирать результат строка за строкой.


--------------------
PM WWW IM ICQ Skype GTalk Jabber AOL YIM MSN   Вверх
Kurt
Дата 4.8.2005, 00:32 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Увлеченный
***


Профиль
Группа: Участник Клуба
Сообщений: 1662
Регистрация: 22.8.2003
Где: Краснодар

Репутация: нет
Всего: 36



Кстати, кто-то мне говорил, что в SQLite проблемы с удалением записей - приходилось копировать таблицу, убивать и заново воссоздавать.
В новых версиях SQLite эта проблема решена?

P.S. Сории за оффтоп, просто интересно очень..


--------------------
Для корабля, который не знает куда плыть, нет попутного ветра... ((С) Архимед)
...
Все знают, что это невозможно. Но случайно находится невежда, который этого не знает. Он-то и делает открытие.. ((С) А. Эйнштейн)
PM ICQ   Вверх
sergejzr
Дата 4.8.2005, 00:45 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Un salsero
Group Icon


Профиль
Группа: Админ
Сообщений: 13285
Регистрация: 10.2.2004
Где: Германия г .Ганновер

Репутация: 2
Всего: 360



Цитата(Kurt @ 3.8.2005, 23:32)
Кстати, кто-то мне говорил, что в SQLite проблемы с удалением записей - приходилось копировать таблицу, убивать и заново воссоздавать.
В новых версиях SQLite эта проблема решена?

smile Это в любой базе проблема. одно дело запись удалить из базы, а другое дело физически с диска. Просто "нормальные" базы находятся где то далеко на сервере и юзер не задумывается о том, что становится с удалёнными записями, а в локальной штуке конечно становится подозрительным, что при удалении записей размер не меняется. Это всё конечно обязательно для скорости. Чтобы не перелопачивать всю структуру, при удалении запись лишь помечается как удалённая.
Проблема сводится к вырезанию части контента из файла. Так что решить её однозначно никогда и нигде не удастся.
Хотим удалить физически, надо переписывать файлы БД чтобы не оставалось там "пустых" мест.
В новой версии есть "autovacuum", который эту процедуру будет повидимому автомвтически делать. Я к сож. первый раз за 2 года за SQLite взялся, поэтому точно говорить не буду. Раньше это делалось коммандой "vacuum"


--------------------
PM WWW IM ICQ Skype GTalk Jabber AOL YIM MSN   Вверх
Coocky
Дата 4.8.2005, 10:32 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Так,
sergej.z, начинаю пробовать.
Думаю эта тема хорошо разовьется.
Сейчас проблема сохранения данных без привязки к установки базы-очень актуальна.
Раньше пользовались файлами DBF, но сейчас на них бесплатных библиотек нет.А в Инете много встречается просьб.SQLite-лучшая тому альтернатива.
Думаю, если хорошо пойдет, тема будет большой, и можно FAQ смело делать.. smile


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 11:55 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



sergej.z
Твой пример работает.
Только вот в 7 студии я не могу так сделать!
Пишет
Код

c:\Documents and Settings\Robert Blake\My Documents\Visual Studio Projects\TestSQl\where.c(1378): fatal error C1010: unexpected end of file while looking for precompiled header directive


Итак со всеми файлами..


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
sergejzr
Дата 4.8.2005, 12:06 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Un salsero
Group Icon


Профиль
Группа: Админ
Сообщений: 13285
Регистрация: 10.2.2004
Где: Германия г .Ганновер

Репутация: 2
Всего: 360



ээээ седьмой у меня нет. И не будет smile. Сказать ничего не могу.
Это компилерские приколы. А может она вообще си не понимает?
Можно в либу скомпилить конечно.


--------------------
PM WWW IM ICQ Skype GTalk Jabber AOL YIM MSN   Вверх
Coocky
Дата 4.8.2005, 12:19 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



B в смысле библиотеку подключить sqlite3.dll?
А пример под либу можешь преписать? smile


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 12:55 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Так Сережа, работаем дальше!
Для тех , кто не понял, добавлякм файлы...
Project->Add Existing Item, и выбираю файлы с расширеним *.C, как писал sergej.z
А потом в свойствах проекта делаем
Цитата
C\C++ -> Precompiled Headers -> Not Using Precompiled Headers

Все работает.. smile
Итак, дальше..
Попробоввал я в коде сделать
Код

argv[1]=" Dima.db";

Появилась библиотека
Dima.dll
Я так понял это моя база?
Добавлено @ 12:58
Вообще,sergej.z,
может вкратце описать создание базы, открытие, выполнение всех запросов, закрытие smile


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
DENNN
Дата 4.8.2005, 13:03 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
****


Профиль
Группа: Участник Клуба
Сообщений: 3878
Регистрация: 27.3.2002
Где: Москва

Репутация: 1
Всего: 43



Цитата
unexpected end of file while looking for precompiled header directive

Может быть простой Rebuild All помог бы?
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 13:23 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



DENNN
Нет, это проблема в файлах, потому, что они на С написаны, а не на С++


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 14:07 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Итак давйте для начала возьмем три кита
sqlite3_open()-Открывает/создает базу данных

sqlite3_exec() -Выполняет запрос.Весь синтаксис у них описан...

sqlite3_close()-Собсвтенно закрывает базу..
Ну что,sergej.z, давай потренируемся на создании трех таблиц..
Я предлагаю это тебе, хотя буду пробывать сам smile


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 14:18 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Так, все нормально, теперь вопрос, как ходить по записям, кроме как
int myCallback(void*,int,char**, char**);



--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Coocky
Дата 4.8.2005, 16:49 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


GUI гуру
****


Профиль
Группа: Участник Клуба
Сообщений: 2879
Регистрация: 16.2.2004
Где: Украина. Запорожь е

Репутация: 2
Всего: 62



Первые проблемы..
Не отрабатывает функция в классе
Код

int ListBox(void*,int,char**, char**);


......
......
......
......
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{

char *zErrMsg = 0;
int i=sqlite3_exec(theApp.db, "SELECT URL FROM Project;",ListBox,&m_wndDlgBar,&zErrMsg);


return 0;
}

ListBox(void *NotUsed, int columns, char **pazValue, char **pazColName)
{
    CDialogBar*dlg=(CDialogBar*)NotUsed;
int i;
 for(i=0; i<columns; i++){
    CComboBox*cmb=(CComboBox*) dlg->GetDlgItem(IDC_COMBO1);
    cmb->AddString(pazValue[i]);
 }

return 0;
}




Все нормально i, возвращает ноль, а вот до функции дело не доходит...

Это сообщение отредактировал(а) Coocky - 4.8.2005, 16:51


--------------------
Верю в смерть после жизни, в любовь после секса ,в крем после бритья smile        
PM ICQ   Вверх
Ответ в темуСоздание новой темы Создание опроса
Правила форума "C++: Базы данных"
chipset

Данный форум предназначен для обсуждения вопросов прямым образом связанных с C++ и БД. Так, вопросы только по C++ следует задавать в C++:Общие вопросы а вопросы по абстрактным БД в Базах данных или в соответствующих под-форумах.

Благодарим за понимание.


Если Вам понравилась атмосфера форума, заходите к нам чаще! С уважением, chipset.

 
1 Пользователей читают эту тему (1 Гостей и 0 Скрытых Пользователей)
0 Пользователей:
« Предыдущая тема | C/C++: Базы данных | Следующая тема »


 




[ Время генерации скрипта: 0.0983 ]   [ Использовано запросов: 21 ]   [ GZIP включён ]


Реклама на сайте     Информационное спонсорство

 
По вопросам размещения рекламы пишите на vladimir(sobaka)vingrad.ru
Отказ от ответственности     Powered by Invision Power Board(R) 1.3 © 2003  IPS, Inc.