在Android的数据存储机制中,文件存储和共享参数存储只适合存储一些简单的数据,如果要存储复杂数据,就需要使用到SQLite数据库存储,在Android中系统内置了数据库,运算速度快,占用资源也比较少。
一。创建数据库:
创建数据库时,用到了SQliteOpenHelper类,这是一个抽象类,故继承它,并实现它的两个方法:onCreate(),onUpdate();
onCreate():只调用一次,当数据库不存在时,执行onCreate()从而创建数据库,当数据库已经存在时,不再执行onCreate();
onUpdate():升级数据库时,即旧版本小于新版本时执行。
因此,将创建表的语句放在onCrete()中,创建数据库时,就会完成表的创建。
但是,onCreate()并不是自动执行的,SQliteOpenHelper中还有两个方法:getReadableDatabase(),getWriteableDatabase(),这两个方法创建或打开已有的数据库,只有SQliteOpenHelper的实例
并返回一个SQLiteDatabase对象的实例,以便进行CRUD操作。调用getReadableDatabase()/getWriteableDatabase()时,会判断是否要执行onCreate();
getReadableDatabase():以只读的方式打开或创建数据库;
getWriteableDatabase():可以对数据库进行CRUD操作;
二。CRDU操作:
添加数据时:调用SQLiteDatabase的insert();
修改数据时:调用SQLiteDatabase的update();
删除数据时:调用SQLiteDatabase的delete();
查询数据时:调用SQLiteDatabase的query();
(各个方法参数在实例中注明了。)
三。具体实例:
(1)创建数据库:
/**
* 继承SQLiteOpenHelper,并重写之方法、构造函数
*
*/
public class MyOpenHelper extends SQLiteOpenHelper {
//建表语句:id为主键、自增 ,name 文本类型,age 整形,sex 文本类型,job 文本类型
private final String CREATE_STUDENT="create table student(" +
"id integer primary key autoincrement," +
"name text," +
"age integer," +
"sex text," +
"job text)";
public MyOpenHelper(Context context, String name, CursorFactory factory,
int version) {
super(context, name, factory, version);
}
/**
* 当dbHelper.getWritableDatabase(),时若存在数据库,则不再执行onCreate(),若不存在则执行onCreate();
* 同时完成对表的创建
*/
@Override
public void onCreate(SQLiteDatabase db) {
// 执行建表语句
db.execSQL(CREATE_STUDENT);
}
/**
* 对数据库的升级
*/
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
if(newVersion>oldVersion){
db.execSQL("drop table id exists student");
onCreate(db);
}
}
}
public class MainActivity extends Activity {
private Button button_createDB;
private MyOpenHelper dbHelper;
private SQLiteDatabase db;
private final String TABLE_NAME="student";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button_createDB=(Button) findViewById(R.id.btn_createDB);
//实例化MyOpenHelper,参数:1.上下文,2.数据库名称,3.游标工厂对象,一般为null,4.版本号
dbHelper=new MyOpenHelper(MainActivity.this, "Person.db", null, 1);
//创建数据库的方法
createDB();
}
private void createDB() {
// TODO Auto-generated method stub
button_createDB.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
/**
* 当调用getWritableDatabase()或getReadableDatabase()后,若该数据库不存在,则会执行MyOpenHelper * 中的onCreate()方法,创建数据库;若已存在数据库,则不再调用onCreate();也就是说,onCreate()
* 只调用一次。
*/
dbHelper.getWritableDatabase();
Toast.makeText(MainActivity.this, "数据库创建成功",Toast.LENGTH_LONG).show();
}
});
}
}
布局效果如下:
(2)添加数据:
public class MainActivity extends Activity {
private Button button_createDB;
private MyOpenHelper dbHelper;
private SQLiteDatabase db;
private final String TABLE_NAME="student";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button_createDB=(Button) findViewById(R.id.btn_createDB);
//实例化MyOpenHelper
dbHelper=new MyOpenHelper(MainActivity.this, "Person.db", null, 1);
//创建数据库的方法
createDB();
}
private void createDB() {
// TODO Auto-generated method stub
button_createDB.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
/**
* 当调用getWritableDatabase()或getReadableDatabase()后,若该数据库不存在,则会执行MyOpenHelper中的
* onCreate()方法,创建数据库;若已存在数据库,则不再调用onCreate();也就是说,onCreate()只调用一次。
*/
dbHelper.getWritableDatabase();
Toast.makeText(MainActivity.this, "数据库创建成功",Toast.LENGTH_LONG).show();
}
});
}
/**
* 添加数据
* @param view
*/
public void insertData(View view){
db=dbHelper.getWritableDatabase();
/**
* 添加数据和修改数据时,都会把数据放入ContentValues的实例中,根据键值对的形式进行操作,键是数据库中相应的字段名,值为要存储的数据
*/
ContentValues values=new ContentValues();
values.put("name", "张三");
values.put("job", "班长");
values.put("age", 21);
values.put("sex", "男");
db.insert(TABLE_NAME, null, values);
//清空values,以便放入第二条数据
values.clear();
values.put("name", "小梅");
values.put("job", "团支部");
values.put("age", 21);
values.put("sex", "女");
db.insert(TABLE_NAME, null, values);
Toast.makeText(MainActivity.this, "添加成功", Toast.LENGTH_SHORT).show();
}
}
截屏结果:
(3)修改数据:
public void updateData(View v){
db=dbHelper.getWritableDatabase();
/**
* 添加数据和修改数据时,都会把数据放入ContentValues的实例中,根据键值对的形式进行操作,键是数据库中相应的字段名,值为要存储的数据
*/
ContentValues values=new ContentValues();
values.put("name", "新张三");
//参数:1.表名,2.contentValues值,3.限制条件,4.占位符
db.update(TABLE_NAME, values, "id=?", new String[]{String.valueOf(1)});
Toast.makeText(MainActivity.this, "修改成功", Toast.LENGTH_SHORT).show();
}
截屏结果:
(4)删除数据:
public void deleteData(View view){
db=dbHelper.getWritableDatabase();
//参数:1.表名,2.限制条件,3.占位符
db.delete(TABLE_NAME, "sex=?", new String[]{"女"});
db.delete(TABLE_NAME, "sex=?", new String[]{"女"});
Toast.makeText(MainActivity.this, "删除成功", Toast.LENGTH_SHORT).show();
}
截屏结果:
(5)查询数据:
public void queryData(View view){
db=dbHelper.getReadableDatabase();
/**
* 调用query()查询数据时,返回一个Cursor对象,对Cursor对象进行遍历,可得到每条数据各个字段的数据
* cursosr.getColumnIndex("sex"):根据字段名得到对应字段的列数;
* cursosr.getString(int column):根据所在列数得到数据
*/
Cursor cursosr=db.query(TABLE_NAME, null, null, null, null, null, null);
while(cursosr.moveToNext()){
String name=cursosr.getString(cursosr.getColumnIndex("name"));
String sex=cursosr.getString(cursosr.getColumnIndex("sex"));
String age=cursosr.getString(cursosr.getColumnIndex("age"));
String job=cursosr.getString(cursosr.getColumnIndex("job"));
Log.i("TAG", name+","+sex+","+age+","+job);
Toast.makeText(MainActivity.this, "查询成功", Toast.LENGTH_SHORT).show();
}
}
截屏结果: