二、常用布局&存儲

數據的存儲

1.測試的相關概念 (了解)

    SUV  好的軟件不是開發出來的是測試出來的

    jd 黑客
    當當: -10 

    1.測試是否知道源代碼
        黑盒測試  不知道代碼
        白盒測試  知道代碼
    2.按照測試的粒度
        方法測試
        單元測試 Junit
        集成測試
        系統測試
    3.按照測試的暴力程度
        冒煙測試  硬件
        壓力測試 12306

    monkey測試: adb shell下的一個測試指令。 adb shell + monkey -p packagename count;

2.單元測試(了解,會用即可)

1.創建一個類集成AndroidTestCase,那么該類就具備單元測試的功能。

2.需要在androidmanifest.xml中的application節點下配置一個uses-library;
    <uses-library android:name="android.test.runner" />
3.需要在androidmanifest.xml中的manifest節點下配置一個instrumentation;targetPackage:需要測試的工程的包名。
        <instrumentation
    android:name="android.test.InstrumentationTestRunner"
    android:targetPackage="com.itheima.junit" />

4.如果不知道怎么配置androidmanifest.xml,可以新建一個android test project工程,會自動配置.

3.Logcat日志貓工具的使用 (會用即可)

    包括五種級別,可以添加過濾器過濾日志信息。能夠幫助我們觀察程序運行的狀態。
    e:
    w:
    i:
    d:
    v:

    在公司開發中一般打印日志用Log類,通常會封裝一個LogUtils,通過開關來控制日志信息的打印。
package top.mengmei219.login.util;
import android.util.Log;

public class LogUtil {

    private static Boolean isLog = Boolean.TRUE;

    public static void e(String tag, String msg){
        if (isLog){
            Log.e(tag,msg);
        }
    }
    public static void w(String tag, String msg){
        if (isLog){
            Log.w(tag,msg);
        }
    }
    public static void i(String tag, String msg){
        if (isLog){
            Log.i(tag,msg);
        }
    }
    public static void d(String tag, String msg){
        if (isLog){
            Log.d(tag,msg);
        }
    }
    public static void v(String tag, String msg){
        if (isLog){
            Log.v(tag,msg);
        }
    }
}

4.把數據存儲到文件(login案例) android 下的數據存儲

1.寫布局
    LinearLayout + RelativeLayout
2.寫業務邏輯
    a.找到相應控件

    b.設置按鈕的點擊事件

    c.在onclick方法中,獲取用戶輸入的用戶名密碼和是否記住密碼

    d.判斷用戶名密碼是否為空,不為空請求服務器(省略,默認請求成功)

    e.判斷是否記住密碼,如果記住,將用戶名密碼保存本地。???? 

    f.回顯用戶名密碼 ??


    //通過context對象獲取私有目錄,/data/data/packagename/filse
    context.getFileDir().getPath()
package top.mengmei219.login.util;

import android.content.Context;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.InputStreamReader;

public class UserInfoUtil {

    //static String path = "/data/data/top.mengmei219.login/";
    //context.getDataDir().getPath();

    static String fileName = "userInfo.txt";
    static String splitStr = "_##_";

    public static boolean saveUserInfo(Context context, String username, String password) {
        try {
            String userInfo = username + splitStr + password;
            File file = new File(context.getDataDir().getPath(), fileName);
            FileOutputStream out = new FileOutputStream(file);
            out.write(userInfo.getBytes());
            out.close();
            return true;
        } catch (java.io.IOException e) {
            e.printStackTrace();
        }
        return false;
    }

    public static boolean deleteUserInfo(Context context) {
        File file = new File(context.getDataDir().getPath(), fileName);
        return file.delete();
    }

    public static boolean isRemembered(Context context) {
        File file = new File(context.getDataDir().getPath(), fileName);
        boolean flag = file.exists();
        return flag;
    }

    public static CharSequence getUserName(Context context) {
        return getUserInfo(context, 0);
    }

    public static CharSequence getPassword(Context context) {
        return getUserInfo(context, 1);
    }

    private static CharSequence getUserInfo(Context context, int index) {
        if (isRemembered(context)) {
            try {
                File file = new File(context.getDataDir().getPath(), fileName);
                BufferedReader reader = new BufferedReader(new FileReader(file));
                String user_pwd = reader.readLine();
                reader.close();
                return user_pwd.split(splitStr)[index];
            } catch (java.io.IOException e) {
                e.printStackTrace();
            }
        }
        return "";
    }



    /////////////////////////////////////_android_//////////////////////////////////////////////////


    public static boolean saveUserInfo_android(Context context, String username, String password) {
        try {
            String userInfo = username + splitStr + password;
            FileOutputStream out = context.openFileOutput(fileName,Context.MODE_PRIVATE);
            out.write(userInfo.getBytes());
            out.close();
            return true;
        } catch (java.io.IOException e) {
            e.printStackTrace();
        }
        return false;
    }

    public static boolean deleteUserInfo_android(Context context) {
        File file = context.getFileStreamPath(fileName);
        return file.delete();
    }

    public static boolean isRemembered_android(Context context) {
        File file = context.getFileStreamPath(fileName);
        boolean flag = file.exists();
        return flag;
    }

    public static CharSequence getUserName_android(Context context) {
        return getUserInfo_android(context, 0);
    }

    public static CharSequence getPassword_android(Context context) {
        return getUserInfo_android(context, 1);
    }

    private static CharSequence getUserInfo_android(Context context, int index) {
        if (isRemembered_android(context)) {
            try {
                BufferedReader reader = new BufferedReader(new InputStreamReader(context.openFileInput(fileName)));
                String user_pwd = reader.readLine();
                reader.close();
                return user_pwd.split(splitStr)[index];
            } catch (java.io.IOException e) {
                e.printStackTrace();
            }
        }
        return "";
    }

}

5.存儲到SD卡,獲取SD的大小及可用空間 (重點)

使用Sdcard注意事項:

1.權限問題: 
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
2.硬性編碼問題:通過 Environment可以獲取sdcard的路徑
     Environment.getExternalStorageDirectory().getPath();
3.使用前需要判斷sdcard狀態
        if(!Environment.getExternalStorageState().equals( Environment.MEDIA_MOUNTED)){
                //sdcard狀態是沒有掛載的情況
                Toast.makeText(mContext, "sdcard不存在或未掛載", Toast.LENGTH_SHORT).show();
                return ;
            }
4.需要判斷sdcard剩余空間
                //判斷sdcard存儲空間是否滿足文件的存儲
            File sdcard_filedir = Environment.getExternalStorageDirectory();//得到sdcard的目錄作為一個文件對象
            long usableSpace = sdcard_filedir.getUsableSpace();//獲取文件目錄對象剩余空間
            long totalSpace = sdcard_filedir.getTotalSpace();
            //將一個long類型的文件大小格式化成用戶可以看懂的M,G字符串
            String usableSpace_str = Formatter.formatFileSize(mContext, usableSpace);
            String totalSpace_str = Formatter.formatFileSize(mContext, totalSpace);
            if(usableSpace < 1024 * 1024 * 200){//判斷剩余空間是否小于200M
                Toast.makeText(mContext, "sdcard剩余空間不足,無法滿足下載;剩余空間為:"+usableSpace_str, Toast.LENGTH_SHORT).show();
                return ;    
            }


    /data/data: context.getFileDir().getPath();
                是一個應用程序的私有目錄,只有當前應用程序有權限訪問讀寫,其他應用無權限訪問。一些安全性要求比較高的數據存放在該目錄,一般用來存放size比較小的數據。
    /sdcard:  Enviroment.getExternalStorageDirectory().getPath();
                是一個外部存儲目錄,只用應用聲明了<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>的一個權限,就可以訪問讀寫sdcard目錄;所以一般用來存放一些安全性不高的數據,文件size比較大的數據。
package top.mengmei219.login_sdCard.util;

import android.content.Context;
import android.os.Environment;
import android.text.format.Formatter;
import android.widget.Toast;
![權限圖.png](http://upload-images.jianshu.io/upload_images/6065021-e70ee76c2dbfb159.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

import java.io.BufferedReader;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;

public class UserInfoUtil {

    //App私有目錄(屬于data目錄:存儲App)
    //static String path = "/data/data/top.mengmei219.login_sdCard/";
    //context.getDataDir().getPath();

    //sd卡(存儲:電影、圖片等)
    //static String path = "/mnt/sdcard/";
    static String path = Environment.getExternalStorageDirectory().getPath();

    static String fileName = "userInfo.txt";
    static String splitStr = "_##_";

    public static boolean saveUserInfo(Context context, String username, String password) {
        if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
            Toast.makeText(context, "sd卡不存在或未掛載", Toast.LENGTH_SHORT).show();
        }
        long usableSpace = Environment.getExternalStorageDirectory().getUsableSpace();
        Toast.makeText(context, "sd卡剩余存儲空間:"+Formatter.formatFileSize(context,usableSpace), Toast.LENGTH_SHORT).show();

        String userInfo = username + splitStr + password;
        File file = new File(path, fileName);
        try {
            FileOutputStream out = new FileOutputStream(file);
            out.write(userInfo.getBytes());
            out.close();
            return true;
        } catch (java.io.IOException e) {
            e.printStackTrace();
        }
        return false;
    }

    public static boolean deleteUserInfo(Context context) {
        File file = new File(path, fileName);
        return file.delete();
    }

    public static boolean isRemembered(Context context) {
        File file = new File(path, fileName);
        return file.exists();
    }

    public static CharSequence getUserInfo(Context context, int index) {
        if (isRemembered(context)) {
            File file = new File(path, fileName);
            try {
                BufferedReader reader = new BufferedReader(new FileReader(file));
                String user_pwd = reader.readLine();
                reader.close();
                return user_pwd.split(splitStr)[index];
            } catch (java.io.IOException e) {
                e.printStackTrace();
            }
        }
        return "";
    }

    public static CharSequence getUserName(Context context) {
        return getUserInfo(context, 0);
    }

    public static CharSequence getPassword(Context context) {
        return getUserInfo(context, 1);
    }

}

7.文件的權限概念 (了解)

//通過context對象獲取一個私有目錄的文件讀取流  /data/data/packagename/files/userinfoi.txt
FileInputStream fileInputStream = context.openFileInput("userinfo.txt");

//通過context對象得到私有目錄下一個文件寫入流; name : 私有目錄文件的名稱    mode: 文件的操作模式, 私有,追加,全局讀,全局寫
    FileOutputStream fileOutputStream = context.openFileOutput("userinfo.txt", Context.MODE_PRIVATE);   



linux下一個文件的權限由10位標示:
1位:文件的類型,d:文件夾 l:快捷方式  -:文件
2-4: 該文件所屬用戶對本文件的權限 , rwx :用二進制標示,如果不是-就用1標示,是-用0標示;chmod指令賦權限。
5-7:該文件所屬用戶組對本文件的權限
8-10:其他用戶對該文件的權限。
權限圖.png

8.SharedPreferences介紹 (重點) 用來做數據存儲

    sharedPreferences是通過xml文件來做數據存儲的。
    一般用來存放一些標記性的數據,一些設置信息。


    *********使用sharedPreferences存儲數據

            
        1.通過Context對象創建一個SharedPreference對象
            //name:sharedpreference文件的名稱    mode:文件的操作模式
            SharedPreferences sharedPreferences = context.getSharedPreferences("userinfo.txt", Context.MODE_PRIVATE);
        2.通過sharedPreferences對象獲取一個Editor對象
            Editor editor = sharedPreferences.edit();
        3.往Editor中添加數據
            editor.putString("username", username);
            editor.putString("password", password);
        4.提交Editor對象
            editor.commit();

    *********使用sharedPreferences讀取數據

        1.通過Context對象創建一個SharedPreference對象
            SharedPreferences sharedPreferences = context.getSharedPreferences("userinfo.txt", Context.MODE_PRIVATE);
            
        2.通過sharedPreference獲取存放的數據
            //key:存放數據時的key   defValue: 默認值,根據業務需求來寫
            String username = sharedPreferences.getString("username", "");
            String password = sharedPreferences.getString("password", "");
            


    通過PreferenceManager可以獲取一個默認的sharepreferences對象      
    SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
package top.mengmei219.login_sharedReference.util;

import android.content.Context;
import android.content.SharedPreferences;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.InputStreamReader;

public class UserInfoUtil {

    static String fileName = "userInfo";
    static String key_username = "username";
    static String key_password = "password";

    public static boolean saveUserInfo(Context context, String username, String password) {
        SharedPreferences sharedPreferences = context.getSharedPreferences(fileName, Context.MODE_PRIVATE);
        SharedPreferences.Editor editor = sharedPreferences.edit();
        editor.putString(key_username, username);
        editor.putString(key_password, password);
        editor.commit();
        return true;
    }

    public static boolean deleteUserInfo(Context context) {
        return saveUserInfo(context,"","");
    }

    public static CharSequence getUserName(Context context) {
        return context.getSharedPreferences(fileName,Context.MODE_PRIVATE).getString(key_username,"");
    }

    public static CharSequence getPassword(Context context) {
        return context.getSharedPreferences(fileName,Context.MODE_PRIVATE).getString(key_password,"");
    }

}

9 生成xml的2種方式

一種硬編碼
一種使用:XmlSerializer ,如下:

package top.mengmei219.xmlserializer;

import android.content.Context;
import android.util.Xml;
import org.xmlpull.v1.XmlSerializer;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.List;

public class XMLUtil {

    public static <T> boolean bean2XML(List<T> list, Class<? extends T> clazz, String xmlFileName, Context context) {
        String clazzName = clazz.getName();
        Field[] fields = clazz.getDeclaredFields();
        try {
            XmlSerializer seri = Xml.newSerializer();
            seri.setOutput(context.openFileOutput(xmlFileName, Context.MODE_PRIVATE), "UTF-8");
          
            seri.startDocument("UTF-8", true);
            seri.startTag(null, clazzName + "es");
            for (int i = 0; i < list.size(); i++) {
                T t = list.get(i);
                seri.startTag(null, clazzName);
                for (Field field : fields) {
                    if (!Modifier.isStatic(field.getModifiers())) {
                        field.setAccessible(true);
                        String fieldName = field.getName();
                        seri.startTag(null, fieldName);
                        seri.text(field.get(t).toString());
                        seri.endTag(null, fieldName);
                    }
                }
                seri.endTag(null, clazzName);
            }
            seri.endTag(null, clazzName + "es");
            seri.endDocument();
            return true;
        } catch (IOException | IllegalAccessException e) {
            e.printStackTrace();
        }
        return false;
    }

}

10.使用pull解析xml格式的數據

    public static <T> List<T> xml2Bean(String xmlFileName, Class<? extends T> clazz, Context context) {
        Field[] fields = clazz.getDeclaredFields();
        ArrayList<T> list = null;
        T t = null;
        try {
            XmlPullParser pull = Xml.newPullParser();
            pull.setInput(context.openFileInput(xmlFileName), "UTF-8");
            int type = pull.getEventType();
            while (type != XmlPullParser.END_DOCUMENT) {
                String tagName = pull.getName();
                switch (type) {
                    case XmlPullParser.START_TAG:
                        if (tagName.equals(clazz.getName() + "es")) { //list標簽開始
                            list = new ArrayList<T>();
                        } else if (tagName.equals(clazz.getName())) { //對象標簽開始
                            t = clazz.newInstance();
                        } else { //屬性標簽開始
                            for (Field field : fields) {
                                if (!Modifier.isStatic(field.getModifiers())) {
                                    if (tagName.equals(field.getName())) {
                                        field.setAccessible(true);
                                        field.set(t, pull.nextText());
                                    }
                                }
                            }
                        }
                        break;
                    case XmlPullParser.END_TAG:
                        if (tagName.equals(clazz.getName())) {
                            list.add(t);
                        }
                        break;
                    default:
                        break;
                }
                type = pull.next();
            }
        } catch (IOException | XmlPullParserException | InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
        }
        return list;
    }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,698評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,202評論 3 426
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,742評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,580評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,297評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,688評論 1 327
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,693評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,875評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,438評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,183評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,384評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,931評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,612評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,022評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,297評論 1 292
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,093評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,330評論 2 377

推薦閱讀更多精彩內容