无法在 qSqlite ( QT 5.9) 中加载 Spatialite 扩展
Could not load Spatialite extension in qSqlite ( QT 5.9)
我正在尝试将 Spatialite 作为扩展加载到 qSqlite(Qt 5.9)中,我之前用 Qt4.8 做过,但我在 QT5.9 上失败了。
我通过删除 "SQLITE_OMIT_LOAD_EXTENSION" 更改了 sqlite.pri,并通过删除“#define SQLITE_OMIT_LOAD_EXTENSION 1
”和添加“#define SQLITE_ENABLE_LOAD_EXTENSION 1
”对 sqlite.c 做了一些更改。
我还将以下行添加到 openDatabase(....)
#if defined(SQLITE_ENABLE_LOAD_EXTENSION)
| SQLITE_LoadExtension|SQLITE_LoadExtFunc
#endif
现在 "requet.setQuery("SELECT load_extension('spatialite')", dbProject);"功能被识别,但我收到此消息:
错误 "The specified procedure could not be found.\r\nUnable to fetch row"
如果我查看 MSVC14 中的调试输出,我可以看到 spatialite.dll 及其所有依赖项都已加载。
注意:我用我的 Spatialite 和我从他们的网站下载的 mod_spatialite 测试了这个。
关于这个问题有什么想法吗?
提前致谢。
基于示例here 我在sqlite 中启用了spatialite,该函数启用了该模块。为此,您必须 link sqlite3 库。
所做的修改是:
将"SELECT load_extension('libspatialite.so')"
改为"SELECT load_extension('mod_spatialite')"
将"SELECT InitSpatialMetadata()"
改为"SELECT InitSpatialMetadata(1)"
#include <sqlite3.h>
#include <QSqlDatabase>
#include <QSqlDriver>
#include <QSqlError>
int enable_spatialite(QSqlDatabase db){
QVariant v = db.driver()->handle();
if (v.isValid() && qstrcmp(v.typeName(), "sqlite3*")==0)
{
sqlite3_initialize();
sqlite3 *db_handle = *static_cast<sqlite3 **>(v.data());
if (db_handle != 0) {
sqlite3_enable_load_extension(db_handle, 1);
QSqlQuery query;
query.exec("SELECT load_extension('mod_spatialite')");
if (query.lastError() .isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << query.lastError().text()<<")";
return 0;
}
qDebug()<<"**** SpatiaLite loaded as an extension ***";
query.exec("SELECT InitSpatialMetadata(1)");
if (query.lastError() .isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << query.lastError().text()<<")";
return 0;
}
qDebug()<<"**** InitSpatialMetadata successful ***";
return 1;
}
}
return 0;
}
示例:
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName("memory.db");
if (!db.open()) {
qDebug()<<"not open";
}
qDebug()<<enable_spatialite(db);
QSqlQuery query;
qDebug()<<query.exec("CREATE TABLE test_geom (id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, name TEXT NOT NULL, measured_value DOUBLE NOT NULL);");
qDebug()<<query.exec("SELECT AddGeometryColumn('test_geom', 'the_geom', 4326, 'POINT', 'XY');");
for(int i=0; i< 10; i++){
QString q = QString("INSERT INTO test_geom(id, name, measured_value, the_geom) VALUES (%1,'point %2', %3, GeomFromText('POINT(1.01 2.02)', 4326))")
.arg("NULL").arg(i).arg(i);
query.prepare(q);
qDebug()<< i<<query.exec();
}
qDebug()<<query.exec("SELECT id, name, measured_value, AsText(the_geom), ST_GeometryType(the_geom), ST_Srid(the_geom) FROM test_geom");
while (query.next()) {
QString str;
for(int i=0; i < query.record().count(); i++)
str += query.value(i).toString() + " ";
qDebug()<<str;
}
return a.exec();
}
输出:
**** SpatiaLite loaded as an extension ***
**** InitSpatialMetadata successful ***
1
true
true
0 true
1 true
2 true
3 true
4 true
5 true
6 true
7 true
8 true
9 true
true
"1 point 0 0 POINT(1.01 2.02) POINT 4326 "
"2 point 1 1 POINT(1.01 2.02) POINT 4326 "
"3 point 2 2 POINT(1.01 2.02) POINT 4326 "
"4 point 3 3 POINT(1.01 2.02) POINT 4326 "
"5 point 4 4 POINT(1.01 2.02) POINT 4326 "
"6 point 5 5 POINT(1.01 2.02) POINT 4326 "
"7 point 6 6 POINT(1.01 2.02) POINT 4326 "
"8 point 7 7 POINT(1.01 2.02) POINT 4326 "
"9 point 8 8 POINT(1.01 2.02) POINT 4326 "
"10 point 9 9 POINT(1.01 2.02) POINT 4326 "
可以找到完整的示例 here。
此代码已在 linux Arch Linux 4.11.3-1-ARCH、Qt 5.8
上测试
感谢eyllanesc,感谢您的解释和建议,我找到了解决方案。
问题出在我编译的库(我仍然不知道为什么?),以及我下载的 mod_spatialite。当我们将它与 visual studio 一起使用时,最后一个需要替换 libstdc++_64-6.dll
,因为它会导致崩溃。
我的问题就出在这里,我用的那个不好,导致了The specified procedure could not be found
,所以我下载了x86_64-5.3.0-release-win32-seh-rt_v4-rev0
,然后用了libstdc++-6.dll
(我用 libgcc_s_seh-1.dll
将名称更改为 libstdc++_64-6.dll
)。我还把 libxml2-2.dll
改成了我之前编译的另一个
然后我使用了以下解决方案之一:
解决方案一:
我使用之前的代码进行修改,因为 ret = sqlite3_exec (db_handle, sql, NULL, NULL, &err_msg);
导致崩溃,而且我通过参数发送了数据库,因为它不像以前的代码那样工作。
所以我现在的工作代码是:
#include <QtCore/QCoreApplication>
#include <QtSql/QtSql>
#include "sqlite3.h"
int enable_spatialite(QSqlDatabase db) {
QVariant v = db.driver()->handle();
if (v.isValid() && qstrcmp(v.typeName(), "sqlite3*") == 0)
{
sqlite3_initialize();
sqlite3 *db_handle = *static_cast<sqlite3 **>(v.data());
if (db_handle != 0) {
sqlite3_enable_load_extension(db_handle, 1);
QSqlQueryModel sql;
sql.setQuery("SELECT load_extension('mod_spatialite')", db);
if (sql.lastError().isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << sql.lastError().text()<<")";
return 0;
}
else return 1;
}
}
return 0;
}
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE", "Project");
db.setDatabaseName("dbTest.db");
if (!db.open())
{
qDebug()<<"Critical"<< "Impossible to intialize the database !\n" + db.lastError().text();
return 0;
}
qDebug()<<enable_spatialite(db);
//just a test
QSqlQueryModel sql;
sql.setQuery("SELECT HEX(GeomFromText('POINT(10 20)'));", db);
qDebug() << sql.index(0, 0).data().toString();
return a.exec();
}
还有一件重要的事情是,我重新编译了 qsqlite 驱动程序以删除 SQLITE_OMIT_LOAD_EXTENSION
方案二:(直接方案)
打开 "Qt5.9.0.9\Src\qtbase\srcrdparty\sqlite\" 文件夹并更改 sqlite3.c 如下:
评论或删除 #define SQLITE_OMIT_LOAD_EXTENSION 1
添加:
#ifndef SQLITE_ENABLE_LOAD_EXTENSION
#define SQLITE_ENABLE_LOAD_EXTENSION 1
#endif
- 转到
static int openDatabase( const char *zFilename,sqlite3 **ppDb, unsigned int flags, const char *zVfs)
函数并将| SQLITE_LoadExtFunc
添加到SQLITE_ENABLE_LOAD_EXTENSION
如下:
#ifdef SQLITE_ENABLE_LOAD_EXTENSION
| SQLITE_LoadExtension | SQLITE_LoadExtFunc
#endif
在 Qt5.9 中使用您的编译器(在我的例子中是 nmake)再次编译您的插件。0.9\Src\qtbase\src\plugins\sqldrivers\sqlite\sqlite.pro
调用以下代码加载您的空间:
QSqlQueryModel sql;
sql.setQuery("SELECT load_extension('mod_spatialite')", db);
我正在尝试将 Spatialite 作为扩展加载到 qSqlite(Qt 5.9)中,我之前用 Qt4.8 做过,但我在 QT5.9 上失败了。
我通过删除 "SQLITE_OMIT_LOAD_EXTENSION" 更改了 sqlite.pri,并通过删除“#define SQLITE_OMIT_LOAD_EXTENSION 1
”和添加“#define SQLITE_ENABLE_LOAD_EXTENSION 1
”对 sqlite.c 做了一些更改。
我还将以下行添加到 openDatabase(....)
#if defined(SQLITE_ENABLE_LOAD_EXTENSION)
| SQLITE_LoadExtension|SQLITE_LoadExtFunc
#endif
现在 "requet.setQuery("SELECT load_extension('spatialite')", dbProject);"功能被识别,但我收到此消息: 错误 "The specified procedure could not be found.\r\nUnable to fetch row" 如果我查看 MSVC14 中的调试输出,我可以看到 spatialite.dll 及其所有依赖项都已加载。
注意:我用我的 Spatialite 和我从他们的网站下载的 mod_spatialite 测试了这个。
关于这个问题有什么想法吗? 提前致谢。
基于示例here 我在sqlite 中启用了spatialite,该函数启用了该模块。为此,您必须 link sqlite3 库。
所做的修改是:
将
"SELECT load_extension('libspatialite.so')"
改为"SELECT load_extension('mod_spatialite')"
将
"SELECT InitSpatialMetadata()"
改为"SELECT InitSpatialMetadata(1)"
#include <sqlite3.h>
#include <QSqlDatabase>
#include <QSqlDriver>
#include <QSqlError>
int enable_spatialite(QSqlDatabase db){
QVariant v = db.driver()->handle();
if (v.isValid() && qstrcmp(v.typeName(), "sqlite3*")==0)
{
sqlite3_initialize();
sqlite3 *db_handle = *static_cast<sqlite3 **>(v.data());
if (db_handle != 0) {
sqlite3_enable_load_extension(db_handle, 1);
QSqlQuery query;
query.exec("SELECT load_extension('mod_spatialite')");
if (query.lastError() .isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << query.lastError().text()<<")";
return 0;
}
qDebug()<<"**** SpatiaLite loaded as an extension ***";
query.exec("SELECT InitSpatialMetadata(1)");
if (query.lastError() .isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << query.lastError().text()<<")";
return 0;
}
qDebug()<<"**** InitSpatialMetadata successful ***";
return 1;
}
}
return 0;
}
示例:
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName("memory.db");
if (!db.open()) {
qDebug()<<"not open";
}
qDebug()<<enable_spatialite(db);
QSqlQuery query;
qDebug()<<query.exec("CREATE TABLE test_geom (id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, name TEXT NOT NULL, measured_value DOUBLE NOT NULL);");
qDebug()<<query.exec("SELECT AddGeometryColumn('test_geom', 'the_geom', 4326, 'POINT', 'XY');");
for(int i=0; i< 10; i++){
QString q = QString("INSERT INTO test_geom(id, name, measured_value, the_geom) VALUES (%1,'point %2', %3, GeomFromText('POINT(1.01 2.02)', 4326))")
.arg("NULL").arg(i).arg(i);
query.prepare(q);
qDebug()<< i<<query.exec();
}
qDebug()<<query.exec("SELECT id, name, measured_value, AsText(the_geom), ST_GeometryType(the_geom), ST_Srid(the_geom) FROM test_geom");
while (query.next()) {
QString str;
for(int i=0; i < query.record().count(); i++)
str += query.value(i).toString() + " ";
qDebug()<<str;
}
return a.exec();
}
输出:
**** SpatiaLite loaded as an extension ***
**** InitSpatialMetadata successful ***
1
true
true
0 true
1 true
2 true
3 true
4 true
5 true
6 true
7 true
8 true
9 true
true
"1 point 0 0 POINT(1.01 2.02) POINT 4326 "
"2 point 1 1 POINT(1.01 2.02) POINT 4326 "
"3 point 2 2 POINT(1.01 2.02) POINT 4326 "
"4 point 3 3 POINT(1.01 2.02) POINT 4326 "
"5 point 4 4 POINT(1.01 2.02) POINT 4326 "
"6 point 5 5 POINT(1.01 2.02) POINT 4326 "
"7 point 6 6 POINT(1.01 2.02) POINT 4326 "
"8 point 7 7 POINT(1.01 2.02) POINT 4326 "
"9 point 8 8 POINT(1.01 2.02) POINT 4326 "
"10 point 9 9 POINT(1.01 2.02) POINT 4326 "
可以找到完整的示例 here。
此代码已在 linux Arch Linux 4.11.3-1-ARCH、Qt 5.8
上测试感谢eyllanesc,感谢您的解释和建议,我找到了解决方案。
问题出在我编译的库(我仍然不知道为什么?),以及我下载的 mod_spatialite。当我们将它与 visual studio 一起使用时,最后一个需要替换 libstdc++_64-6.dll
,因为它会导致崩溃。
我的问题就出在这里,我用的那个不好,导致了The specified procedure could not be found
,所以我下载了x86_64-5.3.0-release-win32-seh-rt_v4-rev0
,然后用了libstdc++-6.dll
(我用 libgcc_s_seh-1.dll
将名称更改为 libstdc++_64-6.dll
)。我还把 libxml2-2.dll
改成了我之前编译的另一个
然后我使用了以下解决方案之一:
解决方案一:
我使用之前的代码进行修改,因为 ret = sqlite3_exec (db_handle, sql, NULL, NULL, &err_msg);
导致崩溃,而且我通过参数发送了数据库,因为它不像以前的代码那样工作。
所以我现在的工作代码是:
#include <QtCore/QCoreApplication>
#include <QtSql/QtSql>
#include "sqlite3.h"
int enable_spatialite(QSqlDatabase db) {
QVariant v = db.driver()->handle();
if (v.isValid() && qstrcmp(v.typeName(), "sqlite3*") == 0)
{
sqlite3_initialize();
sqlite3 *db_handle = *static_cast<sqlite3 **>(v.data());
if (db_handle != 0) {
sqlite3_enable_load_extension(db_handle, 1);
QSqlQueryModel sql;
sql.setQuery("SELECT load_extension('mod_spatialite')", db);
if (sql.lastError().isValid())
{
qDebug() << "Error: cannot load the Spatialite extension (" << sql.lastError().text()<<")";
return 0;
}
else return 1;
}
}
return 0;
}
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE", "Project");
db.setDatabaseName("dbTest.db");
if (!db.open())
{
qDebug()<<"Critical"<< "Impossible to intialize the database !\n" + db.lastError().text();
return 0;
}
qDebug()<<enable_spatialite(db);
//just a test
QSqlQueryModel sql;
sql.setQuery("SELECT HEX(GeomFromText('POINT(10 20)'));", db);
qDebug() << sql.index(0, 0).data().toString();
return a.exec();
}
还有一件重要的事情是,我重新编译了 qsqlite 驱动程序以删除 SQLITE_OMIT_LOAD_EXTENSION
方案二:(直接方案)
打开 "Qt5.9.0.9\Src\qtbase\srcrdparty\sqlite\" 文件夹并更改 sqlite3.c 如下:
评论或删除
#define SQLITE_OMIT_LOAD_EXTENSION 1
添加:
#ifndef SQLITE_ENABLE_LOAD_EXTENSION
#define SQLITE_ENABLE_LOAD_EXTENSION 1
#endif
- 转到
static int openDatabase( const char *zFilename,sqlite3 **ppDb, unsigned int flags, const char *zVfs)
函数并将| SQLITE_LoadExtFunc
添加到SQLITE_ENABLE_LOAD_EXTENSION
如下:
#ifdef SQLITE_ENABLE_LOAD_EXTENSION
| SQLITE_LoadExtension | SQLITE_LoadExtFunc
#endif
在 Qt5.9 中使用您的编译器(在我的例子中是 nmake)再次编译您的插件。0.9\Src\qtbase\src\plugins\sqldrivers\sqlite\sqlite.pro
调用以下代码加载您的空间:
QSqlQueryModel sql;
sql.setQuery("SELECT load_extension('mod_spatialite')", db);