Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> 二、測試,Logcat,數據存儲,文件權限,SharedPreferences,XML生成和解析

二、測試,Logcat,數據存儲,文件權限,SharedPreferences,XML生成和解析

編輯:關於Android編程

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;
    
3.需要在androidmanifest.xml中的manifest節點下配置一個instrumentation;targetPackage:需要測試的工程的包名。
    
4.如果不知道怎麼配置androidmanifest.xml,可以新建一個android test project工程,會自動配置.
* ADB無法連接時,可以查找5037對應的進程,可能一個也可能幾個,殺死後重啟adb,還是不行的話嘗試重啟編譯器

3.Logcat日志貓工具的使用

    包括五種級別,可以添加過濾器過濾日志信息。能夠幫助我們觀察程序運行的狀態。
    e:
    w:
    i:
    d:
    v:
    在公司開發中一般打印日志用Log類,通常會封裝一個LogUtils,通過開關來控制日志信息的打印。

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

1.寫布局
    LinearLayout + RelativeLayout
2.寫業務邏輯
    a.找到相應控件
    b.設置按鈕的點擊事件
    c.在onclick方法中,獲取用戶輸入的用戶名密碼和是否記住密碼
    d.判斷用戶名密碼是否為空,不為空請求服務器(省略,默認請求成功)
    e.判斷是否記住密碼,如果記住,將用戶名密碼保存本地。???? 
    f.回顯用戶名密碼 ??
    // 通過context對象獲取私有目錄(/data/data/packagename/filse)
    context.getFileDir().getPath()

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

使用Sdcard注意事項:
1.權限問題: 
    
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();
                是一個外部存儲目錄,只用應用聲明了的一個權限,就可以訪問讀寫sdcard目錄;所以一般用來存放一些安全性不高的數據,文件size比較大的數據。

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:其他用戶對該文件的權限

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: 默認值(KEY中取不到值時的默認返回值)
        String username = sharedPreferences.getString("username", "");
        String password = sharedPreferences.getString("password", "");

獲取默認的sharepreferences(文件名稱是默認的)的方式:通過PreferenceManager
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);

9 生成xml的2種方式

1.寫布局
2.業務邏輯
    a.備份
        1.封裝短信數據到list中
        2.將list中的數據寫到xml文件中。
    b.恢復
        1.解析xml文件中短信數據,封裝到list集合中
        2.將解析數據打印。
// 使用XmlSerializer來序列化xml文件
public static boolean backupSms_android(Context context){   
    try{
        // 0.獲取短信數據
        ArrayList allSms = SmsDao.getAllSms();
        // 1.通過Xml獲取一個XmlSerializer對象
        XmlSerializer xs = Xml.newSerializer();
        // 2.設置XmlSerializer的一些參數,比如:設置xml寫入到哪個文件中
            // os:xml文件寫入流   encoding:流的編碼
        xs.setOutput(context.openFileOutput("backupsms2.xml", Context.MODE_PRIVATE), "utf-8");
        // 3.序列化一個xml的聲明頭
            //encoding:xml文件的編碼  standalone:是否獨立
        xs.startDocument("utf-8", true);
        // 4.序列化一個根節點的開始節點
        //namespace:命名空間  name: 標簽的名稱
        xs.startTag(null, "Smss");
        // 5.循環遍歷list集合序列化一條條短信
            for (SmsBean smsBean : allSms) {
                xs.startTag(null, "Sms");
                //name:屬性的名稱  value:屬性值
                xs.attribute(null, "id", smsBean.id+"");
                xs.startTag(null, "num");
                //寫一個標簽的內容
                xs.text(smsBean.num);
                xs.endTag(null, "num");
                xs.startTag(null, "msg");
                xs.text(smsBean.msg);
                xs.endTag(null, "msg");
                xs.startTag(null, "date");
                xs.text(smsBean.date);
                xs.endTag(null, "date");
                xs.endTag(null, "Sms");
            }
        // 6.序列化一個根節點的結束節點
            xs.endTag(null, "Smss");
        // 7.將xml寫入到文件中,完成xml的序列化
            xs.endDocument();
            return true;
    }catch (Exception e) {
        e.printStackTrace();
    }
    return false;
}

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

dom解析:基於全文加載的解析方式(把文檔中所有內容都封裝成對象)
    * Dom的優勢:將文檔結構(所有內容)都以對象的形式保留在了內存中,我們可以對內存中的(Dom樹)進行增刪改查操作並且操作很方便
    * Dom的劣勢:因為保留了全部文檔內容,資源消耗比較大
sax解析:基於事件的逐行解析方式(事件驅動:從源文件中挑出特定節點的數據返回,中途不可控)
    * 優點: 在讀取xml文檔時,已經根據定義好的事件,對xml內容進行了篩選,解析完成後,內存中只保留了我們想要的內容,比較節約資源,在資源比較匮乏的平台使用
    * 缺點: 沒有保留文檔的結構,無法進行增刪改的操作
pull解析:同sax(中途可控)
    與Sax一樣.都屬於事件驅動的解析方式,相比Sax解析過程更加靈活,sax一旦開始解析就是從頭讀到尾,不解析完整個文檔不會停。而pull解析較為靈活,是以事件為單位手動向下繼續,如果獲得到我們要找的內容可以停止繼續解析.

// 解析xml文件讀取短信內容
public static int restoreSms(Context context) {
    ArrayList arrayList = null;
    SmsBean smsBean = null;
    try{
        // 1.通過Xml獲取一個XmlPullParser對象
        XmlPullParser xpp = Xml.newPullParser();
        // 2.設置XmlPullParser對象的參數,需要解析的是哪個xml文件,設置一個文件讀取流
            // 通過context獲取一個資產管理者對象
        AssetManager assets = context.getAssets();
            // 通過資產管理者對象能獲取一個文件讀取流
        InputStream inputStream = assets.open("backupsms.xml");
        xpp.setInput(inputStream,"utf-8");
            // xpp.setInput(context.openFileInput("backupsms2.xml"), "utf-8");
        // 3.獲取當前xml行的事件類型
        int type = xpp.getEventType();
        // 4.判斷事件類型是否是文檔結束的事件類型
        while(type != XmlPullParser.END_DOCUMENT){
            // 5.如果不是,循環遍歷解析每一行的數據。解析一行後,獲取下一行的事件類型
            String currentTagName = xpp.getName();
                // 判斷當前行的事件類型是開始標簽還是結束標簽
            switch (type) {
            case XmlPullParser.START_TAG:
                if(currentTagName.equals("Smss")){
                    // 如果當前標簽是Smss,需要初始化一個集合
                    arrayList = new ArrayList();
                }else if(currentTagName.equals("Sms")){
                    smsBean = new SmsBean();
                    smsBean.id = Integer.valueOf(xpp.getAttributeValue(null, "id"));
                }else if(currentTagName.equals("num")){
                    smsBean.num =  xpp.nextText();
                }else if(currentTagName.equals("msg")){
                    smsBean.msg =  xpp.nextText();
                }else if(currentTagName.equals("date")){
                    smsBean.date =  xpp.nextText();
                }
                break;
            case XmlPullParser.END_TAG:
                // 當前結束標簽是Sms的話,一條短信數據封裝完成, 可以加入list中
                if(currentTagName.equals("Sms")){
                    arrayList.add(smsBean);
                }
                break;
            default:
                break;
            }
            type = xpp.next();//獲取下一行的事件類型
        }
        return arrayList.size();
    }catch (Exception e) {
        e.printStackTrace();
    }
    return 0;
}
  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved