20170104第一行代码第六章数据存储

持久化技术:文件存储、SharedPreference存储和数据库存储。

一、文件存储

不对存储内容进行任何格式化处理,原封不动存储到文件中,适合于存储简单文本数据或二进制数据。

1、存储5部曲:

首先,获取FileOutputStream对象;该对象通过Context类提供的openFileOutput()方法获取;该方法需要两个参数:文件名和写入方式Context.MODE_PRIVATE或Context.MODE_APPEND。

然后,获取OutputStreamWriter对象;该对象传入参数为上面获取的FileOutputStream对象;

之后,获取BufferedWriter对象;该对象传入参数为上面获取的OutputStreamWriter对象;

最后,利用BufferedWriter对象的write()方法将数据写入。

写入完毕之后,要关闭BufferedWriter对象。

public void save(String data)

{try {

fileOutputStream=openFileOutput("mydata", Context.MODE_PRIVATE);

outputStreamWriter=new OutputStreamWriter(fileOutputStream);

bufferedWriter=new BufferedWriter(outputStreamWriter);

bufferedWriter.write(data);

} catch (Exception e) {

e.printStackTrace();}

finally

{if(bufferedWriter!=null)

try {

bufferedWriter.close();

} catch (IOException e) {

e.printStackTrace();}}}

2、读取5部曲:

首先,获取FileInputStream对象,该对象通过Context类提供的openFileInput()方法获取,该方法接收一个参数,文件名;

然后,获取InputStreamReader对象,需要传入上面获取的FileInputStream对象;

之后,获取BufferedReader对象,需要传入上面的InputStreamReader对象;

最后,利用readLine()读取。

读取完毕之后,记得关闭BufferedReader对象;

读取方法为:

StringBuilder content=new StringBuilder();

String line=””;

while((line=bufferedReader.readLine())!=null)

{content.append(line);}

String text=load();

if(!TextUtils.isEmpty(text))           // TextUtils.isEmpty(text)可以判断为空或null

{editText.setText(text);

editText.setSelection(text.length());   //将输入光标放置到文本末尾

}

public String load()

{StringBuilder content=new StringBuilder();

String line="";

try {

fileInputStream = openFileInput("mydata");

inputStreamReader=new InputStreamReader(fileInputStream);

bufferedReader=new BufferedReader(inputStreamReader);

while((line=bufferedReader.readLine())!=null)

{

content.append(line);

}

} catch (Exception e) {

e.printStackTrace();

}finally

{

if(bufferedReader!=null)

try {

bufferedReader.close();

} catch (IOException e) {

e.printStackTrace();

}}

return content.toString();}

二、SharedPreferences存储

SharedPreferences采用键值对方式存储数据。

1、存储4部曲

首先,获取SharedPreferences对象:

Context类中getSharedPreferences()方法:两个参数,名称和操作模式MODE_PRIVATE,MODE_MULTI_PROCESS。

Activity类中getPreferences()方法:一个参数,操作模式。

PreferenceManager类中getDefaultSharedPreferences()方法:一个参数context参数。

然后,调用SharedPreferences对象的edit()方法获取SharedPreferences.Editor对象;

之后,向SharedPreferences.Editor对象中添加数据,putString()、putBoolean()等;

最后,调用commit()写入数据。

SharedPreferences sharedPreferences=getSharedPreferences("data", MODE_PRIVATE);

SharedPreferences.Editor editor=sharedPreferences.edit();

editor.putString("name", "hujun");

editor.putInt("age", 24);

editor.putBoolean("married", false);

editor.commit();

2、读取2部曲

首先,获取SharedPreferences对象;

然后,利用getString()、getInt()等获取数据。传入两个参数,一是key,二是默认值,即如果找不到该key,则返回该默认值。

三、SQLite数据库

1、创建数据库

首先,建立自己的public class MyDatabaseHelper extents SQLiteOpenHelper{}

SQLiteOpenHelper类为抽象类,需要重写两个抽象方法:onCreate()和onUpdate();

还要重写构造方法。参数最少的构造方法有四个参数:Context、database_name、Cursor(一般写null、vision版本号)。

然后,建立MyDatabaseHelper的对象,调用其getReadableDatabase()或getWritableDatabase()方法。

public class MyDatabaseHelper extends SQLiteOpenHelper{

public static final String

CREATE_BOOK="create table Book(id integer primary key autoincrement,author text,price real,pages integer,name text)";

private Context mcontext;

publicMyDatabaseHelper(Context context, String name,

CursorFactory factory, int version) {

super(context, name, factory, version);

mcontext=context;}

public voidonCreate(SQLiteDatabase db) {

db.execSQL(CREATE_BOOK);}

public voidonUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {}}

之后,在activity中调用。

databaseHelper = new MyDatabaseHelper(this, "BookStore", null, 1);

databaseHelper.getWritableDatabase();

此时,创建了一个名为BookStore,版本为1的数据库。并且,调用了onCreate()方法,建立了一个名为Book的表。

2、升级数据库

注意:

(1)当相同版本的数据库BookStore已经存在了,则MyDatabaseHelper中的onCreate()方法就不会执行了。

(2)如果创建表时发现该表已经存在,则会直接报错,所以需要将已经存在的表删除。

(3)只有当版本号比原来高,onUpdate()方法才会执行。

databaseHelper = new MyDatabaseHelper(this, "BookStore", null,2);

public void onCreate(SQLiteDatabase db) {

db.execSQL(CREATE_BOOK);

db.execSQL(CREATE_CATEGORY);

Toast.makeText(mcontext, "创建成功", Toast.LENGTH_SHORT).show();

}

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

db.execSQL("drop table if exists Book");

db.execSQL("drop table if exists Category");

onCreate(db);

}

缺点:需要删除已经存在的表,以前的数据也会被删除。

3、添加数据

4、更新数据

5、删除数据

6、查询数据

7、直接使用SQL操作数据库

添加数据:db.execSQL(“insert into Book(name,author,pages,price) values(?,?,?,?)”,new String[] {“God”,”hujun”,”324”,”34.23”});

更新数据:db.execSQL(“update Book set price=? where name=?”, new String[] {“12.33”,”God”});

删除数据:db.execSQL(“delete from Book where pages>?”,new String[] {“100”});

查询数据:db.rawQuery(“select * from Book”,null);

查询数据方法:

首先,获取Cursor:

Cursor cursor=null;

cursor=db.rawQuery("select * from Book", null);

然后,利用cursor得到返回的数据:

while(cursor.moveToNext())

{

String author=cursor.getString(cursor.getColumnIndex("author"));

String name=cursor.getString(cursor.getColumnIndex("name"));

double price=cursor.getDouble(cursor.getColumnIndex("price"));

int pages=cursor.getInt(cursor.getColumnIndex("pages"));

Log.i("main", author);

Log.i("main", name);

Log.i("main", price+"");

Log.i("main", pages+"");

}

如果是第一次登陆,则将初始化为早7点,晚11点;

如果不是第一次登陆,则加载之前的设置

四、数据库最佳实践

1、使用事务

事务操作可以保证某一系列的操作要么全部完成,要么全部都不完成,保证安全。

首先,开启事务:db.beginTransaction();

然后,执行数据库操作;

最后,事务成功:db.setTransactionSuccessful();

当执行操作出现故障时,事务不会成功,则以上的操作不会完成,数据库不会变化。

public void replaceData()

{

SQLiteDatabase db=databaseHelper.getWritableDatabase();

db.beginTransaction(); //开启事务

//首先,删除数据

db.execSQL("delete from Book where name=?",new String[] {"God"});

//手动设定一个断点,模仿删除失败

if(true)

try {

throw new IOException();

} catch (IOException e) {

e.printStackTrace();

}

//然后,添加数据

db.execSQL("insert into Book(name,author,pages,price) values(?,?,?,?)",new String[] {"God","hujun","324","34.23"});

//事务执行成功

db.setTransactionSuccessful();

}

2、升级数据库最佳写法

之前升级数据库时,为了确保新增加的table之前没有,就先把table全部删除了,这样的行为是很傻比的。

改进方法为:

改变version,然后在onUpdate()中进行判断,根据version执行不同的更新操作。

databaseHelper = new MyDatabaseHelper(this, "BookStore", null, 1);

比如app一共更新了两次,第一次添加了table Category,第二次在table Book中添加category_id.

于是,就有三个版本:

version=1

建数据库,添加table Book

version=2

添加table Category

version=3

在table Book中添加category_id

主程序中:databaseHelper = new MyDatabaseHelper(this, "BookStore", null, 3);

将版本号改为3。

如果是第一次安装,则直接进入onCreate(),直接安装最新版本;

如果第一次安装版本1,则进入onUpdate(),连续进行了两次升级;

如果第一次安装版本2,则进入onUpdate(),只进行了最后一次升级。

public static final String

CREATE_BOOK="create table Book(id integer primary key autoincrement," +

"author text,price real,pages integer,name text,category_id,integer)";

public static final String

CREATE_CATEGORY="create table Category(id integer primary key autoincrement,category_name text,category_code integer)";

private Context mcontext;

public void onCreate(SQLiteDatabase db) {

db.execSQL(CREATE_BOOK);

db.execSQL(CREATE_CATEGORY);

Toast.makeText(mcontext, "创建成功", Toast.LENGTH_SHORT).show();

}

//注意,switch语句里面,没有添加break!

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

switch (oldVersion) {

case 1:

db.execSQL(CREATE_CATEGORY);

case 2:

db.execSQL("alter table Book add column category_id integer");

default:}}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容