二、常用布局&存储

数据的存储

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阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容