服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Android - 四种Android数据存储方式

四种Android数据存储方式

2021-06-23 15:52Ron Ngai Android

这篇文章主要为大家详细介绍了四种Android数据存储方式,感兴趣的小伙伴们可以参考一下

android提供以下四种存储方式

  • sharepreference
  • sqlite
  • file
  • contentprovider

android系统中数据基本都是私有的,一般存放在“data/data/程序包名”目录下。如果要实现数据共享,正确的方式是使用contentprovider。 

sharedpreference
sharedpreference是一种轻型的数据存储方式,实际上是基于xml文件存储的“key-value”键值对数据。通常用来存储程序的一些配置信息。其存储在“data/data/程序包名/shared_prefs目录下。
sharedpreference本身只能获取数据,不支持存储和修改。存储和修改要通过editor对象来实现。 

1)、修改和存储数据

  1. 根据context的getsharedprerences(key, [模式])方法获取sharedpreference对象;
  2. 利用sharedpreference的editor()方法获取editor对象;
  3. 通过editor的putxxx()方法,将键值对存储数据;
  4. 通过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)、获取数据

  1. 同样根据context对象获取sharedpreference对象;
  2. 直接使用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)方法创建,如果库已创建,则打开数据库。

复制代码 代码如下:
sqlitedatabase db =this.openorcreatedatabase("test_db.db", context.mode_private, null);

2)、创建表
sqlitedatabase没有提供创建表的方法,所以要靠execsql()方法来实现。看名字也知道execsql()用于直接执行sql的。
复制代码 代码如下:
string sql="create table t_user (id integer primary key autoincrement, name text not null,password text not null)";
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();
}

以上就是本文的全部内容,希望对大家的学习有所帮助。

延伸 · 阅读

精彩推荐