21 ноября 2018 г. 8:07

Как использовать поиск "или" в QSqlTableModel для SQLite

QSqlTableModel, Qt, SQLite

Здравствуйте.

У меня есть поле со многими checkBox. При выборе нескольких  checkBox нужно реализовать поиск в SQLite c помощью QSqlTableModel::setSort()  так, чтобы выполнилось условие или тут нашло, или тут, или тут или везде.



Виртуальный хостинг со скидкой 10 процентов
Виртуальный хостинг со скидкой 10 процентов
EVILEG предлагает надёжный хостинг со скидкой 10% на виртуальный хостинг и 5% на VPS
30
  • 21 ноября 2018 г. 9:11
  • (ред.)
  • Ответ был помечен как решение.

Вот пример моего кода:

void FindWindow::on_lineEditFind_textChanged(const QString &arg1)
{
    if(arg1.length()>=3){
        filter =QString("ZN LIKE ('%%1%') OR FN LIKE ('%%1%')").arg(arg1);
    } else {
        filter="ZN LIKE ('') OR FN LIKE ('')";
    }
    modelRro->select();
    modelRro->setFilter(filter);
    ui->tableViewRezult->resizeColumnsToContents();
}

При изменении содержимого lineEdit происходит установка фильтра,

2

Спасибо.

В 4 строчке ZN и FN это названия столбцов?

0

Да это название столбцов

0

Что то у меня не работает.

Если делать так, то запрос выполняется

setFilter( "Gender LIKE ('мужчина') ")

а если так, то почему-то не выполняется условие или

setFilter("Name LIKE ('мужчина') OR Gender LIKE ('мужчина') OR PhoneNumber LIKE ('мужчина') ")


0

Тестируйте запросы в менеджере БД, а потом что все после WHERE используйте как аргумент  setFilter.

Не плохо бы было увидеть саму таблицу и задачу которая вам необходима для установки фильтра


0

Я проверял названия столбцов.

Вот код бд

PRAGMA foreign_keys = 0;


CREATE TABLE sqlitestudio_temp_table AS SELECT *

FROM SQLliteResumeArray;


DROP TABLE SQLliteResumeArray;


CREATE TABLE SQLliteResumeArray (

NumberID              INTEGER PRIMARY KEY AUTOINCREMENT

NOT NULL

UNIQUE,

Name                  TEXT,

Gender                TEXT,

Age                   INTEGER,

DateOfBirth           DATE,

PhoneNumber           TEXT,

Email                 TEXT,

City                  TEXT,

BusinessTrip          TEXT,

PostDesired           TEXT,

Salary                INTEGER,

CVPost                TEXT,

NameCompany           TEXT,

Experiance            TEXT,

Languages             TEXT,

AdditionalInformation TEXT,

Education             TEXT,

Skills                TEXT,

Comments              TEXT,

Resume                TEXT

);


INSERT INTO SQLliteResumeArray (

NumberID,

Name,

Gender,

Age,

DateOfBirth,

PhoneNumber,

Email,

City,

BusinessTrip,

PostDesired,

Salary,

CVPost,

NameCompany,

Experiance,

Languages,

AdditionalInformation,

Education,

Skills,

Comments,

Resume

)

SELECT NumberID,

Name,

Gender,

Age,

DateOfBirth,

PhoneNumber,

Email,

City,

BusinessTrip,

PostDesired,

Salary,

CVPost,

NameCompany,

Experiance,

Languages,

AdditionalInformation,

Education,

Skills,

Comments,

Resume

FROM sqlitestudio_temp_table;


DROP TABLE sqlitestudio_temp_table;


CREATE INDEX IndexResumeArray ON SQLliteResumeArray (

NumberID,

Name,

Age,

DateOfBirth,

PostDesired,

CVPost,

PhoneNumber,

Email,

City,

Salary,

NameCompany,

Education,

Languages,

Experiance,

Gender,

AdditionalInformation,

Comments,

Resume,

BusinessTrip,

Skills

);


PRAGMA foreign_keys = 1;



0

Сделал базу по вашей таблице

Запрос работает.

select * from SQLliteResumeArray
where Name LIKE ('мужчина') OR Gender LIKE ('мужчина') OR PhoneNumber LIKE ('мужчина') 

Проверьте в своем менеджере БД, возможно у вас база пустая.

во вложении SQL файл с вашей таблицей. Выполните импорт и пробуйте.



0

Спасибо.

Извините за беспокойство.

Просто запрос написал с маленькой буквы, а в бд он с большой.


0

Что бы избежать подобных трудностей я всегда тестирую запросы напрямую а потом уже использую их в Qt

2

Скажите пожалуйста, как добавить по столбу  с INTEGER поиск от int a; до int b; ?

0

разобрался, так

SearchTableModel->setFilter(" Age > 30 ");

    SearchTableModel->setFilter(" Age < 35 ");
0

Еще можно так

SearchTableModel->setFilter("Age BETWEEN 30 AND 35");

Или  так

SearchTableModel->setFilter("Age>=30 AND Age <=35");


0

Да, ваш вариант правильный, а мой не работал, т.к. оказалось QSqlTableModel запоминает только последний setFilter().

А можно ли сделать сначала один запрос setFilter, а потом искать уже в найденном еще одним setFilter ,и так дальше?

0

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

0

Нужно писать вложенные запросы. Или очень сложные фильтры конкретно для QSqlTableModel. Если хотите сохранить предыдущий запрос, то выполняйте это вручную в какую-нибудь переменную.

Ну или создавать временные таблицы в базе данных сырыми запросами, но это ещё сложнее будет.


0

На примере этой таблицы:

Сначала Age от 30 до 40

Потом Languages Languages LIKE ('Russian%')

Потом Languages City LIKE ('Москва%')

И так дальше. Просто количество фильтров не ограниченно и критерии поиска могут быть разнообразны.

С помощью вложенных запросов врятли получится так написать.

Если в переменную записывать, выйдет слишком громоздко и не оптимально будет идти поиск.

А как это сделать с помощью сложных фильтров QSqlTableModel?




0

Тогда использовать QSortFilterProxyModel

https://doc.qt.io/qt-5/qtwidgets-itemviews-customsortfiltermodel-example.html

Либо пересмотреть структуру БД, для оптимизации построения запросов.

0

Похоже что придется создавать временную таблицу.

Новую таблицу нужно создавать через QSqlQuery?

Не знаете, как новой таблице дать данные из setFilter()?



0

Пытаюсь создать временную таблицу так, но выдает ошибку QSqlError("1", "Unable to execute statement", "no such column: Name")

   QSqlQuery query;
        query.exec("CREATE TEMP TABLE TempTable1 AS SELECT "
           " NumberID           ," 
           "Name                ,"
           "Gender              ,"
           "Age                 ,"
           "DateOfBirth         ,"
           "PhoneNumber         ,"
           "Email               ,"
           "City                ,"
           "BusinessTrip        ,"
           "PostDesired         ,"
           "Salary              ,"
           "CVPost              ,"
           "NameCompany         ,"
           "Experiance          ,"
           "Languages           ,"
           "AdditionalInformation  ,"
           "Education           ,"
           "Skills              ,"
           "Comments            ,"
           "Resume               "
           ";");
        qDebug() << "exec :" << query.lastError();
0

Если у вас фильтрация таблицы из разных фильтров, то лучше всё-таки попытаться написать фильтр правильно, например

QString strFilter;

if (count > 30)
{
    strFilter.append("count_column > 30");
}
else if (/* ещё какое-нибудь условие*/)
{
    strFilter.append(" AND ");
    strFilter.append("second_column LIKE 'value'");
}

ну и так далее, главное правильно сформировать фильтрующую запись. Это будет формирование более сложного запроса, но это будет всё-таки легче, чем использовать временную таблицу.

0

Спасибо.

А можно ли сделать поиск не чувствительным к регистру

0

У вас база данных SQLite?

Как я понимаю, у неё нет встроенных механизмов, поэтому нужно извращаться, например так

select * from mytable where  UPPER(field) like 'AAA';

0

SQLite из коробки не поддерживает кириллицу в UPPER :-(

0

То есть, решение из разряда фантастики стандартными средствами для SQLite?

0
select
  t.test_text
, upper(
    (
        WITH RECURSIVE
        under_name(test_text, char, level) as 
            (select t.test_text, '', 0
            union
            select test_text, coalesce(lu.u,substr(test_text,level,1)), under_name.level+1
            from under_name
            left join (
                select 'А' as u, 'а' as l union select 'Б' as u, 'б' as l union select 'В' as u, 'в' as l union select 'Г' as u, 'г' as l union select 'Д' as u, 'д' as l union select 'Е' as u, 'е' as l union select 'Ё' as u, 'ё' as l union select 'Ж' as u, 'ж' as l union select 'З' as u, 'з' as l union select 'И' as u, 'и' as l union select 'Й' as u, 'й' as l union select 'К' as u, 'к' as l union select 'Л' as u, 'л' as l union select 'М' as u, 'м' as l union select 'Н' as u, 'н' as l union select 'О' as u, 'о' as l union select 'П' as u, 'п' as l union select 'Р' as u, 'р' as l union select 'С' as u, 'с' as l union select 'Т' as u, 'т' as l union select 'У' as u, 'у' as l union select 'Ф' as u, 'ф' as l union select 'Х' as u, 'х' as l union select 'Ц' as u, 'ц' as l union select 'Ч' as u, 'ч' as l union select 'Ш' as u, 'ш' as l union select 'Щ' as u, 'щ' as l union select 'Ь' as u, 'ь' as l union select 'Ы' as u, 'ы' as l union select 'Ъ' as u, 'ъ' as l union select 'Э' as u, 'э' as l union select 'Ю' as u, 'ю' as l union select 'Я' as u, 'я' as l
            ) lu on substr(test_text,level,1)=lu.l
            where level <= length(test_text)
        )
        select group_concat(char,'') from under_name
    )
) upper_text
from
(select 'Съешь ещё этих  мягких французских булок, да выпей же чаю. Test.' test_text) t
0

мне срочно нужно добавлять на сайт смайлики... фейспалма очень не хватает...

0

А как такой запрос отразится на скорости поиска?

И скажите пожалуйста, как это решение можно добавить в запрос

TableModel->setFilter(Name LIKE 'Вася');

0

я думаю, если заменить  SQLite на другую БД, например  postgresql, то можно будет стандартными средствами этот поиск сделать, без таких извращений. И работать будет быстрее точно. Но пока это у вас не является узким местом, то и не замрочивайтесь. Преждевременная оптимизация тоже не нужна.

0

А вы не знаете, как в  postgresql сделать автозаполнение столбца с ключом?

0

AUTOINCREMENT

bool DataBase::createTable()
{
    QSqlQuery query;
    if(!query.exec( "CREATE TABLE " TABLE " ("
                            "id INTEGER PRIMARY KEY AUTOINCREMENT, "
                            TABLE_DATE      " DATE            NOT NULL,"
                            TABLE_TIME      " TIME            NOT NULL,"
                            TABLE_RANDOM    " INTEGER         NOT NULL,"
                            TABLE_MESSAGE   " VARCHAR(255)    NOT NULL"
                        " )"
                    )){
        qDebug() << "DataBase: error of create " << TABLE;
        qDebug() << query.lastError().text();
        return false;
    } else {
        return true;
    }
    return false;
}

P/S/ этот ваш последний вопрос уже не имеет прямого отношения к фильтрам в таблице, создавайте, пожалуйста, новые темы для новых вопросов.

0

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
m
19 мая 2019 г. 1:49
mahhaki

Qt - Тест 001. Сигналы и слоты

  • Результат:78баллов,
  • Очки рейтинга2
S
17 мая 2019 г. 13:14
SunBro

Qt - Тест 001. Сигналы и слоты

  • Результат:42баллов,
  • Очки рейтинга-8
b
17 мая 2019 г. 4:18
banana

C++ - Тест 003. Условия и циклы

  • Результат:57баллов,
  • Очки рейтинга-2
Последние комментарии
P.
18 мая 2019 г. 14:03
PELMYACH .

Спасибо большое! Вскоре буду разбираться!
18 мая 2019 г. 9:13
Евгений Легоцкой

Добрый день! Отнимать значение общего счётчика можно в деструкторе класса кнопки QDynamicButton::~QDynamicButton(){ ResID--;} При этом я бы ещё переустанавливал значения вс...
P.
14 мая 2019 г. 22:33
PELMYACH .

Здравствуйте!А не подскажите, как можно при удалении какой либо кнопки, у щётчика отнять значение?Дабы например четвёртой кнопке соответствовал ID 4, а не 5 скажем
6 мая 2019 г. 6:39
Евгений Легоцкой

Добрый день. Этот урок для Qt Quick Control версии 1, Вы используете вторую версию. Здесь style уже нет, кастомизацию можно делать уже или черещ соответствующие property или через ...
U
4 мая 2019 г. 3:14
Unreal_man

Делаю вроде правильно, а ничего не получается. Что упустил? После button1. в выпадающем списке нет style.Да, и откуда в уроке взялся файл .pri и зачем он нужен?
Сейчас обсуждают на форуме
19 мая 2019 г. 12:45
Михаиллл

Скачал openssl-1.1.1 от сюда , но не понимаю что делать с этой папкой
19 мая 2019 г. 10:52
Евгений Легоцкой

Если честно, то мне нужно самому время потратить, чтобы глянуть это дело. Я использовал Flutter для разработки, а не Qt. Просто исходя из опыта, могу сказать, что по большей части всё на эмуля...
16 мая 2019 г. 23:08
BlinCT

Решил через indexOf сделать. Возвращает или номер позиции где нашел символ или строку или -1 если не найдено.
15 мая 2019 г. 15:06
Михаиллл

Спасибо , заработало.Получаю ответный сигнал.Но теоретически, в ответ на запрос должен прийти json файл. Скажите пожалуйста, как можно открыть ответные данные, прочитать их, и потом удалить...
14 мая 2019 г. 11:07
Евгений Легоцкой

Из той задачи, которую вы привели, у вас есть база данных и таблица в ней с текстами. Для представления данных из базы данных обычно используют QTableView, а text browser здесь не к мест...

Для зарегистрированных пользователей на сайте присутствует минимальное количество рекламы

EVILEG
О нас
Услуги
Присоединяйтесь к нам
© EVILEG 2015-2019
Рекомендует хостинг TIMEWEB