Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Qt 5 with SQLite: bindValue() results in "Parameter count mismatch" error

I'm doing a simple parameterized query with Qt 5.3.1 (64-bit) on Windows 7 using the SQLite driver. When I use bindValue() to set the value of the single parameter of my query, I systematically get the dreaded "Parameter count mismatch" error. Everything works fine when I use addBindValue(). Note that the code with bindValue() works fine with Qt 4.8.5 (64-bit).

Here is the full code (main.cpp):

#include <QtSql>

int main(int, char* [])
{
    auto db = QSqlDatabase::addDatabase("QSQLITE");
    db.setDatabaseName("db.sqlite");
    db.open();

    {
        QSqlQuery query("CREATE TABLE IF NOT EXISTS hashes (filepath TEXT, modified INTEGER, hash TEXT)", db);
        query.exec();
    }

    QSqlQuery query("SELECT modified FROM hashes WHERE filepath = :fp", db);
    query.bindValue(":fp", "test.jpg");

    if (!query.exec())
        qDebug() << query.lastError();

    db.close();

    return 0;
}

QtCreator project file (qtsqltest.pro):

QT += core sql
TARGET = qtsqltest
TEMPLATE = app
SOURCES += main.cpp

Program output on my machine:

QSqlError("", "Parameter count mismatch", "")

Any idea?

like image 651
François Beaune Avatar asked Jul 10 '14 19:07

François Beaune


2 Answers

I just found out that my question is a duplicate of this one and that the corresponding answer is correct.

It turns out that query strings passed to the constructor of QSqlQuery are executed immediately, as per the documentation.

It isn't clear why such queries are working fine with Qt 4.8.5, or why they do work with Qt 5.3.1 when using positional parameters (using ? placeholders) while they fail with named parameters (using : placeholders).

like image 130
François Beaune Avatar answered Oct 23 '22 07:10

François Beaune


I don't have the reputation to add a comment to the solution, so here, just to add yet another potential culprit: I was trying to bind a value to the column name and I got the Parameter count mismatch error, which seems to be the go-to error message no matter what the actual error is :)

It turns out this is simply an invalid approach to SQL, see QSqlQuery with prepare and bindValue for column name Sqlite.

So instead of

query.prepare("UPDATE connections SET (:columnName) = (:value) WHERE id = (:id)");
query.bindValue(":columnName", someColumnName);
query.bindValue(":value", someValue);
query.bindValue(":id", someId);

do this:

query.prepare(QString("UPDATE connections SET %1 = (:value) WHERE id = (:id)").arg(someColumnName));
query.bindValue(":value", someValue);
query.bindValue(":id", someId);
// etc.

This is not what the OP did but this thread is what came first during my search and it might save somebody else some time.

like image 5
Graprar Avatar answered Oct 23 '22 07:10

Graprar