您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“Android中的存儲實例分析”,內容詳細,步驟清晰,細節處理妥當,希望這篇“Android中的存儲實例分析”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
最簡單的一種。在嘗試過程中發現,手機中很多文件夾都沒有權限讀寫。我們可以將我們需要寫的文件存放到App中的files
文件夾中,當然我們有權限在整個App中讀寫文件
可以通過API獲取一個file對象,這里的this就是MainActivity類
// 獲取當前包下的files路徑 /data/data/top.woodwhale.qqlogin/files File filesDir = this.getFilesDir();
之后就可以通過文件輸出流寫入文件:
File filesFile = new File(filesDir,"info.txt"); boolean flag = (filesFile.exists() || filesFile.createNewFile()); FileOutputStream fos = new FileOutputStream(file); fos.write((ac+"***"+pwd).getBytes(StandardCharsets.UTF_8)); fos.close();
寫入成功:
當然,我們既然在這個App中都有權限,那么所有目錄都可以寫:
// 寫入到自己有權限寫的地方 File file = new File("/data/data/top.woodwhale.qqlogin/info.txt");
雖然現在很多的手機都不支持SD卡了,但是仍然有平板使用。
直接放出一個Activity類,其中調用了nvironment.getExternalStorageDirectory();
方法類獲取一個sd卡file對象,使用Formatter.formatFileSize(this,externalStorageDirectory.getFreeSpace()));
Formatter類中的轉化,將long類型轉化為大小類型,同時調用sd卡file對象的getFreeSpace()
方法,獲取卡中剩余的空間,之后就是寫入externalStorageDirectory.getPath()
卡中的路徑
public class SdcardActivity extends Activity { private Button btn; public static String TAG = "SdcardActivity"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_sc_card_rw); btn = this.findViewById(R.id.bt_sdw); btn.setOnClickListener(view -> { File externalStorageDirectory = Environment.getExternalStorageDirectory(); Log.d(TAG, "sd卡路徑是:"+externalStorageDirectory.getPath()); Log.d(TAG,"sd卡剩余空間是"+ Formatter.formatFileSize(this,externalStorageDirectory.getFreeSpace())); File file = new File(externalStorageDirectory,"love.txt"); try { boolean flag = file.exists() || file.createNewFile(); if (flag) { FileOutputStream fos = new FileOutputStream(file); fos.write("woodwhale love sheepbotany".getBytes(StandardCharsets.UTF_8)); fos.close(); } } catch (Exception e) { e.printStackTrace(); } }); } }
但是,在此之前,我們需要一個SD卡的讀寫權限,我們在AndrodiManifest.xml中配置下面的ses-permission
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
最終,在我們的sd卡中看到如下結果,證明寫入成功:
SharedPreferences
是android下的一個類,功能就是記錄偏好設置,形成一個xml文件
我們可以用SharedPreferences
來存儲一些信息。
例如常見的這種:
我們勾選之后,再次打開app仍然處于勾選狀態。
那么這種情況如何實現呢?
通過xml生成上面的布局
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="80dp"> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:layout_centerVertical="true"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="未知來源" android:textColor="@color/teal_200" android:layout_marginLeft="10dp" android:textSize="20sp" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="運行安裝未知來源的應用" android:layout_marginLeft="10dp" android:layout_marginTop="10dp" android:textSize="18sp"/> </LinearLayout> <Switch android: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來進行設置偏好,整體代碼如下
package top.woodwhale.qqlogin; import android.app.Activity; import android.content.SharedPreferences; import android.os.Bundle; import android.util.Log; import android.widget.CompoundButton; import android.widget.Switch; import androidx.annotation.Nullable; public class PreferenceDemoActivity extends Activity { private Switch sw; public static String TAG = "PreferenceDemoActivity"; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_pre_demo); sw = (Switch) this.findViewById(R.id.sw_source); SharedPreferences settingInfo = this.getSharedPreferences("settingInfo", MODE_PRIVATE); SharedPreferences.Editor edit = settingInfo.edit(); sw.setOnCheckedChangeListener(new MyListener(edit)); boolean state = settingInfo.getBoolean("state", true); Log.d(TAG,"STATE=="+ state); sw.setChecked(state); } } // 改變狀態的監聽器 class MyListener implements CompoundButton.OnCheckedChangeListener { SharedPreferences.Editor editor; public MyListener(SharedPreferences.Editor editor) { this.editor = editor; } @Override public void onCheckedChanged(CompoundButton compoundButton, boolean b) { Log.d(PreferenceDemoActivity.TAG,"current state : "+ b); editor.putBoolean("state",b); // 要保存的數據類型 editor.commit(); // 保存數據 } }
其中,editor的功能是保存數據
其次,為了每次打開App都可以看到我們的配置,通過讀取偏好配置文件,設置switch框的勾選
這樣就可以同步偏好設置的勾選啦!
最后我們可以在手機內部看到我們寫入的偏好設置xml文件了,這樣也屬于存儲在App內部
Android設備自帶SQLite數據庫,如果掌握過mysql,那么SQLite非常容易上手,且不說提供了非常簡便的API,就算是自己寫也比JDBC簡單!
首先我們不適用提供的API來實現一次增刪改查!
BaseDao類本來是用來連接數據庫等基礎的,具體的增刪改查應該在service層實現,但為了這里測試,我們將crud的方法寫入到BaseDao類中封裝起來。具體架構如下:
首先是Constants
類,是常量類,其中有我們的數據庫名、版本號、表名
public class Constants { public static final String DATABASE_NAME = "woodwhale.db"; public static final int VERSION_CODE = 1; public static final String TABLE_NAME = "user"; }
其次是DatabaseHelper
類,繼承SQLiteOpenHelper
類,用來開啟數據庫,其中的onCreate方法是數據庫創建時的回調,onUpgrade方法時升級數據時的回調,我們在Constans類中寫了一個版本號,爸爸那邊每次升級可以加入新的功能,可以寫在onUpgrade方法中,通過switch實現。不過需要注意,升級只能讓版本號上升,不能降級,否則會報錯!
package top.woodwhale.qqlogin.SQLite.utils; import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper; import android.util.Log; public class DatabaseHelper extends SQLiteOpenHelper { public static String TAG = "DatabaseHelper"; /** * @param context 上下文 */ public DatabaseHelper( Context context) { super(context, Constants.DATABASE_NAME, null, Constants.VERSION_CODE); } @Override public void onCreate(SQLiteDatabase sqLiteDatabase) { // 創建時的回調 Log.d(TAG, "創建數據庫"); String sql = "create table " + Constants.TABLE_NAME + " (id integer,name varchar,age integer)"; sqLiteDatabase.execSQL(sql); } @Override public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) { // 升級數據庫的回調 Log.d(TAG, "升級數據庫!"); String sql = null; switch (i) { case 1: sql = "alter table "+ Constants.TABLE_NAME + " add phone integer"; sqLiteDatabase.execSQL(sql); break; case 2: sql = "alter table "+ Constants.TABLE_NAME + " add address varchar"; sqLiteDatabase.execSQL(sql); break; } } }
最后就是我們封裝好的數據庫BaseDao
類,通過語句實現了增刪改查
package top.woodwhale.qqlogin.SQLite.dao; import android.annotation.SuppressLint; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.util.Log; import top.woodwhale.qqlogin.SQLite.utils.Constants; import top.woodwhale.qqlogin.SQLite.utils.DatabaseHelper; // BaseDao類 public class BaseDao { private final DatabaseHelper dbh; private SQLiteDatabase db; public static String TAG = "BaseDao"; public BaseDao(Context context) { dbh = new DatabaseHelper(context); } // 增 public void add(int id, String name, int age) { db = dbh.getWritableDatabase(); String sql = "insert into " + Constants.TABLE_NAME + "(id,name,age) values(?,?,?)"; Object[] params = new Object[]{id,name,age}; db.execSQL(sql,params); db.close(); } // 刪 public void free(int id) { db = dbh.getWritableDatabase(); String sql = "delete from " + Constants.TABLE_NAME + " where id=?"; Object[] params = new Object[]{id}; db.execSQL(sql,params); db.close(); } // 改 public void edit(int id, int age) { db = dbh.getWritableDatabase(); String sql = "update " + Constants.TABLE_NAME +" set age = ? where id = ?"; Object[] params = new Object[]{age,id}; db.execSQL(sql,params); db.close(); } // 查 @SuppressLint("Range") public void show(int id) { db = dbh.getReadableDatabase(); String sql = "select * from " + Constants.TABLE_NAME +" where id = ?"; String[] params = new String[]{String.valueOf(id)}; Cursor cursor = db.rawQuery(sql, params); while (cursor.moveToNext()) { String name = cursor.getString(cursor.getColumnIndex("name")); Log.d(TAG,"name == "+name); int age = cursor.getInt(cursor.getColumnIndex("age")); Log.d(TAG,"age == "+age); } cursor.close(); db.close(); } }
接著我們在AndroidTest包下進行測試
package top.woodwhale.qqlogin; import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.util.Log; import androidx.test.platform.app.InstrumentationRegistry; import androidx.test.ext.junit.runners.AndroidJUnit4; import org.junit.Test; import org.junit.runner.RunWith; import static org.junit.Assert.*; import top.woodwhale.qqlogin.SQLite.dao.BaseDao; import top.woodwhale.qqlogin.SQLite.utils.DatabaseHelper; /** * Instrumented test, which will execute on an Android device. * * @see <a href="http://d.android.com/tools/testing" rel="external nofollow" >Testing documentation</a> */ @RunWith(AndroidJUnit4.class) public class ExampleInstrumentedTest { public static final String TAG = "ExampleInstrumentedTest"; public static final Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();; public static final BaseDao dao = new BaseDao(appContext);; @Test public void useAppContext() { // Context of the app under test. assertEquals("top.woodwhale.qqlogin", appContext.getPackageName()); } @Test public void testCreate() { DatabaseHelper dbh = new DatabaseHelper(appContext); SQLiteDatabase writableDatabase = dbh.getWritableDatabase(); Log.d(TAG, writableDatabase.getPath()); } @Test public void testAdd() { dao.add(1,"woodwhale",19); dao.add(2,"sheepbotany",21); } @Test public void testFree() { dao.free(1); } @Test public void testEdit() { dao.edit(1,3); } @Test public void testShow() { dao.show(1); } }
增刪改查都成功,成功就如圖所示:
由于只有查詢有log回顯,在logcat中之后show方法出現了log
那么使用Google寫好的增刪改查api可以避免我們sql語句的格式問題和語法錯誤
經過測試,如下代碼沒有問題(在BaseDao類中)
// 使用API的添加 public void addByAPI(int id, String name, int age) { ContentValues contentValues = new ContentValues(); contentValues.put("id",id); contentValues.put("name",name); contentValues.put("age",age); db = dbh.getWritableDatabase(); db.insert(Constants.TABLE_NAME,null,contentValues); db.close(); } // 刪除 public void freeByAPI(int id) { db = dbh.getWritableDatabase(); db.delete(Constants.TABLE_NAME,"id = ?",new String[]{String.valueOf(id)}); db.close(); Log.d(TAG,"API刪除成功!"); } // 修改 public void editByAPI(int id, String name, Integer age) { ContentValues contentValues = new ContentValues(); if (name != null) { contentValues.put("name",name); } if (age != null) { contentValues.put("age",age); } db = dbh.getWritableDatabase(); db.update(Constants.TABLE_NAME,contentValues,"id = ?", new String[]{String.valueOf(id)}); db.close(); } // 查詢 public void showByAPI(int id) { db = dbh.getReadableDatabase(); Cursor cursor = db.query(false, Constants.TABLE_NAME, new String[]{"id", "name", "age"}, "id = ?", new String[]{String.valueOf(id)}, "id", null, null, null); while (cursor.moveToNext()) { int ID = cursor.getInt(0); Log.d(TAG,"ID == "+ID); String name = cursor.getString(1); Log.d(TAG,"name == "+name); int age = cursor.getInt(2); Log.d(TAG,"age == "+age); } cursor.close(); db.close(); }
使用db.beginTransaction();
db.setTransactionSuccessful();
db.endTransaction();
三個方法來進行事務的處理。
簡單的一個測試類
// 測試一個數據庫事物 @Test public void testTransaction() { DatabaseHelper dbh = new DatabaseHelper(appContext); SQLiteDatabase db = dbh.getWritableDatabase(); db.beginTransaction(); Log.d(TAG,"事物開啟!"); try { db.execSQL("update " + Constants.TABLE_NAME +" set age = 114 where id = 1"); int i = 10 / 0; db.execSQL("update " + Constants.TABLE_NAME +" set age = 114 where id = 2"); db.setTransactionSuccessful(); Log.d(TAG,"事物成功!"); } catch (Exception e) { e.printStackTrace(); } finally { db.endTransaction(); db.close(); Log.d(TAG,"事物關閉!"); } dao.showByAPI(1); dao.showByAPI(2); }
看看logcat,首先是進入了 事物開啟,然后程序進入了try中,因為除以了一個0所以報錯,捕獲異常了之后就是進入到finally中關閉了事務,可以發現我們sql中的信息都回滾了,沒有改變。
我們把int i = 10 / 0;
刪了試一試,可以看到成功執行事物。
值得注意的是,事物開啟之后,僅有當前的db對象可以執行sql語句,使用Dao類中的方法是無法進行增刪改查的,因為對這些得到的db對象上了鎖!
讀到這里,這篇“Android中的存儲實例分析”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。