數據的存儲
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;

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;
}