android提供以下四种存储方式:
- sharepreference
- sqlite
- file
- contentprovider
android系统中数据基本都是私有的,一般存放在“data/data/程序包名”目录下。如果要实现数据共享,正确的方式是使用contentprovider。
sharedpreference
sharedpreference是一种轻型的数据存储方式,实际上是基于xml文件存储的“key-value”键值对数据。通常用来存储程序的一些配置信息。其存储在“data/data/程序包名/shared_prefs目录下。
sharedpreference本身只能获取数据,不支持存储和修改。存储和修改要通过editor对象来实现。
1)、修改和存储数据
- 根据context的getsharedprerences(key, [模式])方法获取sharedpreference对象;
- 利用sharedpreference的editor()方法获取editor对象;
- 通过editor的putxxx()方法,将键值对存储数据;
- 通过editor的commit()方法将数据提交到sharedpreference内。
综合例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
//设置单例里面的数值,然后再将数值写入到sharedpreference里 private string setcityname(string _cityname){ city.getcity().setcityname(_cityname); context ctx =mainactivity. this ; sharedpreferences sp =ctx.getsharedpreferences( "city" , mode_private); editor editor=sp.edit(); editor.putstring( "cityname" , city.getcity().getcityname()); editor.commit(); return city.getcity().getcityname(); } |
2)、获取数据
- 同样根据context对象获取sharedpreference对象;
- 直接使用sharedpreference的getxxx(key)方法获取数据。
综合例子:
1
2
3
4
5
6
7
8
9
10
11
|
//从单例里面找,如果不存在则在sharedpreferences里面读取 private string getcityname(){ string cityname = city.getcity().getcityname(); if (cityname== null ||cityname== "" ){ context ctx =mainactivity. this ; sharedpreferences sp =ctx.getsharedpreferences( "city" , mode_private); city.getcity().setcityname(sp.getstring( "cityname" , "广州" )); } return city.getcity().getcityname(); } |
注意
getsharedprerences(key, [模式])方法中,第一个参数其实对应到xml的文件名,相同key的数据会保存到同一个文件下。
使用sharedpreference的getxxx(key)方法获取数据的时候,如果key不存在的活,不会出现报错,会返回none。建议使用getxxx()的时候指定默认值。
sqlite
sqlite是一个轻量级关系型数据库,既然是关系型数据库,那操作起来其实跟mysql、sql server差不多的。
需要注意的一点是,sqlite只有null、integer、real(浮点数)、text(字符串)和blob(大数据)五种类型,不存在boolean和date类型。
1)、创建数据库
通过openorcreatedatabase(string path, sqlitedatabase.cursorfactory factory)方法创建,如果库已创建,则打开数据库。
2)、创建表
sqlitedatabase没有提供创建表的方法,所以要靠execsql()方法来实现。看名字也知道execsql()用于直接执行sql的。
db.execsql(sql);
增
使用sqlitedatabase的insert(string table, string nullcolumnhack, contentvalues values)方法插入数据。contentvalues 类,类似于java中的map,以键值对的方式保存数据。
1
2
3
4
|
contentvalues values= new contentvalues(); values.put( "name" , "liangjh" ); values.put( "password" , "123456" ); db.insert( "t_user" , "id" , values); |
删
删除数据就比较直接了。使用sqlitedatabase的delete(string table, string whereclause, string[] whereargs)实现。如果不想把参数写在whereargs里面,可以直接把条件写在whereclause里面。
1
2
3
4
|
// 方式1 直接将条件写入到条件里面(个人觉得容易被注入,但其实数据都在客户端,没啥安全性可言) db.delete( "t_user" , "id=1" , null ); // 方式2 条件分开写,感觉比较安全 db.delete( "t_user" , "name=? and password =?" , new string[]{ "weiyg" , "112233" }); |
查
查询有2个方法,query()和rawquery()两个方法,区别在于query()是将sql里面的各参数提取出query()对应的参数中。可参考下面例子。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
// 使用rawquery // cursor c = db.rawquery("select * from t_user", null); // db.rawquery("select * from t_user where id=1", null); // db.rawquery("select * from t_user where id=?", new string[]{"1"}); // 使用query() cursor c = db.query( "t_user" , new string[]{ "id" , "name" }, "name=?" , new string[]{ "weiyg" }, null , null , null ); c.movetofirst(); while (!c.isafterlast()){ string msg= "" ; for ( int i= 0 ,j=c.getcolumncount();i<j;i++){ msg+= "--" +c.getstring(i); } log.v( "sqlite" , "data:" +msg); c.movetonext(); } |
改
使用sqlitedatabase的update(string table, contentvalues values, string whereclause, string[] whereargs)可以修改数据。whereclause和whereargs用于设置其条件。contentvalues对象为数据。
1
2
3
4
5
6
|
contentvalues values= new contentvalues(); values.put( "password" , "111111" ); // 方式1 条件写在字符串内 db.update( "t_user" , values, "id=1" , null ); // 方式2 条件和字符串分开 db.update( "t_user" , values, "name=? or password=?" , new string[]{ "weiyg" , "123456" }); |
其它
无论何时,打开的数据库,记得关闭。
db.close()
另外使用begintransaction()和endtransaction()可以设置事务。
file
文件储存方式,很久以前讲过,这里不说明。
contentprovider
contentprovider相对于其它的方式比较复杂,当然其功能相对于其它的方式也是革命性的改变。它能够实现跨应用之间的数据操作。利用contentresolver对象的delete、update、insert、query等方法去操contentprovider的对象,让contentprovider对象的方法去对数据操作。实现方式为:
在a程序中定义一个contentprovider,重载其增删查改等方法;
在a程序中的androidmanifest.xml中注册contentprovider;
在b程序中通过contentresolver和uri来获取contentprovider的数据,同样利用resolver的增删查改方法来获得和处理数据。
1)、在a程序定义一个provider
新建一个类,继承contentprovider,并重载其delete()、insert()、query()、update()、gettype()、oncreate()方法。譬如下面的例子,重载其oncreate和query方法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
public class myprovider extends contentprovider { @override public int delete(uri uri, string selection, string[] selectionargs) { // todo auto-generated method stub return 0 ; } @override public string gettype(uri uri) { // todo auto-generated method stub return null ; } @override public uri insert(uri uri, contentvalues values) { // todo auto-generated method stub return null ; } @override public boolean oncreate() { // 新建个数据库并插入一条数据 sqlitedatabase db= this .getcontext().openorcreatedatabase( "test_db2.db" , context.mode_private, null ); db.execsql( "create table t_user (id integer primary key autoincrement,name text not null)" ); contentvalues values= new contentvalues(); values.put( "name" , "liangjh2" ); db.insert( "t_user" , "id" , values); db.close(); return false ; } @override public cursor query(uri uri, string[] projection, string selection, string[] selectionargs, string sortorder) { // 获取数据 sqlitedatabase db= this .getcontext().openorcreatedatabase( "test_db2.db" , context.mode_private, null ); cursor c = db.query( "t_user" , null , null , null , null , null , null ); db.close(); return c; } @override public int update(uri uri, contentvalues values, string selection, string[] selectionargs) { // todo auto-generated method stub return 0 ; } } |
注册contentprovider
在androidmanifest.xml中声明contentprovider,authorities属性定义了contentprovider的uri标识。关于uri标识属另一个范畴,自行查询。provider标识要放在<application></application>里面。如果遇到了"permission denial: opening provide..."的错误,可以试试在节点加“android:exported="true"”。
1
2
3
4
|
<application ...> ... <provider android:name= ".myprovider" android:authorities= "com.example.androidtestdemo" android:exported= "true" /> </application> |
2)、在b程序获取数据
用context获取到当前的contentresolver,根据uri地址和contentresolver的query方法获取a程序的数据。uri地址和a程序中androidmanifest.xml定义的autorities要一致。当然,同类可以进行其它的操作。
1
2
3
4
5
6
7
8
9
10
11
|
context ctx=mainactivity. this ; contentresolver resolver =ctx.getcontentresolver(); uri uri=uri.parse( "content://com.example.androidtestdemo" ); cursor c = resolver.query(uri, null , null , null , null ); c.movetofirst(); while (!c.isafterlast()){ for ( int i= 0 ,j=c.getcolumncount();i<j;i++){ log.v( "android2" , "" +c.getstring(i)); } c.movetonext(); } |
以上就是本文的全部内容,希望对大家的学习有所帮助。