我在 Qt 上使用帶有 Spatialite 的 SQLite3 進行插入查詢,有時它只是無法回傳隨機語法錯誤。如果我在 SpatialiteGUI 上運行查詢,它永遠不會失敗。
我正在使用 SQLite3 版本 3.27.2。
構建和運行查詢的方法:
bool DatabaseManager::insertPolygons(QList<QList<QGeoCoordinate>> polygons, int workId)
{
sqlite3_stmt *dbStatement = nullptr;
QString strQuery = "INSERT INTO \"WORK_" QString::number(workId) "\" (Geometry) VALUES ";
bool inserted = false;
for(int i = 0; i < polygons.size(); i ) {
strQuery = "(GeomFromText('POLYGON((";
foreach (QGeoCoordinate coordinate, polygons[i]) {
strQuery = QString::number(coordinate.longitude(), 'f', 10) " "
QString::number(coordinate.latitude(), 'f', 10) ",";
}
strQuery = strQuery.left(strQuery.size() - 1) "))', 4326)),";
}
strQuery = strQuery.left(strQuery.size() - 1) ";";
char *query = strQuery.toLatin1().data();
int status = sqlite3_prepare_v2(database, query, strQuery.toLatin1().size(), &dbStatement, 0);
if(status == SQLITE_OK) {
int status = 0;
status = sqlite3_step(dbStatement);
if(status == SQLITE_DONE)
inserted = true;
}else
qDebug().noquote() << "status:" << status << "error:" << sqlite3_errmsg(database);
sqlite3_finalize(dbStatement);
return inserted;
}
一些查詢示例:
INSERT INTO "WORK_264" (Geometry) VALUES
(GeomFromText('POLYGON((-52.3855298461 -28.2283621371,-52.3855220463 -28.2283563298,-52.3855103297 -28.2283685464,-52.3855181295 -28.2283743537,-52.3855298461 -28.2283621371))', 4326)),
(GeomFromText('POLYGON((-52.3855454459 -28.2283737516,-52.3855376460 -28.2283679443,-52.3855259294 -28.2283801609,-52.3855337292 -28.2283859682,-52.3855454459 -28.2283737516))', 4326)),
(GeomFromText('POLYGON((-52.3855610456 -28.2283853661,-52.3855532457 -28.2283795588,-52.3855415291 -28.2283917755,-52.3855493289 -28.2283975828,-52.3855610456 -28.2283853661))', 4326)),
(GeomFromText('POLYGON((-52.3855766453 -28.2283969805,-52.3855688455 -28.2283911733,-52.3855571288 -28.2284033900,-52.3855649286 -28.2284091973,-52.3855766453 -28.2283969805))', 4326));
INSERT INTO "WORK_264" (Geometry) VALUES
(GeomFromText('POLYGON((-52.3868293314 -28.2269741900,-52.3868371280 -28.2269800006,-52.3868488522 -28.2269677737,-52.3868410531 -28.2269619658,-52.3868293314 -28.2269741900))', 4326)),
(GeomFromText('POLYGON((-52.3868137382 -28.2269625689,-52.3868215348 -28.2269683795,-52.3868332540 -28.2269561579,-52.3868254549 -28.2269503500,-52.3868137382 -28.2269625689))', 4326)),
(GeomFromText('POLYGON((-52.3867981450 -28.2269509478,-52.3868059416 -28.2269567584,-52.3868176557 -28.2269445420,-52.3868098566 -28.2269387341,-52.3867981450 -28.2269509478))', 4326)),
(GeomFromText('POLYGON((-52.3867825518 -28.2269393267,-52.3867903484 -28.2269451373,-52.3868020575 -28.2269329262,-52.3867942584 -28.2269271183,-52.3867825518 -28.2269393267))', 4326));
輸出:
status: 1 error: near "database": syntax error
status: 1 error: near "?": syntax error
uj5u.com熱心網友回復:
strQuery.toLatin1()是一個臨時值,并.data()在該值內抓取一個指標。這實際上是一個懸空指標。
添加一個中間保持變數:(并在使用時使用 UTF8 而不是 Latin1)
auto queryBA = strQuery.toUtf8();
int status = sqlite3_prepare_v2(database, queryBA.data(), queryBA.size(), &dbStatement, 0);
uj5u.com熱心網友回復:
為了解開這個謎團,我們所要做的就是通過閱讀 Qt 的檔案來仔細檢查發生了什么。
char *query = strQuery.toLatin1().data();
根據 Qt 的檔案,QString的toLatin1()方法回傳一個QByteArray物件。
接下來,我們確定QByteArray的data()方法回傳一個“只要位元組陣列未被重新分配或銷毀的指標 [that] 保持有效”。
好了,現在,問題很清楚了。toLatin1()回傳一個臨時QByteArray物件,按照 C 管理臨時物件的通常規則,該物件在運算式結束時被銷毀。在此之前data()回傳一個指向其內臟的指標,該指標保存在query.
但該指標立即成為指向已釋放記憶體的指標,任何進一步使用該指標都會導致未定義的行為。
這里最簡單的解決方法是獲取toLatin1() 的物件,并將其保存為區域變數,只有在data()不再需要它時才會銷毀。就像是:
QByteArray strarray = strQuery.toLatin1();
char *query=strarray.data();
轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/367208.html
上一篇:根據另一個單元格值設定約束
下一篇:如何使用SQLite3資料庫更好地格式化 1000個條目的Django過濾器查詢?(運算式樹太大(最大深度1000))
