- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
在我的 Android 项目中,我有一个使用了一段时间的表。现在我想删除一列“image_byte”并添加一个新列“image_path”。我不能只是在使用应用程序时重命名该列,并且需要删除 image_byte 中的文本,以便不再使用它。
我实现了SQLiteOpenHelper,以下代码位于onUpgrade中:
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion){
//check if upgrade is really needed
if(newVersion > oldVersion) {
tableSQL = "CREATE TABLE IF NOT EXISTS table1 ( id INTEGER PRIMARY KEY, name TEXT, image_path TEXT)";
db.execSQL(tableSQL);
//get all the current tableName table columns
List<String> columns = getColumns(db, "table1");
//rename the old table
db.execSQL("ALTER TABLE table1 RENAME TO temp_table1;");
//create the new tableName table
db.execSQL(tableSQL);
//merge the old and new columns, keep all the old ones
columns.retainAll(getColumns(db, "table1"));
String cols = joinColumnNames(columns, ",");
//insert the old data in the new table
db.execSQL(String.format("INSERT INTO %s (%s) SELECT %s FROM temp_%s", "table1", cols, cols, "table1"));
//delete the old temp table
db.execSQL("DROP table 'temp_table1'");
}
}
以及 getColumns 函数:
private List<String> getColumns(SQLiteDatabase db, String tableName){
List<String> ar = null;
Cursor c = null;
try{
c = db.rawQuery("SELECT * FROM " + tableName + " limit 1", null);
if(c != null){
ar = new ArrayList< >(Arrays.asList(c.getColumnNames()));
}
} catch (Exception e){
} finally {
if (c != null){
c.close();
}
}
return ar;
}
但是在调试应用程序时,它说列“image_byte”不存在,并且无法将内容从 temp_table1 复制到 table1。这是正确的,因为它现在是“image_path”并且不应该复制 image_byte。但为什么 getColumns 返回旧列呢?因为当我使用 Log.d 时,它显示“id”、“name”、“image_byte”。因此,retainAll 函数不会删除“image_byte”。我似乎无法弄清楚为什么它仍然返回 image_byte。这可能是愚蠢和简单的,但我就是不明白。
最佳答案
我认为该问题很可能是由于 columns.retainAll(getColumns(db, "table1"));
中的缓存造成的似乎是从原始 table1 中检索列。
这是我进行的一些测试的结果:-
10-15 13:37:19.319 26740-26740/? D/DBINFO: onUpgrade Started.
10-15 13:37:19.320 26740-26740/? D/DBINFO: Databse info in onUpgrade after table rename to temp_table1
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: DatabaseList Row 1 Name=main File=/data/data/mjt.sqlitedbexamples/databases/imagestore
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Database Version = 1
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table Name = android_metadata Created Using = CREATE TABLE android_metadata (locale TEXT)
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table = android_metadata ColumnName = locale ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table Name = temp_table1 Created Using = CREATE TABLE "temp_table1"(id INTEGER PRIMARY KEY,name TEXT,image_byte TEXT)
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.320 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = image_byte ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/DBINFO: Databse info in onUpgrade after creation of new table1.
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: DatabaseList Row 1 Name=main File=/data/data/mjt.sqlitedbexamples/databases/imagestore
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Database Version = 1
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table Name = android_metadata Created Using = CREATE TABLE android_metadata (locale TEXT)
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = android_metadata ColumnName = locale ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table Name = temp_table1 Created Using = CREATE TABLE "temp_table1"(id INTEGER PRIMARY KEY,name TEXT,image_byte TEXT)
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = temp_table1 ColumnName = image_byte ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table Name = table1 Created Using = CREATE TABLE table1(id INTEGER PRIMARY KEY, name TEXT,image_path TEXT)
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/SQLITE_CSU: Table = table1 ColumnName = image_path ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 13:37:19.321 26740-26740/? D/ORIG COLUMNS: Orignal Column Info - Column=id Column=name Column=image_byte
10-15 13:37:19.321 26740-26740/? D/NEWTAB COLUMNS: Columns from new table1 - Column=id Column=name Column=image_byte
10-15 13:37:19.321 26740-26740/? D/RETAIN COLUMNS: Columns After Retain - Column=id Column=name Column=image_byte
10-15 13:37:19.321 26740-26740/? D/JOINED COLUMNS: id,name,image_byte
10-15 13:37:19.321 26740-26740/? E/SQLiteLog: (1) table table1 has no column named image_byte
10-15 13:37:19.329 26740-26740/? D/AndroidRuntime: Shutting down VM
最后几行 D/ORIG COLUMNS、D/NEWTAB COLUMNS、D/RETAIN COLUMNS。 D/RETAIN COLUMNS 表示 image_byte 已被提取。
也许引入中间表名称,然后最终将其重命名为原始表名称可能会解决该问题。
请注意,SQLITE_CSU 输出是由提到的实用程序生成的 here .
这是用于测试/生成上述内容的 onUpgrade 代码:-
@Override
public void onUpgrade(SQLiteDatabase db, int oldversion, int newversion) {
String tableSQL = "CREATE TABLE IF NOT EXISTS " + TBNAME + "(" +
"id INTEGER PRIMARY KEY, " +
"name TEXT," +
"image_path TEXT" +
")";
if (newversion > oldversion) {
Log.d("DBINFO","onUpgrade Started.");
db.execSQL(tableSQL);
//get all the current tableName table columns
List<String> columns = getColumns(db, "table1");
//rename the old table
db.execSQL("ALTER TABLE table1 RENAME TO temp_table1;");
Log.d("DBINFO","Databse info in onUpgrade after table rename to temp_table1");
CommonSQLiteUtilities.logDatabaseInfo(db);
//create the new tableName table
db.execSQL(tableSQL);
Log.d("DBINFO","Databse info in onUpgrade after creation of new table1.");
CommonSQLiteUtilities.logDatabaseInfo(db);
//merge the old and new columns, keep all the old ones
String loginfo = "Orignal Column Info - ";
for (String s:columns) {
loginfo = loginfo + "Column=" + s + " ";
}
Log.d("ORIG COLUMNS",loginfo);
loginfo = "Columns from new table1 - ";
for (String s: getColumns(db, "table1")) {
loginfo = loginfo + "Column=" + s + " ";
}
Log.d("NEWTAB COLUMNS",loginfo);
columns.retainAll(getColumns(db, "table1"));
loginfo = "Columns After Retain - ";
for (String s: columns) {
loginfo = loginfo + "Column=" + s + " ";
}
Log.d("RETAIN COLUMNS",loginfo);
String cols = joinColumnNames(columns, ",");
Log.d("JOINED COLUMNS",cols);
//insert the old data in the new table
db.execSQL(String.format("INSERT INTO %s (%s) SELECT %s FROM temp_%s", "table1", cols, cols, "table1"));
//delete the old temp table
db.execSQL("DROP table 'temp_table1'");
Log.d("DBINFO","Databse info in onUpgrade after DELETING temp_table1.");
}
<小时/>
在此示例中,数据从原始表中提取到游标中,使用不同的名称创建新表,加载数据,然后删除旧表,最后重命名新表。
这可能稍微复杂一些,因为它维护 id,但它只有 1 行:-
cv.put("id",Long.toString(olddata.getLong(olddata.getColumnIndex("id"))));
注意!行注释为 //<<<<<
纯粹用于记录 Activity 。虽然olddata.moveToPosition(-1); //<<<<<
如果读取/遍历旧数据游标(即将位置返回到第一行之前),则需要。
@Override
public void onUpgrade(SQLiteDatabase db, int oldversion, int newversion) {
String temptableSQL = "CREATE TABLE IF NOT EXISTS temp_" + TBNAME + "(" +
"id INTEGER PRIMARY KEY, " +
"name TEXT," +
"image_path TEXT" +
")";
if (newversion > oldversion) {
Log.d("DBINFO","onUpgrade Started.");
db.execSQL(temptableSQL);
Cursor olddata = db.query(TBNAME,null,null,null,null,null,null);
CommonSQLiteUtilities.logCursorData(olddata); //<<<<
olddata.moveToPosition(-1); //<<<<<
//db.execSQL(tableSQL);
Log.d("DBINFO","Databse info in onUpgrade after creation of new table1.");
CommonSQLiteUtilities.logDatabaseInfo(db); //<<<<<
Log.d("DBCOPYDATA",
"Copying data from OLD Data. # rows = " +
Integer.toString(olddata.getCount())
); //<<<<<
while (olddata.moveToNext()) {
ContentValues cv = new ContentValues();
Log.d("DBCOPYDATA","ROW " + Integer.toString(olddata.getPosition()) +
"\n\tID=" + Long.toString(olddata.getLong(olddata.getColumnIndex("id"))) +
"\n\tNAME=" + olddata.getString(olddata.getColumnIndex("name")) +
"\n\tIMG =" + olddata.getString(olddata.getColumnIndex("image_byte"))
); //<<<<<
cv.put("id",Long.toString(olddata.getLong(olddata.getColumnIndex("id"))));
cv.put("name",olddata.getString(olddata.getColumnIndex("name")));
cv.put("image_path",olddata.getString(olddata.getColumnIndex("image_byte")));
Long id = db.insert("temp_" + TBNAME,null,cv);
Log.d("DBCOPYDATA","INSERTED ID=" + Long.toString(id)); //<<<<<
}
olddata.close();
db.execSQL("DROP TABLE IF EXISTS " + TBNAME);
db.execSQL("ALTER TABLE temp_" + TBNAME + " RENAME TO " + TBNAME);
Cursor csr = db.query(TBNAME,null,null,null,null,null,null); //<<<<<
CommonSQLiteUtilities.logDatabaseInfo(db); //<<<<<
CommonSQLiteUtilities.logCursorData(csr); //<<<<<
csr.close(); //<<<<<
}
}
成功测试的输出:-
10-15 17:54:41.618 16960-16960/? D/DBINFO: onUpgrade Started.
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: logCursorData Cursor has 3 rows with 3 columns.
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: Information for row 1 offset=0
For Column id Type is INTEGER value as String is 1 value as long is 1 value as double is 1.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image001 value as long is 0 value as double is 0.0
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: Information for row 2 offset=1
For Column id Type is INTEGER value as String is 2 value as long is 2 value as double is 2.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image002 value as long is 0 value as double is 0.0
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: Information for row 3 offset=2
For Column id Type is INTEGER value as String is 3 value as long is 3 value as double is 3.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image003 value as long is 0 value as double is 0.0
10-15 17:54:41.619 16960-16960/? D/DBINFO: Databse info in onUpgrade after creation of new table1.
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: DatabaseList Row 1 Name=main File=/data/data/mjt.sqlitedbexamples/databases/imagestore
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: Database Version = 1
10-15 17:54:41.619 16960-16960/? D/SQLITE_CSU: Table Name = android_metadata Created Using = CREATE TABLE android_metadata (locale TEXT)
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = android_metadata ColumnName = locale ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table Name = table1 Created Using = CREATE TABLE table1(id INTEGER PRIMARY KEY,name TEXT,image_byte TEXT)
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = image_byte ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table Name = temp_table1 Created Using = CREATE TABLE temp_table1(id INTEGER PRIMARY KEY, name TEXT,image_path TEXT)
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = temp_table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = temp_table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.620 16960-16960/? D/SQLITE_CSU: Table = temp_table1 ColumnName = image_path ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: Copying data from OLD Data. # rows = 3
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: ROW 0
ID=1
NAME=Test001
IMG =mypath/mypath/images/image001
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: INSERTED ID=1
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: ROW 1
ID=2
NAME=Test001
IMG =mypath/mypath/images/image002
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: INSERTED ID=2
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: ROW 2
ID=3
NAME=Test001
IMG =mypath/mypath/images/image003
10-15 17:54:41.620 16960-16960/? D/DBCOPYDATA: INSERTED ID=3
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: DatabaseList Row 1 Name=main File=/data/data/mjt.sqlitedbexamples/databases/imagestore
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Database Version = 1
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table Name = android_metadata Created Using = CREATE TABLE android_metadata (locale TEXT)
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table = android_metadata ColumnName = locale ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table Name = table1 Created Using = CREATE TABLE "table1"(id INTEGER PRIMARY KEY, name TEXT,image_path TEXT)
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = id ColumnType = INTEGER Default Value = null PRIMARY KEY SEQUENCE = 1
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = name ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Table = table1 ColumnName = image_path ColumnType = TEXT Default Value = null PRIMARY KEY SEQUENCE = 0
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: logCursorData Cursor has 3 rows with 3 columns.
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Information for row 1 offset=0
For Column id Type is INTEGER value as String is 1 value as long is 1 value as double is 1.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image001 value as long is 0 value as double is 0.0
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Information for row 2 offset=1
For Column id Type is INTEGER value as String is 2 value as long is 2 value as double is 2.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image002 value as long is 0 value as double is 0.0
10-15 17:54:41.621 16960-16960/? D/SQLITE_CSU: Information for row 3 offset=2
For Column id Type is INTEGER value as String is 3 value as long is 3 value as double is 3.0
For Column name Type is STRING value as String is Test001 value as long is 0 value as double is 0.0
For Column image_byte Type is STRING value as String is mypath/mypath/images/image003 value as long is 0 value as double is 0.0
关于java - SQLite onUpgrade 删除和添加列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46745513/
我正在 csv 上使用 hadoop 来分析一些数据。我使用sql/mysql(不确定)来分析数据,现在陷入了僵局。 我花了好几个小时在谷歌上搜索,却没有找到任何相关的东西。我需要一个查询,在该查询中
我正在为 Bootstrap 网格布局的“简单”任务而苦苦挣扎。我希望在大视口(viewport)上有 4 列,然后在中型设备上有 2 列,最后在较小的设备上只有 1 列。 当我测试我的代码片段时,似
对于这个令人困惑的标题,我深表歉意,我想不出这个问题的正确措辞。相反,我只会给你背景信息和目标: 这是在一个表中,一个人可能有也可能没有多行数据,这些行可能包含相同的 activity_id 值,也可
具有 3 列的数据库表 - A int , B int , C int 我的问题是: 如何使用 Sequelize 结果找到 A > B + C const countTasks = await Ta
我在通过以下功能编写此查询时遇到问题: 首先按第 2 列 DESC 排序,然后从“不同的第 1 列”中选择 只有 Column1 是 DISTINCT 此查询没有帮助,因为它首先从第 1 列中进行选择
使用 Bootstrap 非常有趣和有帮助,目前我在创建以下需求时遇到问题。 “使用 bootstrap 在桌面上有 4 列,在平板电脑上有 2 列,在移动设备上有 1 列”谁能告诉我正确的结构 最佳
我是 R 新手,正在问一个非常基本的问题。当然,我在尝试从所提供的示例中获取指导的同时做了功课here和 here ,但无法在我的案例中实现这个想法,即可能是由于我的问题中的比较维度更大。 我的实
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个 df , delta1 delta2 0 -1 2 0 -1 0 0 0 我想知道如何分配 delt
您好,我想知道是否可以执行以下操作。显然,我已经尝试在 phpMyAdmin 中运行它,但出现错误。也许还有另一种方式来编写此查询。 SELECT * FROM eat_eat_restaurants
我有 2 个列表(标题和数据值)。我想要将数据值列 1 匹配并替换为头文件列 1,以获得与 dataValue 列 1 和标题值列 2 匹配的值 头文件 TotalLoad,M0001001 Hois
我有两个不同长度的文件,file2 是一个很大的引用文件,我从中提取文件 1 的数据。 我有一行 awk,我通常会对其进行调整以在我的文件中进行查找和替换,但它总是在同一列中进行查找和替换。 所以对于
假设我有两个表,如下所示。 create table contract( c_ID number(1) primary key, c_name varchar2(50) not
我有一个带有 varchar 列的 H2 表,其检查约束定义如下: CONSTRAINT my_constraint CHECK (varchar_field <> '') 以下插入语句失败,但当我删
这是最少量的代码,可以清楚地说明我的问题: One Two Three 前 2 个 div 应该是 2 个左列。第三个应该占据页面的其余部分。最后,我将添加选项来隐藏和
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
今天我开始使用 JexcelApi 并遇到了这个:当您尝试从特定位置获取元素时,不是像您通常期望的那样使用sheet.getCell(row,col),而是使用sheet.getCell(col,ro
我有一个包含 28 列的数据库。第一列是代码,第二列是名称,其余是值。 public void displayData() { con.Open(); MySqlDataAdapter
我很沮丧:每当我缩小这个网页时,一切都变得一团糟。我如何将网页居中,以便我可以缩小并且元素不会被错误定位。 (它应该是 2 列,但所有内容都合并为 1)我试过 但由于某种原因,这不起作用。 www.o
我是一名优秀的程序员,十分优秀!