一.存储方式分类:SharedPreferences存储

二.SharedPreferences存储

1.特点

①存储单一数据,例如数值,字符串,布尔

②文件:/date/date/包名/shared_prefs/xxx.xml: value

③以键值对的形式存储

④可以设置不被其他应用操作

2.API

(1)SharedPreferences

①获取实例context.getSharedPreferences():

1)name 存储文件名; 2)mode 操作模式:MODE_PRIVATE不能被别的应用访问,覆盖模式;MODE_APPEND不能被别的应用访问,追加模式;

②启动编辑器:edit():返回Editor

③读取Value:1)getString(key,defValue 缺省值) 2)getAll( ) 返回所有键值对的 Map集合

(2)Editor

①存放数据:1)putString(key,value) 2)putLong(key,value) 3)putInt 4)putFloat 5)putBoolean 6)putStringSet

②提交存储

③clear() 清除

④remove(String key) 移除指令key的键值对

下面请允许我盗一张图

下面开始我的表演

看样子中午还是要休息一下,打疲劳战队身体没好处,而且写的代码质量也不高!

场景一

首先我要实现的功能就是简单的注销用户,当然方法有很多,你可以通过EventBus或者RXBus注册订阅者然后发布一个事件出去,另外一种就是通过后台移除session,清除当前的sesssionid 因为相信大家都知道,客户端只有一个Cookie,Cookie对应服务端Session的key,首先登录的时候服务端返回一个sessionid给我然后我通过sp把拿到的SessionId保存起来,然后在点击退出当前用户的时候把之前登录的SessionId提交过去与服务端那边对比,如果对比一致说明我已经登录过了,然后这时候服务端移除session,返回给我,提示成功退出!

听起来好像思路有点清晰,下面开始我的表演Action

因为我这边提交的都是json格式的数据下面是点击到登录按钮提交的json数据这时候服务器给我返回sid

        Gson gson = new Gson();
        final HashMap<String, String> params = new HashMap<>();
        params.put("username", "18388483434");
        params.put("pass", "123456");
        final String strEntity = gson.toJson(params);
        RequestBody body = RequestBody.create(MediaType.parse("application/json;charset=UTF-8"), strEntity);
        AccountService account = RetrofitApiManager.getInstance().getService(AccountService.class);
        Call<LoginResponse> call = account.login(body);
        final String sessionId=null;
        call.enqueue(new Callback<LoginResponse>() {
            @Override
            public void onResponse(Call<LoginResponse> call, Response<LoginResponse> response) {
                if (response.isSuccessful()) {
                    if (sp == null) {
                        sp = getSharedPreferences("exitsid",MODE_PRIVATE);
                    }
                    SharedPreferences.Editor editor = sp.edit();
                    //sessionid
                    editor.putString("sid", sessionId);
                    //异步提交

                    Toast.makeText(LoginActivity.this,  response.body().getMessage(), Toast.LENGTH_SHORT).show();
//                    Log.i(TAG, "onResponse: " + response.body().getCode() + response.body().getMessage());
                    startActivity(new Intent(LoginActivity.this, MainActivity.class));
                    editor.apply();
                }
            }

            @Override
            public void onFailure(Call<LoginResponse> call, Throwable t) {
                Log.e(TAG, "onFailure: " + t.getMessage());

            }
        });

登录抓包信息如下Fiddler

请求的接口地址因为这里提交的是json数据所以这里添加一个请求头信息,如果你自己觉得太死板,请自定义拦截器,这个没有问题,根据自己的需求来。

    @Headers("Content-type:application/json;charset=UTF-8")
    @POST("xxx/login")
    Call<LoginResponse> login(@Body RequestBody body);

下面是我的响应bean我这里直接GsonFormat生成

package com.visoport.medicine.http.api.response;

/**
 * 登录服务器返回LoginResponse
 */

public class LoginResponse {

    private int code;
    private DataBean data;
    private String message;

    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }

    public DataBean getData() {
        return data;
    }

    public void setData(DataBean data) {
        this.data = data;
    }

    public String getMessage() {
        return message;
    }

    public void setMessage(String message) {
        this.message = message;
    }

    public static class DataBean {

        private String sid;

        public String getSid() {
            return sid;
        }

        public void setSid(String sid) {
            this.sid = sid;
        }
    }
}

下面是注销用户

请求接口loginout

 //退出当前用户
    @Headers("Content-type:application/json;charset=UTF-8")
    @POST("xxx/login/logout")
    Call<ExitUserResponse> logout(@Body RequestBody body);

服务端响应的bean跟上面的LoginResponse一致同样是GsonFormat生成


/**
 * 退出登录
 */

public class ExitUserResponse {

    /**
     * code : 1200
     * data : {"sid":"67b4edd5346c447afe7b9cd0ef"}
     * message : 成功退出
     */

    private int code;
    private DataBean data;
    private String message;

    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }

    public DataBean getData() {
        return data;
    }

    public void setData(DataBean data) {
        this.data = data;
    }

    public String getMessage() {
        return message;
    }

    public void setMessage(String message) {
        this.message = message;
    }

    public static class DataBean {

        private String sid;

        public String getSid() {
            return sid;
        }

        public void setSid(String sid) {
            this.sid = sid;
        }
    }
}

然后点击退出按钮 代码很乱后期整理

              final String sessionId;
                if (sp == null) {
                    sp = getSharedPreferences("exitsid",MODE_PRIVATE);
                }
                //后面是默认值defaultvalue
                sessionId = sp.getString("sid", "");
                Gson gson = new Gson();
                final HashMap<String, String> params = new HashMap<>();
                params.put("sid", sessionId);
                Log.i(TAG, "onClick: "+sessionId);
                String strEntity = gson.toJson(params);
                RequestBody body = RequestBody.create(MediaType.parse("application/json;charset=UTF-8"), strEntity);
                AccountService account = RetrofitApiManager.getInstance().getService(AccountService.class);
                Call<ExitUserResponse> call = account.logout(body);
                call.enqueue(new Callback<ExitUserResponse>() {
                    @Override
                    public void onResponse(Call<ExitUserResponse> call, Response<ExitUserResponse> response) {
                        if(response.isSuccessful())
                        {
                            Toast.makeText(SettingActivity.this,response.body().getMessage(),Toast.LENGTH_SHORT).show();
                            startActivity(new Intent(SettingActivity.this,LoginActivity.class));
                        }
                    }
                    @Override
                    public void onFailure(Call<ExitUserResponse> call, Throwable t) {
                        Toast.makeText(SettingActivity.this,t.getMessage(),Toast.LENGTH_SHORT).show();
                    }
                });

我去,什么鬼,缺少参数,什么参数,为什么会这样,不就是一个sessionid吗?好吧我承认我菜,那断点试试,工欲善其事,必先利其器,那就debug走起不好意思debug今天出了点问题,哈哈!

其实就是自己不小心把那句代码注释掉了手贱!

String sessionId = response.body().getData().getSid();

因为之前的sessionId初始化就是null你怎么可能拿得到吗?后面的读取时没有问题的!狠狠的给自己一巴掌!

final String sessionId=null;

下面直接看最后的表演图

总结:

1、 还是基本功好扎实,首先了解sp简单用法包括存储 写入文件 以及读取文件等

2、出现问题学会debug慢慢调试 看程序的走势进一步找出问题所在

3、必要的时候使用抓包神器Fildder那么问题迎刃而解

[置顶] Android 关于SP读取与存储正确打开方式?的更多相关文章

  1. [置顶] Android开发笔记(成长轨迹)

    分类: 开发学习笔记2013-06-21 09:44 26043人阅读 评论(5) 收藏 Android开发笔记 1.控制台输出:called unimplemented OpenGL ES API ...

  2. [置顶] Android应用开发之版本更新你莫愁

    传送门 ☞ 轮子的专栏 ☞ 转载请注明 ☞ http://blog.csdn.net/leverage_1229 今天我们学习如何实现Android应用的自动更新版本功能,这是在各种语言编写的应用中都 ...

  3. [置顶] Android 状态栏那些小坑?

    背景:因为之前老板上次问我我们的app能不能自定义上面的状态栏我说可以啊!当时没管,今天试了下果然很多坑,之前github上也有很多大佬写了一个开源库有兴趣的可以点进去看下支持DrawLayout沉侵 ...

  4. [置顶] Android系统五大布局详解Layout

    我们知道Android系统应用程序一般是由多个Activity组成,而这些Activity以视图的形式展现在我们面前,视图都是由一个一个的组件构成的.组件就是我们常见的Button.TextEdit等 ...

  5. 【Android开发笔记】返回上层Activity的正确打开方式

    技术支持 http://stackoverflow.com/questions/12276027/how-can-i-return-to-a-parent-activity-correctly 首先, ...

  6. [置顶] android之存储篇_SQLite数据库_让你彻底学会SQLite的使用

    SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段声明的数据类型是什么. 例如:可以在Integer类型的字段中存放字符串,或者在布尔型字段中存放浮点数,或者在字符型字段中 ...

  7. [置顶] Android开发之ProcessState和IPCThreadState类分析

    在Android中ProcessState是客户端和服务端公共的部分,作为Binder通信的基础,ProcessState是一个singleton类,每个 进程只有一个对象,这个对象负责打开Binde ...

  8. [置顶] Android开发之MediaPlayerService服务详解(一)

    前面一节我们分析了Binder通信相关的两个重要类:ProcessState 和 IPCThreadState.ProcessState负责打开Binder 驱动,每个进程只有一个.而 IPCThre ...

  9. Android:Xml(读取与存储)

    1.读取XML文件 参数xml是建含xml数据的输入流,List<Person> persons用于存储xml流中的数据. XmlPullParser类的几个方法:next(),nextT ...

随机推荐

  1. 20145201李子璇《网络对抗》逆向及Bof基础实践

    20145201李子璇<网络对抗>逆向及Bof基础实践 实践目标 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回 ...

  2. mongodb 索引的创建

    mongodb 创建常用的创建索引的有 signle Field Indexes Compound multikey,创建索引就是按照索引字段把documnet 进行排序,索引只存储了document ...

  3. JasperReports实现报表调出excel

    一.利用工具iReport 创建task.jrxml 模板 并生成 task.jasper 文件 二.搭建工程导入以下jar包 commons-beanutils-1.9.2.jar commons- ...

  4. LeetCode——Longest Palindromic Subsequence

    1. Question Given a string s, find the longest palindromic subsequence's length in s. You may assume ...

  5. Educational Codeforces Round 13 A、B、C、D

    A. Johny Likes Numbers time limit per test 0.5 seconds memory limit per test 256 megabytes input sta ...

  6. vue2.0--vue-router路由

    一.vue-router如何进行参数传递 1.name  $route.name 通过在路由router/index.js中配置路由时定义的name属性来传递 ① 有一个页面components/he ...

  7. 转载:Chrome 控制台不完全指南

    Chrome的开发者工具已经强大到没朋友的地步了,特别是其功能丰富界面友好的console,使用得当可以有如下功效: 更高「逼格」更快「开发调试」更强「进阶级的Frontender」 Bug无处遁形「 ...

  8. tags

    运行tags在你的.vimrc 中加一个路径,set tags=/home/lh/1407k/arm/tags 注意此文件下的东西要注销必须用“执行一个ctags -R *;ctrl + ]过去,ct ...

  9. Python打包分发工具setuptools简介(转)

    作为Python标准的打包及分发工具,setuptools可以说相当地简单易用.它会随着Python一起安装在你的机器上.你只需写一个简短的setup.py安装文件,就可以将你的Python应用打包. ...

  10. Codeforces Round #412

    第一题水题,8分钟1a #include<map> #include<set> #include<cmath> #include<queue> #inc ...