文章目录
一、SharedPreferences概述
实际上sp数据也是存在data/data/包名下面,他会创建一个特殊类型的文件,然后再往里面写
SharedPreferences是Android平台上一个轻量级的存储辅助类,用来保存应用的一些常用配置,它提供了string,set, int, long, float, boolean六种数据类型。最终数据是以xm1形式进行存储。在应用中通常做一些简单数据的持久化缓存。
sp比较特殊的地方在于,它不是以一种txt的形式存储,他是以xml的形式来存储,我们之前将网络请求的时候就有讲到xml的形式来传输数据,只是因为后来json的流行,所以xml就逐渐被淘汰了,那么在安卓里面,xml本地存储还没有被淘汰,而SharedPreferences本质上就是一种xml的形式来存储,存储的方式可以来存一些安卓的基本类型, int, long, float, boolean类型,也包括string,set类型
下面举一个sp应用的场景
比如说在我们qq登录的时候,第一次登录成功了,第二次就会直接进入主界面了,不会再进到登录界面,实际上当我们第一次登录成功的时候,系统会把我们登陆的状态,可以用一个boolean类型的值保存到本地,登陆成功就可以保存为ture,保存到本地的形式可以用我们上述讲到的文件,但是文件存储可能还要对里面的信息进行解析,我们这里就有这么一种更加方便的方式就是接下来所要讲到的sp,sp主要是来存储这种少量的数据
,而且优势就是sp读写起来会比文件更加方便,所以像这种情况我们就可以用sp的形式来存储
二、SharedPreferences的使用方法
因为sp是存储在data/data/包名目录下的,所以我们不需要动态申请权限
1.添加button并添加点击事件
核心代码
button.findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
}
});
2.1写数据
1.activity代码(拿到SharedPreferences,再添加数据)
package com.hnucm.a_test13;
import androidx.appcompat.app.AppCompatActivity;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
public class MainActivity extends AppCompatActivity {
Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button=findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
SharedPreferences sharedPreferences=getSharedPreferences("user",MODE_PRIVATE);//拿到 sharedPreferences 的引用 MODE_PRIVATE表示只能被本应用读写,不能被其他应用读写
SharedPreferences.Editor editor=sharedPreferences.edit();//拿到编辑对象
editor.putBoolean("isLogin",true);//添加数据
editor.commit();
}
});
}
}
2.运行,点击按钮,刷新一下data/data/包名目录,会发现我们的数据被保存到了一个文件夹
2.2读数据
1.核心代码
package com.hnucm.a_test13;
import androidx.appcompat.app.AppCompatActivity;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
public class MainActivity extends AppCompatActivity {
Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button=findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
SharedPreferences sharedPreferences=getSharedPreferences("user",MODE_PRIVATE);//拿到 sharedPreferences 的引用 MODE_PRIVATE表示只能被本应用读写,不能被其他应用读写
boolean isLogin = sharedPreferences.getBoolean("isLogin", false);//当key不存在时,默认返回defValue的值
Log.i("MainActivity","isLogin "+isLogin);
Toast.makeText(MainActivity.this,""+isLogin,Toast.LENGTH_SHORT).show();
}
});
}
}
2.运行结果
三、SharedPreferences案例-qq登录demo
需求分析:qq登录第一次登录需要输入账号密码,如果登录成功,第二次启动qq可以直接跳到主界面,不需要先跳转到登录页面
3.1第一步,创建activity
我们需要先创建三个activity
因为我们需要一个启动跳转页面,一个登录页面,一个主界面,所以需要三个页面
1.默认第一个见到的页面是MainActivity,我们需要把LaunchActivity设置为我们第一个见到的activity
2.并且去掉应用的标题框
3.2第二步,设置启动跳转页面
1.给启动跳转页面设置背景图片(图片如下)
2.并且设置延时跳转(需要判断用户是否登录过)
3.LaunchActivity代码
package com.hnucm.a_test13;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import android.annotation.SuppressLint;
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
public class LaunchActivity extends AppCompatActivity {
@SuppressLint("HandlerLeak")
Handler mHandler=new Handler(){
@Override
public void handleMessage(@NonNull Message msg) {
super.handleMessage(msg);
SharedPreferences sharedPreferences= getSharedPreferences("user", MODE_PRIVATE);
boolean islogin = sharedPreferences.getBoolean("islogin",false);//如果读不到文件或者当key不存在时 返回false
//判断用户是否登录过
if (islogin) { //如果登录过 跳转到 MainActivity
Intent intent=new Intent(LaunchActivity.this,MainActivity.class);
startActivity(intent);
}else { //如果没有登录过 跳转到 LoginActivity
Intent intent=new Intent(LaunchActivity.this,LoginActivity.class);
startActivity(intent);
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_launch);
mHandler.sendEmptyMessageDelayed(1,3000);//3秒后发送消息
}
}
4.运行结果
3.3第三步,设置登录界面
1.(简单设置一下)
核心代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".LoginActivity">
<TextView
android:id="@+id/textView2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="148dp"
android:text="这是登录界面"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<Button
android:id="@+id/button2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="124dp"
android:text="登录"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/textView2" />
</androidx.constraintlayout.widget.ConstraintLayout>
2.再到LoginActivity,设置登录界面点击按钮返回登录状态到文件
核心代码
package com.hnucm.a_test13;
import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
public class LoginActivity extends AppCompatActivity {
Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
button=findViewById(R.id.button2);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//点击按钮 代表登录成功
//获取SharedPreferences对象
SharedPreferences sharedPreferences=getSharedPreferences("user",MODE_PRIVATE);
//获取Editor对象的引用
SharedPreferences.Editor editor=sharedPreferences.edit();
//将登录的状态放入文件
editor.putBoolean("islogin",true);
editor.putString("username","dd");
//提交数据
editor.commit();
Intent intent=new Intent(LoginActivity.this,MainActivity.class);
startActivity(intent);
}
});
}
}
3.主界面页面
MainActivity代码
package com.hnucm.a_test13;
import androidx.appcompat.app.AppCompatActivity;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
public class MainActivity extends AppCompatActivity {
Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
SharedPreferences sharedPreferences= getSharedPreferences("user", MODE_PRIVATE);
String username=sharedPreferences.getString("username","不存在");
Toast.makeText(MainActivity.this,""+username,Toast.LENGTH_SHORT).show();
}
}
可以看到第一次点击按钮之后跳转到登录页面,第二次直接进入主界面
3.4存在的两个问题
问题一
主界面点击返回,会返回到上一个页面并不会直接退出
在之前的博客中有讲过,activity是存在任务栈的,启动过几个activity都会存在栈里,需要finish()方法
1.在LaunchActivity中加入finsh方法
2.在LoginActivity中也加入finish方法
3.点击运行看最终结果
问题二
每次进入程序会有一小段时间 程序出现空白页面
1.问题分析:
2.解决方案:自定义主题
<style name="Theme.A_Test13" parent="Theme.MaterialComponents.DayNight.NoActionBar">
<item name= "android:windowNoTitle">true</item>//无标题.
<item name= "android:windowFullscreen">true</item>//全屏即无通知栏
<item name="android:windowContentOverlay" >@null</item>//是否有遮盖
<item name="android:windowBackground" >@drawable/qq</item>
</style>
3.然后在mainifests给第一个界面定义主题
4.运行(会有明显改善)
本讲关于SharedPreferences的知识就讲到这里啦,是不是又掌握了一个小技巧,谢谢您的阅读