Android中的存储实例分析(android,开发技术)

时间:2024-05-09 05:27:41 作者 : 石家庄SEO 分类 : 开发技术
  • TAG :

    1、存储在App内部

    最简单的一种。在尝试过程中发现,手机中很多文件夹都没有权限读写。我们可以将我们需要写的文件存放到App中的files文件夹中,当然我们有权限在整个App中读写文件

    Android中的存储实例分析

    可以通过API获取一个file对象,这里的this就是MainActivity类

    //获取当前包下的files路径/data/data/top.woodwhale.qqlogin/filesFilefilesDir=this.getFilesDir();
    之后就可以通过文件输出流写入文件:
    FilefilesFile=newFile(filesDir,"info.txt");booleanflag=(filesFile.exists()||filesFile.createNewFile());FileOutputStreamfos=newFileOutputStream(file);fos.write((ac+"***"+pwd).getBytes(StandardCharsets.UTF_8));fos.close();

    写入成功:

    Android中的存储实例分析

    当然,我们既然在这个App中都有权限,那么所有目录都可以写:

    //写入到自己有权限写的地方Filefile=newFile("/data/data/top.woodwhale.qqlogin/info.txt");

    2、SD卡外部存储

    虽然现在很多的手机都不支持SD卡了,但是仍然有平板使用。

    直接放出一个Activity类,其中调用了nvironment.getExternalStorageDirectory();方法类获取一个sd卡file对象,使用Formatter.formatFileSize(this,externalStorageDirectory.getFreeSpace()));Formatter类中的转化,将long类型转化为大小类型,同时调用sd卡file对象的getFreeSpace()方法,获取卡中剩余的空间,之后就是写入externalStorageDirectory.getPath()卡中的路径

    publicclassSdcardActivityextendsActivity{privateButtonbtn;publicstaticStringTAG="SdcardActivity";@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_sc_card_rw);btn=this.findViewById(R.id.bt_sdw);btn.setOnClickListener(view->{FileexternalStorageDirectory=Environment.getExternalStorageDirectory();Log.d(TAG,"sd卡路径是:"+externalStorageDirectory.getPath());Log.d(TAG,"sd卡剩余空间是"+Formatter.formatFileSize(this,externalStorageDirectory.getFreeSpace()));Filefile=newFile(externalStorageDirectory,"love.txt");try{booleanflag=file.exists()||file.createNewFile();if(flag){FileOutputStreamfos=newFileOutputStream(file);fos.write("woodwhalelovesheepbotany".getBytes(StandardCharsets.UTF_8));fos.close();}}catch(Exceptione){e.printStackTrace();}});}}
    但是,在此之前,我们需要一个SD卡的读写权限,我们在AndrodiManifest.xml中配置下面的ses-permission
    <uses-permissionandroid:name="android.permission.WRITE_EXTERNAL_STORAGE"/><uses-permissionandroid:name="android.permission.READ_EXTERNAL_STORAGE"/>
    最终,在我们的sd卡中看到如下结果,证明写入成功:

    Android中的存储实例分析

    3、SharedPreferences存储

    SharedPreferences是android下的一个类,功能就是记录偏好设置,形成一个xml文件

    我们可以用SharedPreferences来存储一些信息。

    例如常见的这种:

    Android中的存储实例分析

    我们勾选之后,再次打开app仍然处于勾选状态。

    那么这种情况如何实现呢?

    通过xml生成上面的布局

    <?xmlversion="1.0"encoding="utf-8"?><RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="80dp"><LinearLayoutandroid:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:layout_centerVertical="true"><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="未知来源"android:textColor="@color/teal_200"android:layout_marginLeft="10dp"android:textSize="20sp"/><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="运行安装未知来源的应用"android:layout_marginLeft="10dp"android:layout_marginTop="10dp"android:textSize="18sp"/></LinearLayout><Switchandroid:id="@+id/sw_source"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentRight="true"android:layout_marginTop="30dp"android:layout_marginRight="10dp"/></RelativeLayout>
    我们把Switch这个选择框在activity类中赋予一个变量,给他加上一个OnCheckedChangeListener,再使用SharedPreferences来进行设置偏好,整体代码如下
    packagetop.woodwhale.qqlogin;importandroid.app.Activity;importandroid.content.SharedPreferences;importandroid.os.Bundle;importandroid.util.Log;importandroid.widget.CompoundButton;importandroid.widget.Switch;importandroidx.annotation.Nullable;publicclassPreferenceDemoActivityextendsActivity{privateSwitchsw;publicstaticStringTAG="PreferenceDemoActivity";@OverrideprotectedvoidonCreate(@NullableBundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_pre_demo);sw=(Switch)this.findViewById(R.id.sw_source);SharedPreferencessettingInfo=this.getSharedPreferences("settingInfo",MODE_PRIVATE);SharedPreferences.Editoredit=settingInfo.edit();sw.setOnCheckedChangeListener(newMyListener(edit));booleanstate=settingInfo.getBoolean("state",true);Log.d(TAG,"STATE=="+state);sw.setChecked(state);}}//改变状态的监听器classMyListenerimplementsCompoundButton.OnCheckedChangeListener{SharedPreferences.Editoreditor;publicMyListener(SharedPreferences.Editoreditor){this.editor=editor;}@OverridepublicvoidonCheckedChanged(CompoundButtoncompoundButton,booleanb){Log.d(PreferenceDemoActivity.TAG,"currentstate:"+b);editor.putBoolean("state",b);//要保存的数据类型editor.commit();//保存数据}}
    其中,editor的功能是保存数据

    Android中的存储实例分析

    其次,为了每次打开App都可以看到我们的配置,通过读取偏好配置文件,设置switch框的勾选

    Android中的存储实例分析

    这样就可以同步偏好设置的勾选啦!

    最后我们可以在手机内部看到我们写入的偏好设置xml文件了,这样也属于存储在App内部

    Android中的存储实例分析

    Android中的存储实例分析

    4、使用SQLite数据库存储

    Android设备自带SQLite数据库,如果掌握过mysql,那么SQLite非常容易上手,且不说提供了非常简便的API,就算是自己写也比JDBC简单!

    首先我们不适用提供的API来实现一次增删改查!

    4.1 自己完成一个BaseDao类

    BaseDao类本来是用来连接数据库等基础的,具体的增删改查应该在service层实现,但为了这里测试,我们将crud的方法写入到BaseDao类中封装起来。具体架构如下:

    Android中的存储实例分析

    首先是Constants类,是常量类,其中有我们的数据库名、版本号、表名

    publicclassConstants{publicstaticfinalStringDATABASE_NAME="woodwhale.db";publicstaticfinalintVERSION_CODE=1;publicstaticfinalStringTABLE_NAME="user";}

    其次是DatabaseHelper类,继承SQLiteOpenHelper类,用来开启数据库,其中的onCreate方法是数据库创建时的回调,onUpgrade方法时升级数据时的回调,我们在Constans类中写了一个版本号,爸爸那边每次升级可以加入新的功能,可以写在onUpgrade方法中,通过switch实现。不过需要注意,升级只能让版本号上升,不能降级,否则会报错

    packagetop.woodwhale.qqlogin.SQLite.utils;importandroid.content.Context;importandroid.database.sqlite.SQLiteDatabase;importandroid.database.sqlite.SQLiteOpenHelper;importandroid.util.Log;publicclassDatabaseHelperextendsSQLiteOpenHelper{publicstaticStringTAG="DatabaseHelper";/***@paramcontext上下文*/publicDatabaseHelper(Contextcontext){super(context,Constants.DATABASE_NAME,null,Constants.VERSION_CODE);}@OverridepublicvoidonCreate(SQLiteDatabasesqLiteDatabase){//创建时的回调Log.d(TAG,"创建数据库");Stringsql="createtable"+Constants.TABLE_NAME+"(idinteger,namevarchar,ageinteger)";sqLiteDatabase.execSQL(sql);}@OverridepublicvoidonUpgrade(SQLiteDatabasesqLiteDatabase,inti,inti1){//升级数据库的回调Log.d(TAG,"升级数据库!");Stringsql=null;switch(i){case1:sql="altertable"+Constants.TABLE_NAME+"addphoneinteger";sqLiteDatabase.execSQL(sql);break;case2:sql="altertable"+Constants.TABLE_NAME+"addaddressvarchar";sqLiteDatabase.execSQL(sql);break;}}}

    最后就是我们封装好的数据库BaseDao类,通过语句实现了增删改查

    packagetop.woodwhale.qqlogin.SQLite.dao;importandroid.annotation.SuppressLint;importandroid.content.Context;importandroid.database.Cursor;importandroid.database.sqlite.SQLiteDatabase;importandroid.util.Log;importtop.woodwhale.qqlogin.SQLite.utils.Constants;importtop.woodwhale.qqlogin.SQLite.utils.DatabaseHelper;//BaseDao类publicclassBaseDao{privatefinalDatabaseHelperdbh;privateSQLiteDatabasedb;publicstaticStringTAG="BaseDao";publicBaseDao(Contextcontext){dbh=newDatabaseHelper(context);}//增publicvoidadd(intid,Stringname,intage){db=dbh.getWritableDatabase();Stringsql="insertinto"+Constants.TABLE_NAME+"(id,name,age)values(?,?,?)";Object[]params=newObject[]{id,name,age};db.execSQL(sql,params);db.close();}//删publicvoidfree(intid){db=dbh.getWritableDatabase();Stringsql="deletefrom"+Constants.TABLE_NAME+"whereid=?";Object[]params=newObject[]{id};db.execSQL(sql,params);db.close();}//改publicvoidedit(intid,intage){db=dbh.getWritableDatabase();Stringsql="update"+Constants.TABLE_NAME+"setage=?whereid=?";Object[]params=newObject[]{age,id};db.execSQL(sql,params);db.close();}//查@SuppressLint("Range")publicvoidshow(intid){db=dbh.getReadableDatabase();Stringsql="select*from"+Constants.TABLE_NAME+"whereid=?";String[]params=newString[]{String.valueOf(id)};Cursorcursor=db.rawQuery(sql,params);while(cursor.moveToNext()){Stringname=cursor.getString(cursor.getColumnIndex("name"));Log.d(TAG,"name=="+name);intage=cursor.getInt(cursor.getColumnIndex("age"));Log.d(TAG,"age=="+age);}cursor.close();db.close();}}

    接着我们在AndroidTest包下进行测试

    Android中的存储实例分析

    packagetop.woodwhale.qqlogin;importandroid.content.Context;importandroid.database.sqlite.SQLiteDatabase;importandroid.util.Log;importandroidx.test.platform.app.InstrumentationRegistry;importandroidx.test.ext.junit.runners.AndroidJUnit4;importorg.junit.Test;importorg.junit.runner.RunWith;importstaticorg.junit.Assert.*;importtop.woodwhale.qqlogin.SQLite.dao.BaseDao;importtop.woodwhale.qqlogin.SQLite.utils.DatabaseHelper;/***Instrumentedtest,whichwillexecuteonanAndroiddevice.**@see<ahref="http://d.android.com/tools/testing"rel="externalnofollow">Testingdocumentation</a>*/@RunWith(AndroidJUnit4.class)publicclassExampleInstrumentedTest{publicstaticfinalStringTAG="ExampleInstrumentedTest";publicstaticfinalContextappContext=InstrumentationRegistry.getInstrumentation().getTargetContext();;publicstaticfinalBaseDaodao=newBaseDao(appContext);;@TestpublicvoiduseAppContext(){//Contextoftheappundertest.assertEquals("top.woodwhale.qqlogin",appContext.getPackageName());}@TestpublicvoidtestCreate(){DatabaseHelperdbh=newDatabaseHelper(appContext);SQLiteDatabasewritableDatabase=dbh.getWritableDatabase();Log.d(TAG,writableDatabase.getPath());}@TestpublicvoidtestAdd(){dao.add(1,"woodwhale",19);dao.add(2,"sheepbotany",21);}@TestpublicvoidtestFree(){dao.free(1);}@TestpublicvoidtestEdit(){dao.edit(1,3);}@TestpublicvoidtestShow(){dao.show(1);}}

    增删改查都成功,成功就如图所示:

    Android中的存储实例分析

    由于只有查询有log回显,在logcat中之后show方法出现了log

    Android中的存储实例分析

    4.2 使用Google写的API处理

    那么使用Google写好的增删改查api可以避免我们sql语句的格式问题和语法错误

    经过测试,如下代码没有问题(在BaseDao类中)

    //使用API的添加publicvoidaddByAPI(intid,Stringname,intage){ContentValuescontentValues=newContentValues();contentValues.put("id",id);contentValues.put("name",name);contentValues.put("age",age);db=dbh.getWritableDatabase();db.insert(Constants.TABLE_NAME,null,contentValues);db.close();}//删除publicvoidfreeByAPI(intid){db=dbh.getWritableDatabase();db.delete(Constants.TABLE_NAME,"id=?",newString[]{String.valueOf(id)});db.close();Log.d(TAG,"API删除成功!");}//修改publicvoideditByAPI(intid,Stringname,Integerage){ContentValuescontentValues=newContentValues();if(name!=null){contentValues.put("name",name);}if(age!=null){contentValues.put("age",age);}db=dbh.getWritableDatabase();db.update(Constants.TABLE_NAME,contentValues,"id=?",newString[]{String.valueOf(id)});db.close();}//查询publicvoidshowByAPI(intid){db=dbh.getReadableDatabase();Cursorcursor=db.query(false,Constants.TABLE_NAME,newString[]{"id","name","age"},"id=?",newString[]{String.valueOf(id)},"id",null,null,null);while(cursor.moveToNext()){intID=cursor.getInt(0);Log.d(TAG,"ID=="+ID);Stringname=cursor.getString(1);Log.d(TAG,"name=="+name);intage=cursor.getInt(2);Log.d(TAG,"age=="+age);}cursor.close();db.close();}

    4.3 事务使用

    使用db.beginTransaction(); db.setTransactionSuccessful(); db.endTransaction();三个方法来进行事务的处理。

    简单的一个测试类

    //测试一个数据库事物@TestpublicvoidtestTransaction(){DatabaseHelperdbh=newDatabaseHelper(appContext);SQLiteDatabasedb=dbh.getWritableDatabase();db.beginTransaction();Log.d(TAG,"事物开启!");try{db.execSQL("update"+Constants.TABLE_NAME+"setage=114whereid=1");inti=10/0;db.execSQL("update"+Constants.TABLE_NAME+"setage=114whereid=2");db.setTransactionSuccessful();Log.d(TAG,"事物成功!");}catch(Exceptione){e.printStackTrace();}finally{db.endTransaction();db.close();Log.d(TAG,"事物关闭!");}dao.showByAPI(1);dao.showByAPI(2);}

    看看logcat,首先是进入了 事物开启,然后程序进入了try中,因为除以了一个0所以报错,捕获异常了之后就是进入到finally中关闭了事务,可以发现我们sql中的信息都回滚了,没有改变。

    Android中的存储实例分析

    我们把int i = 10 / 0;删了试一试,可以看到成功执行事物。

    Android中的存储实例分析

    值得注意的是,事物开启之后,仅有当前的db对象可以执行sql语句,使用Dao类中的方法是无法进行增删改查的,因为对这些得到的db对象上了锁!

     </div> <div class="zixun-tj-product adv-bottom"></div> </div> </div> <div class="prve-next-news">
    本文:Android中的存储实例分析的详细内容,希望对您有所帮助,信息来源于网络。
    上一篇:如何用C++实现三子棋游戏下一篇:

    11 人围观 / 0 条评论 ↓快速评论↓

    (必须)

    (必须,保密)

    阿狸1 阿狸2 阿狸3 阿狸4 阿狸5 阿狸6 阿狸7 阿狸8 阿狸9 阿狸10 阿狸11 阿狸12 阿狸13 阿狸14 阿狸15 阿狸16 阿狸17 阿狸18