android.database.sqlite.SQLiteConstraintException: UNIQUE constraint failed异常解决

第一次运行正常,后面再运行会出现这个异常:

E/SQLiteDatabase: Error inserting thumb_uri=content://media/external/images/thumbnails/3959 image_id=8474 image_uri=content://media/external/images/media/8474 orientation=360
     android.database.sqlite.SQLiteConstraintException: UNIQUE constraint failed: Book1.image_id (Sqlite code 1555), (OS error - 2:No such file or directory)
           at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Native Method)
           at android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:790)
           at android.database.sqlite.SQLiteSession.executeForLastInsertedRowId(SQLiteSession.java:788)
           at android.database.sqlite.SQLiteStatement.executeInsert(SQLiteStatement.java:86)
           at android.database.sqlite.SQLiteDatabase.insertWithOnConflict(SQLiteDatabase.java:1506)
           at android.database.sqlite.SQLiteDatabase.insert(SQLiteDatabase.java:1371)
           at faceall.imagedivisiondemo.utils.LocalImageHelper.updateImage(LocalImageHelper.java:483)
           at faceall.imagedivisiondemo.fragment.PhotosFragment$1.run(PhotosFragment.java:134)
           at java.lang.Thread.run(Thread.java:833)

查看错误位置:
 value.put("image_id", id);
 value.put("image_uri",uri);
 value.put("thumb_uri", thumbUri);
 value.put("orientation",(360-degree));
 db.insert("Book1", null, value);//错误位置
再查看表Book1的写法:
//所有图片
    public static final String CREATE_BOOK1 = "create table Book1 ("
            + "image_id integer primary key, "
            + "image_uri text, "
            + "thumb_uri text, "
            + "orientation integer)";

可能发生这种BUG的两种情况
1:定义的字段为NOT NULL,而插入时对应的字段为NULL
2:你定义的自动为PRIMARY,而插入时想插入的值已经在表中存在。

在Book1表中把image_id设成了主键,第一次插入正确,当第二次测试时相当于再次插入同样的信息,由于id已存在,所以抛出了异常。
针对情况2,建议用replace代替insert。

在数据库中会要求插入时,某条记录不存在则插入,存在则更新。或更新时,某条记录存在则更新,不存在则插入。比如:人员信息数据库,某个身份证若已存在,重复插入则更新,否则新增记录;网页缓存数据库,某个url已存在,重复插入则更新,否则新增记录。
在mysql中可以使用replace into或是insert into …. on duplicate key update实现。在sqlite中我们同样可以使用replace into实现:
public boolean insertDayData(BleData bleData) {
        ContentValues cv = new ContentValues();
        cv.put("datatime", bleData.getDatetime());
        cv.put("stepnum", bleData.getStepnum());
        cv.put("heartrate", bleData.getHeartrate());

        database.replace("day_ble", null, cv);
               querySevenDayData();
                Log.d(TAG, "insertDayData insert data success!");
        return true;
    }
需要注意的是,当表有一个PRIMARY KEY或UNIQUE索引才有意义

具体步骤分两步。下面以http cache表为例,仅包含三个字段,主键_id, url, content:
第一步:
 新建唯一索引: CREATE UNIQUE INDEX mycolumn_index ON mytable (myclumn);
CREATE UNIQUE INDEX unique_index_url ON http_cache (url);
java中可以直接在SQLiteOpenHelper的OnCreate中添加:
public class DbHelper extends SQLiteOpenHelper {   
   public void onCreate(SQLiteDatabase db) {  
        db.beginTransaction();  
        try {  
            db.execSQL(DbConstants.CREATE_HTTP_RESPONSE_TABLE_UNIQUE_INDEX.toString());  
            db.setTransactionSuccessful();  
        } finally {  
            db.endTransaction();  
        }  
    }  
第二步:
调用replace into语句
REPLACEINTOhttp_cache(url,content)VALUES('http://www.baidu.com/','');
java中可以
SQLiteDatabase.replace(DbConstants.HTTP_RESPONSE_TABLE_TABLE_NAME, null, contentValues); 

primary key与unique的区别:
定义了 UNIQUE 约束的字段中不能包含重复值,可以为一个或多个字段定义 UNIQUE 约束。因此,UNIQUE 即可以在字段级也可以在表级定义, 在 UNIQUED 约束的字段上可以包含空值。ORACLE自动会为具有 PRIMARY KEY 约束的字段(主码字段)建立一个唯一索引和一个NOT NULL约束,定义PRIMARY KEY约束时可以为它的索引;
UNIQUED 可空,可以在一个表里的一个或多个字段定义;PRIMARY KEY 不可空不可重复,在一个表里可以定义联合主键;
简单的说,primary key = unique +  not null
unique 就是唯一,当你需要限定你的某个表字段每个值都唯一,没有重复值时使用。比如说,如果你有一个person 表,并且表中有个身份证的column,那么你就可以指定该字段为unique。 从技术的角度来看,Primary Key和Unique Key有很多相似之处。但还是有以下区别:
一、作为Primary Key的域/域组不能为null,而Unique Key可以。
二、在一个表中只能有一个Primary Key,而多个Unique Key可以同时存在。
更大的区别在逻辑设计上。Primary Key一般在逻辑设计中用作记录标识,这也是设置Primary Key的本来用意,而Unique Key只是为了保证域/域组的唯一性。
oracle的constraint中有两种约束,都是对列的唯一性限制――unique与primary key,但其中是有区别的:
1、unique key要求列唯一,但不包括null字段,也就是约束的列可以为空且仅要求列中的值除null之外不重复即可;
2、primary key也要求列唯一,同时又限制字段的值不能为null,相当于Primary Key=unique + not null。
创建一个primary key和unique key都会相应的创建一个unique index。
0primary key的语法:alter table table name add constraint key name primary key( columns);
unique key的语法:alter table table name add constraint key name unique( columns);

一个表只能有一个主键,但是可以有好多个UNIQUE,而且UNIQUE可以为NULL值,如员工的电话号码一般就用UNIQUE,因为电话号码肯定是唯一的,但是有的员工可能没有电话。
主键肯定是唯一的,但唯一的不一定是主键;
不要总把UNIQUE索引和UNIQUE约束混为一谈
1、primary key = unique + not null
2、唯一约束和主键一样都是约束的范畴,而且都可以作为外键的参考,不同的是,一张表只能有一个主键
3、主键和唯一约束的创建需要依靠索引,如果在创建主键或唯一约束的时候没有已经建好的索引可以使用的话,Oracle会自动建立一个唯一的索引。




你可能感兴趣的:(Android)