c++ - 提交对 SQLite 数据库的更改,这可以在 QTab 之间看到

标签 c++ database qt sqlite qsqldatabase

我有一个简单的应用程序,我可以在其中登录/注销用户。当用户登录时,应用程序会在主窗口(员工/管理员/客户)上显示相应的选项卡。我有一个 QMainWindowQTabWidget在上面。在我的 QMainWindow我创建了一个数据库(我为此实现了一个特殊的类):

class DataBase
{
public:

    DataBase();

    void initDatabase();
    void closeDatabase();

private:

    QSqlDatabase db;
};

DataBase::DataBase()
{
}

void DataBase::initDatabase()
{
    QString filename = "database.sql";
    QFile file(filename);

    db = QSqlDatabase::addDatabase("QSQLITE");
    db.setHostName("localhost");
    db.setDatabaseName(filename);

    // create users table
    if(this->db.open())
    {
        QSqlQuery usersTableQuery;
        QString usersTableQueryStr = "CREATE TABLE IF NOT EXISTS USERS (ID INTEGER PRIMARY KEY NOT NULL, "
                                 "LOGIN TEXT,"
                                 "PASSWORD TEXT,"
                                 "FIRSTNAME TEXT,"
                                 "LASTNAME TEXT,"
                                 "EMAIL TEXT,"
                                 "ACCOUNT_TYPE INTEGER" 
                                 ");";

        if(usersTableQuery.exec(usersTableQueryStr))
        {
            qDebug() << "Create USERS table OK";
        }
        else
        {
            qDebug() << usersTableQuery.lastError().text();
        }
    }
    else
    {
        qDebug() << "DB is not opened!\n";
    }

    // create service table
    if(this->db.open())
    {
        QSqlQuery serviceTableQuery;
        QString serviceTableQueryStr = "CREATE TABLE IF NOT EXISTS SERVICE (ID INTEGER PRIMARY KEY NOT NULL, "
                                 "NAME TEXT,"
                                 "PRICE REAL"
                                 ");";
        if(serviceTableQuery.exec(serviceTableQueryStr))
        {
            qDebug() << "Create SERVICE table OK";
        }
        else
        {
            qDebug() << serviceTableQuery.lastError().text();
        }
    }
    else
    {
        qDebug() << "DB is not opened!\n";
    }
}

void DataBase::closeDatabase()
{
    db.close();
}

我的员工、管理员、客户选项卡如下所示:

class AdminTab : public QWidget
{
    Q_OBJECT
public:
    explicit AdminTab(DataBase *db, QWidget *parent = 0);
//...

每个人(员工、客户、管理员)都可以在数据库中进行更改(例如,管理员可以插入服务,用户可以检查可用服务等)。但是,当管理员添加服务(我在打开的数据库上进行插入操作)并注销时,当客户端登录时,它看不到管理员所做的更改。当我再次启动应用程序并且客户端登录时,它可以看到新添加的服务。

添加服务如下所示:

bool DataBase::insertService(QString name, double price)
{
    if(!db.isOpen())
    {
        qDebug() << query.lastError();
        return false;
    }
    else
    {
        QSqlQuery query;
        query.prepare("INSERT INTO SERVICE (NAME, PRICE) "
                      "VALUES (:NAME, :PRICE)");
        query.bindValue(":NAME", name);
        query.bindValue(":PRICE", price);

        if(query.exec())
        {
            return true;
        }
        else
        {
            qDebug() << query.lastError();
        }
    }
    return false;
}

我猜这是数据库一直打开的问题,但是我怎样才能让更改在我插入/删除数据库后立即可用?我在创建 QMainWindow 时打开数据库并在其析构函数中关闭它。

每次需要使用数据库时,我都考虑过打开/关闭数据库,但我不能说这是否是一个好的解决方案。

甚至添加:

if(query.exec())
                {
                    query.clear();
                    db.commit();
                    return true;
                }

没有帮助。

客户有:QVector<Service*> availableServices;QComboBox *servicesComboBox; ,检查所有可用的服务,当客户端登录时:

void ClientTab::updateAllServices()
{
    availableServices.clear();
    availableServices = db->selectAllServices();

    servicesComboBox->clear();
    for(int i=0; i<availableServices.size(); i++)
        servicesComboBox->addItem(availableServices[i]->getServiceName(), QVariant::fromValue(availableServices[i]));

    servicesComboBox->setCurrentIndex(-1);
}

Service类:

#ifndef SERVICE_H
#define SERVICE_H

#include <QString>
#include <QMetaType>
#include <QVariant>

class Service : public QObject
{

   Q_OBJECT

public:
    Service(int id, QString name, double price);
    Service(){ id = -1; name = ""; price = 0;}

    QString getServiceName() const;
    void setServiceName(const QString &value);

    double getServicePrice() const;
    void setServicePrice(double value);

    int getId() const;
    void setId(int value);

private:

    QString name;
    double price;
    int id;
};

Q_DECLARE_METATYPE(Service*)

#endif // SERVICE_H

最后,从数据库中选择所有服务(我使用此方法填充 ClientTab 上的组合框):

QVector<Service*> DataBase::selectAllServices()
{
    QVector<Service*> services;

    if(!db.isOpen())
    {
        return services;
    }
    else
    {
        QSqlQuery query;
        if(query.exec("SELECT * FROM SERVICE;"))
        {
            while( query.next() )
            {
                int id = query.value(0).toInt();
                QString name = query.value(1).toString();
                double price = query.value(2).toDouble();

                Service *s = new Service(id, name, price);
                services.push_back(s);
            }
        }
        else
        {
            qDebug() << "DataBase::selectAllServices " << query.lastError();
        }
    }

    return services;
}

最佳答案

你能再检查一下吗

void ClientTab::updateAllServices()

每次客户端登录时调用(不仅在应用程序启动时)?

默认情况下,SQLite 数据库具有自动提交功能,因此您无需为这个简单的事情提交任何内容或使用任何事务。

你能看到通过 select * from service 在 sqlite 命令行中添加的新服务吗?如果是这样,则添加服务效果很好,您需要检查何时调用 updateAllServices()。

关于c++ - 提交对 SQLite 数据库的更改,这可以在 QTab 之间看到,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35524393/

相关文章:

c++ - 似乎一切正常,除了添加

java - 如何知道 Solr Optimize 何时完成?

python - 并发进程可以写入共享数据库吗?

c++ - 检查 char 是否是单引号。 C++

c++ - C++中动态数组的长度

sql-server - 关于复合索引和基数的策略

c++ - 是否有用于检测 USB 事件(插入和移除)的 Qt 解决方案?

javascript - 从 Qt 应用程序转换为 Web 应用程序的建议

c++ - 在 Qt 中调整子部件大小后调整大小

c++ - 使用 Xerces C++ 计算/访问 DOMNode 子节点的最快方法