SQLite 将一个数据库中的一行替换为另一个数据库中的一行

SQLite replace a row in one database from a row in another database

我有两个数据库文件,每个文件都有一个共同的 table 构成。当第二个数据库中的列设置了标志时,我想用第二个数据库中的类似行替换第一个数据库中的整个 table 行。我知道如何将一个数据库附加到另一个数据库,但我想不出正确的 UPDATE SET 指令。我正在寻找字符串指令以传递基于 kotlin 的 android studio android.database.sqlite execSQL 调用。

例如: 每个数据库中的 table 有五个名为 - name,address,date,value,flag 的列 我打开主数据库 (main.db) 并调用 ATTACH DATABASE 以附加辅助数据库 (secondary.db) 作为 'db2' 然后我想在主数据库中找到与辅助数据库中名称和地址字段匹配且辅助数据库中的标志字段包含值 1 的行相匹配的行。
然后我想用辅助数据库中的行替换主数据库行

我为 execSQL 尝试了以下字符串命令的变体,但没有成功:

val TABLE_NAME = "users"

val COL_NAME = "name"

val COL_ADR = "address"

val COL_DATE = "date"
    
val COL_VAL = "value"
    
val COL_FLG = "flag"

val db = this.writableDatabase

val attachDb = ("ATTACH DATABASE '$secondaryDbFile' AS 'db2'")            
db.execSQL(attachDb) 

val updateRows = ("UPDATE $TABLE_NAME SET ($COL_NAME, $COL_ADR, $COL_DATE, $COL_VAL, $COL_FLG) FROM db2.$TABLE_NAME WHERE $COL_FLG = 1")
db.execSQL(updateRows) 

我对 SQLite 的掌握不够好,无法构建正确的指令,请帮忙。

您在 UPDATE 语句中使用的语法错误。
您可以使用 ROW VALUES 并仅更新列 datevalueflag,因为其他两列必须匹配。
还使用 2 table 的别名(如 t1t2)来区分具有相同名称的列:

val db = this.writableDatabase
val attachDb = ("ATTACH DATABASE '$secondaryDbFile' AS db2") // no need for single quotes around db2
db.execSQL(attachDb)
val updateRows = "UPDATE $TABLE_NAME AS t1 " + 
                 "SET ($COL_DATE, $COL_VAL, $COL_FLG) = " +
                 "(SELECT t2.$COL_DATE, t2.$COL_VAL, 1 FROM db2.$TABLE_NAME AS t2 WHERE (t2.$COL_NAME, t2.$COL_ADR) = (t1.$COL_NAME, t1.$COL_ADR) AND t2.$COL_FLG = 1) " +
                 "WHERE EXISTS (SELECT 1 FROM db2.$TABLE_NAME AS t2 WHERE (t2.$COL_NAME, t2.$COL_ADR) = (t1.$COL_NAME, t1.$COL_ADR) AND t2.$COL_FLG = 1);"
db.execSQL(updateRows)

对于不允许更新 table 别名的 SQLite 版本,请使用此语句:

val updateRows = "UPDATE $TABLE_NAME " + 
                 "SET ($COL_DATE, $COL_VAL, $COL_FLG) = " +
                 "(SELECT t2.$COL_DATE, t2.$COL_VAL, 1 FROM db2.$TABLE_NAME AS t2 WHERE (t2.$COL_NAME, t2.$COL_ADR) = ($TABLE_NAME.$COL_NAME, $TABLE_NAME.$COL_ADR) AND t2.$COL_FLG = 1) " +
                 "WHERE EXISTS (SELECT 1 FROM db2.$TABLE_NAME AS t2 WHERE (t2.$COL_NAME, t2.$COL_ADR) = ($TABLE_NAME.$COL_NAME, $TABLE_NAME.$COL_ADR) AND t2.$COL_FLG = 1);"

如果SQLite的版本是3.33.0+,UPDATE语句的语法可以简化,但是在Android中是行不通的,目前在[=33=中] ] 31级支持3.32.2版本

这是UPDATE声明的简化版本(供以后的读者使用):

val updateRows = "UPDATE $TABLE_NAME AS t1 " + 
                 "SET ($COL_DATE, $COL_VAL, $COL_FLG) = (t2.$COL_DATE, t2.$COL_VAL, 1) " +
                 "FROM db2.$TABLE_NAME AS t2 WHERE (t2.$COL_NAME, t2.$COL_ADR) = (t1.$COL_NAME, t1.$COL_ADR) AND t2.$COL_FLG = 1;"