SharedPreferences自动登录及存储多组相同类型数据

本文主要是介绍SharedPreferences自动登录及存储多组相同类型数据,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


SharedPreferences自动登录及存储多组相同类型数据

继续 我们的面试题,本文主要说俩个问题:

  • 登录时怎么保存用户名密码实现下次自动登录
  • 如果sp只存储用户名,比如三个用户都存在sp里,取出来怎么取?存进去怎么存?你怎么区分
登录时怎么保存用户名密码实现下次自动登录

首先,我们要实现记住密码的功能,在Android中,轻量级的数据存储一般使用SharedPreferences,当然你也可以使用其他的数据存储方式。SharedPreferences实现记住密码功能一般遵循以下步骤:

  1. 每次登录时,都检测上次登录是否记住了密码,如果记住了就要从sp里读取账户信息,并自动填写到输入框里
  2. 登录成功后,要检测本次登录是否选择了记住密码,如果为true,则需要把本次登录的帐号信息写到sp里,如果为false,则需要清空sp的key-value

另外,这里只所以说记住密码,而没有说记住用户名和密码,我相信这个逻辑大家都可以理得通,没有用户名的密码是无意义的,记住密码的言外之意就是记住用户名和密码。

activity_login.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:paddingBottom="@dimen/activity_vertical_margin"android:paddingLeft="@dimen/activity_horizontal_margin"android:paddingRight="@dimen/activity_horizontal_margin"android:paddingTop="@dimen/activity_vertical_margin"tools:context="com.android.test.LoginActivity"><EditTextandroid:id="@+id/username"android:layout_width="match_parent"android:layout_height="48dip"android:layout_marginTop="8dip"android:hint="输入用户名" /><EditTextandroid:id="@+id/password"android:layout_width="match_parent"android:layout_height="48dip"android:layout_marginTop="8dip"android:hint="输入密码" /><CheckBoxandroid:id="@+id/remembers_password"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="8dip"android:text="记住密码" /><Buttonandroid:id="@+id/login"android:layout_width="match_parent"android:layout_height="48dip"android:layout_marginTop="8dip"android:onClick="exeLogin"android:text="登录" />
</LinearLayout>

QQ截图20160620143545.png
public class LoginActivity extends AppCompatActivity {// 假设服务器里有这个账户信息private static final String U_name = "zhangsan";private static final String U_pswd = "zhang@123";private EditText username, password;private CheckBox remembers_password;private Button login;private SharedPreferences sp;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_login);username = (EditText) findViewById(R.id.username);password = (EditText) findViewById(R.id.password);remembers_password = (CheckBox) findViewById(R.id.remembers_password);login = (Button) findViewById(R.id.login);sp = PreferenceManager.getDefaultSharedPreferences(this);String name = sp.getString("USERNAME", "");String pass = sp.getString("PASSWORD", "");boolean rememberPassword = sp.getBoolean("REMEBERS_PASSWORD", false);//如果上次选了记住密码,那么本次登录会自动填写用户名和密码if (rememberPassword) {username.setText(name);password.setText(pass);remembers_password.setChecked(true);}}// 点击登录按钮public void exeLogin(View view) {String inputUsername = username.getText().toString();String inputUserpswd = password.getText().toString();if (inputUsername.equals(U_name) && inputUserpswd.equals(U_pswd)) {Toast.makeText(LoginActivity.this, "登录成功", Toast.LENGTH_LONG).show();//是否记住密码if(remembers_password.isChecked()){// 保存信息sp.edit().putBoolean("REMEBERS_PASSWORD", true).commit();sp.edit().putString("USERNAME", inputUsername).commit();sp.edit().putString("PASSWORD", inputUserpswd).commit();}else{// 注意,如果本次登录不记住密码的话,需要把sp里的信息清空,当然这里也可以直接删除sp里key-valuesp.edit().putBoolean("REMEBERS_PASSWORD", false).commit();sp.edit().putString("USERNAME", "").commit();sp.edit().putString("PASSWORD", "").commit();}startActivity(new Intent(LoginActivity.this,LoginSucActivity.class));} else {Toast.makeText(LoginActivity.this, "登录失败", Toast.LENGTH_LONG).show();}}
}

以上,我们实现了记住密码功能,接下来就要在此基础上实现自动登录的功能,只需要:

        if (rememberPassword) {username.setText(name);password.setText(pass);remembers_password.setChecked(true);}
在此处增加以下代码:if(remebersLoginSelf){loginself.setChecked(true);exeLogin(login);}

然后,

            //是否记住密码if (remembers_password.isChecked()) {// 保存信息sp.edit().putBoolean("REMEBERS_PASSWORD", true).commit();sp.edit().putString("USERNAME", inputUsername).commit();sp.edit().putString("PASSWORD", inputUserpswd).commit();} else {// 注意,如果本次登录不记住密码的话,需要把sp里的信息清空,当然这里也可以直接删除sp里key-valuesp.edit().putBoolean("REMEBERS_PASSWORD", false).commit();sp.edit().putString("USERNAME", "").commit();sp.edit().putString("PASSWORD", "").commit();}
在此处增加以下代码:if (loginself.isChecked()){sp.edit().putBoolean("LOGIN_SELF", true).commit();}else{sp.edit().putBoolean("LOGIN_SELF", false).commit();}

以上代码在 百度云-自动登录

当然,这里只是个demo,更细致的逻辑还需要你自己去处理,比如:自动登录的前提就是记住密码等。

但是,利用sp虽然实现了我们自动登录的需求,但是实际上一般我们不会这么做,因为这样做存在一定的安全隐患,我们可以在sp中存取数据时进行加密解密,这样的话安全性会相对高一些。当然这里还有更为安全的一种做法:TOKEN思想,也就是说我们在登录的时候,服务器给我们返回一个token值,我们把这个值保存到起来,下次请求接口时,带上这个 token ,服务端判断这个token是否有效或过期,返回对应的错误码,如果本地不存在 token,就是没有登录过,跳转登陆。

如果sp只存储用户名,比如三个用户都存在sp里,取出来怎么取?存进去怎么存?你怎么区分 ?

利用SharedPreferences存储数据,首先我们要获取一个 SharedPreferences实例,我通常一般会使用PreferenceManager.getDefaultSharedPreferences(this);来获取这个实例, 我们知道SharedPreferences有如下操作模式:

  • mode指定为MODE_PRIVATE(或0),则该配置文件只能被自己的应用程序访问。
  • mode指定为MODE_WORLD_READABLE,则该配置文件除了自己访问外还可以被其它应该程序读取。
  • mode指定为MODE_WORLD_WRITEABLE,则该配置文件除了自己访问外还可以被其它应该程序读取和写入

其实,除了以上获取SharedPreferences实例的方法外,目前android还气功了其他获取SharedPreferences实例的方法,细看源码就可以比较出他们的不同之处:

  1. sp = Context.getSharedPreferences("sp_name",0);
    得到一个名为sp_name的实例,sp_name为本组件的配置文件名( 自己定义,也就是一个文件名),当这个文件不存在时,直接创建,如果已经存在,则直接使用,mode为操作模式,默认的模式为0或MODE_PRIVATE,
  2. sp = Activity.getPreferences(0);
    得到一个名为activity名的实例,其中配置文件仅可以被调用的Activity使用
  3. sp = PreferenceManager.getDefaultSharedPreferences(this);
    等到一个名为:yourpackageName_preferences的示例

发现什么了吗?回到我们的问题:“如果sp只存储用户名,比如三个用户都存在sp里,取出来怎么取?存进去怎么存?你怎么区分 ?”。第一个方法说“得到一个名为sp_name的实例”,也就是说我你们可以自定义SharedPreferences的文件名,更直白点说就是我们可以把不同组合的数据存储到不同的文件中去,这样即使有多个“user_name”的key,也不会影响我们的存取,因为他们虽然key值相同,但所处的文件不同,也就是此key非彼key。

================================================
上一篇:RecycleView
上上篇:ListView及本组所有面试题
更多内容请关注 我的专题 
转载请注明 原文链接: 
http://www.jianshu.com/users/c1b4a5542220/latest_articles

这篇关于SharedPreferences自动登录及存储多组相同类型数据的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/983952

相关文章

Security OAuth2 单点登录流程

单点登录(英语:Single sign-on,缩写为 SSO),又译为单一签入,一种对于许多相互关连,但是又是各自独立的软件系统,提供访问控制的属性。当拥有这项属性时,当用户登录时,就可以获取所有系统的访问权限,不用对每个单一系统都逐一登录。这项功能通常是以轻型目录访问协议(LDAP)来实现,在服务器上会将用户信息存储到LDAP数据库中。相同的,单一注销(single sign-off)就是指

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

关于数据埋点,你需要了解这些基本知识

产品汪每天都在和数据打交道,你知道数据来自哪里吗? 移动app端内的用户行为数据大多来自埋点,了解一些埋点知识,能和数据分析师、技术侃大山,参与到前期的数据采集,更重要是让最终的埋点数据能为我所用,否则可怜巴巴等上几个月是常有的事。   埋点类型 根据埋点方式,可以区分为: 手动埋点半自动埋点全自动埋点 秉承“任何事物都有两面性”的道理:自动程度高的,能解决通用统计,便于统一化管理,但个性化定

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

Hadoop集群数据均衡之磁盘间数据均衡

生产环境,由于硬盘空间不足,往往需要增加一块硬盘。刚加载的硬盘没有数据时,可以执行磁盘数据均衡命令。(Hadoop3.x新特性) plan后面带的节点的名字必须是已经存在的,并且是需要均衡的节点。 如果节点不存在,会报如下错误: 如果节点只有一个硬盘的话,不会创建均衡计划: (1)生成均衡计划 hdfs diskbalancer -plan hadoop102 (2)执行均衡计划 hd

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

spoj705( 求不相同的子串个数)

题意:求串s的不同子串的个数 解题思路:任何子串都是某个后缀的前缀,对n个后缀排序,求某个后缀的前缀的个数,减去height[i](第i个后缀与第i-1 个后缀有相同的height[i]个前缀)。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstrin