这篇文章主要是巩固一下前面所学的SharedPreferences存储数据以及广播接收器。如果对这两个部分不熟悉的话可以先看一下Android数据存储(二)以及Android广播接收器(二)由于这篇文章的代码就多,所以就在文章里只展示了比较重要的部分代码。完整代码在github地址。程序运行效果如下图

Activity的管理

为了更好的管理Activity,先创建一个ActivityCollector类用于管理所有的Activity。代码如下

public class ActivityCollector {

public static List activities = new ArrayList();

public static void addActivity(Activity activity) {

activities.add(activity);

}

public static void removeActivity(Activity activity){

activities.remove(activity);

}

public static void finishAll(){

for(Activity activity:activities){

if(!activity.isFinishing()){

activity.finish();

}

}

}

}

下面再创建一个BaseActivity类来作为每个Activity的父类,代码如下

public class BaseActivity extends Activity {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

ActivityCollector.addActivity(this);

}

@Override

protected void onDestroy() {

super.onDestroy();

ActivityCollector.removeActivity(this);

}

}

可以看到BaseActivity也是非常简单,继承自Activity复写onCreate()并调用ActivityCollector.addActivity()方法以及复写onDestroy()并调用ActivityCollector.removeActivity()方法。在创建每个Activity时继承BaseActivity就可以方便的管理每个Activity了。这样做的好处就是不论在哪个Activity中接收强制下线的广播时都可以轻松地finish所有的Activity,而且广播接收器也不用依附于任意一个Activity。

登陆界面

android:layout_width="match_parent"

android:layout_height="match_parent"

android:stretchColumns="1">

android:layout_height="wrap_content"

android:layout_marginLeft="10dp"

android:layout_marginTop="10dp"

android:text="Account:"

android:textSize="18sp" />

android:id="@+id/account_edit_text"

android:layout_height="wrap_content"

android:layout_marginTop="10dp"

android:hint="Input your account " />

……/>

android:id="@+id/remember_password"

android:layout_marginLeft="10dp" />

android:layout_height="wrap_content"

android:text="Remember password"

android:textSize="18sp" />

android:id="@+id/login"

android:layout_height="wrap_content"

android:layout_margin="5dp"

android:layout_span="2"

android:text="Login"

android:textAllCaps="false" />

布局界面主要通过TableLayout来实现的,第二个与第一个类似都是一个TextView和一个EditText,所以第二个就省略了。显示效果如下图所示

登陆界面布局

记住密码功能的实现,代码如下

public class LoginActivity extends BaseActivity {

private SharedPreferences preferences;

private SharedPreferences.Editor editor;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_login);

preferences = PreferenceManager.getDefaultSharedPreferences(this);

isRemember = preferences.getBoolean("rememberPassword", false);

login.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

String account = accountEditText.getText().toString();

String password = passwordEditText.getText().toString();

if (account.equals("admin") && password.equals("123456")) {

editor = preferences.edit();

if (rememberPassword.isChecked()) {

editor.putString("account", account);

editor.putString("password", password);

editor.putBoolean("rememberPassword", true);

} else {

editor.clear();

}

editor.commit();

Intent intent = new Intent(LoginActivity.this, MainActivity.class);

startActivity(intent);

finish();

} else if (account.equals("") && password.equals("")) {

Toast.makeText(LoginActivity.this,"account or password cannot be empty",Toast.LENGTH_SHORT).show();

} else {

Toast.makeText(LoginActivity.this, "account or password is invalid", Toast.LENGTH_SHORT).show();

}

}

});

if (isRemember) {

accountEditText.setText(preferences.getString("account", ""));

passwordEditText.setText(preferences.getString("password", ""));

rememberPassword.setChecked(isRemember);

}

}

}

程序一开始就实例化一个SharedPreferences对象,并从SharedPreferences文件中读取键为"rememberPassword"的值并赋值给isRemember这个变量来确定是否有记住密码。如果没读取到这个值,默认就为false即没有记住密码;如果有值并且为true的话就会执行if(isRemember)这个方法,从SharedPreferences文件中读取之前保存的账户和密码并且将数据显示到对应的EditText中。

接下来看的是login按钮的监听,首先判断账户密码是否正确,如果正确的话再判断是否勾选了记住密码。如果选中了记住密码,就会通过SharedPreferences.Editor将EditText的内容以及CheckBox的状态保存到SharedPreferences文件中;如果没有选中记住密码,就会将SharedPreferences.Editor清空。然后再通过Intent进入MainActivity。

强制下线功能的实现

在主界面中,我放置了一个Button用于发送强制下线的广播,布局就不写出来了。主要还是看如何实现强制下线。

public class MainActivity extends BaseActivity {

private Button forceOffline;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.layout_main);

forceOffline = (Button) findViewById(R.id.force_offline);

forceOffline.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

Intent intent = new Intent("com.feiben.rememberpasswordtest.FORCE_OFFLINE");

sendBroadcast(intent);

}

});

}

}

MainActivity也是十分简单,主要就是发送一个action为"com.feiben.rememberpasswordtest.FORCE_OFFLINE"的广播。想必你应该也猜到这里使用的是静态注册的方法了吧。所以需要在创建一个ForceOfflineReceiver类用来接收广播。

public class ForceOfflineReceiver extends BroadcastReceiver {

@Override

public void onReceive(final Context context, Intent intent) {

AlertDialog.Builder builder = new AlertDialog.Builder(context);

builder.setTitle("Warning");

builder.setMessage("You are forced to be offline.Please try to login again");

builder.setCancelable(false);

builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

ActivityCollector.finishAll();

Intent intent = new Intent(context,LoginActivity.class);

intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

context.startActivity(intent);

}

});

AlertDialog alertDialog = builder.create();

alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

alertDialog.show();

}

}

每当接收到广播时就会弹出一个Dialog通知用户被强制下线了,需要重新登陆。首先实例化一个AlertDialog.Builder对象,并设置其Title、Message以及setPositiveButton。这里的setCancelable(false)主要是让用户只能点击PositiveButton来取消弹窗,如果没有设置这个属性的话,用户只需要按下返回按钮就可以将Dialog取消掉并继续使用程序,很显然这并不符合正常的逻辑。

接下来在给PositiveButton设置监听器,一旦点击PositiveButton就会调用ActivityCollector.finishAll()方法将所有的Activity给Finish掉,再通过Intent来启动LoginActivity来让用户重新登陆。因为实在广播接收里启动Activity,所以需要加上intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK),具体用法可以看Android Intent Flag的介绍。最后需要将Dialog的类型设置为WindowManager.LayoutParams.TYPE_SYSTEM_ALERT才能保证弹窗的正常显示。

最后再看一眼manifest.xml中需要注意的地方

1.因为Dialog的类型设置为WindowManager.LayoutParams.TYPE_SYSTEM_ALERT所以需要加上android.permission.SYSTEM_ALERT_WINDOW这个权限。

2.将LoginActivity设置为主Activity这样程序一启动就会开启LoginActivity。

3.因为广播接收器使用的是静态注册的方法,所以在manifest进行注册。

MainActivity以及弹窗的效果如下图

下一篇文章使用Git向GitHub上传代码将介绍如何使用git将项目上传到github上。这篇文章就到这里。由于本人水平有限,如有错误,欢迎大家指正。共同学习进步!

Java实现第二次登陆强制下线_Android登陆页面记住密码以及强制下线功能的实现...相关推荐

  1. android自动登录不过登陆界面,【教程】Android 记住密码和自动登录界面的实现

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 登录界面源码: LoginActivity.java public class LoginActivity extends Activity { priv ...

  2. php自动永久登陆js,登录页面--记住密码--自动登录

    布局两个 checkbox ,一个记住密码,一个记住登录 自动登录 记住密码 2. 主要利用 jquery.cookie.js 在提交是触发 save() 函数 //记住用户名密码 function  ...

  3. Android编写一个登录界面,利用数据库实现记住密码,注册账号,强制下线,以及类似QQ的下拉列表登录功能

    首先呢,看到这么长的标题,是不是感觉这些功能有点难以实现呢,哈哈,其实并没有想象中的那么复杂,下面就跟着笔者来一起学习一下这些功能是怎么实现的吧! 1.建立一个所有活动的父类,继承自Appcompat ...

  4. 深入理解Java虚拟机——第二章——Java内存区域与内存溢出异常

    运行时数据区域 Java虚拟机运行时数据区域 程序计数器 程序计数器可以看做是当前线程所执行的字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选取下一条所需要执行的字节码指令,分支. ...

  5. java类型转换答案,在java中支持两种类型的类型转换,自动类型转换和强制类型转换。父类转化为子类需要强制转换。...

    在java中支持两种类型的类型转换,自动类型转换和强制类型转换.父类转化为子类需要强制转换. 更多相关问题 计算机病毒通过()传染扩散得极快,危害最大. 当一个现象的数量由小变大,另一个现象的数量相反 ...

  6. java 错误登陆次数_纯java代码实现登陆次数验证,登陆错误5次之后锁定30分钟

    本方法因为是根据思路纯手写,代码可以再简化,功能尝试没问题,最主要就是在登陆验证中的逻辑,checkLogin()方法是登录前的验证,而真正的登陆方式采用的是Shiro,若不是采用Shiro登陆,将该 ...

  7. python编程语言-Python有望超越Java排第二?风变编程解析编程语言新趋势

    导语:随着人工智能时代的来临,编程语言的热度居高不下,随着智能机器人的出现,在2020年更是迎来一波新的高峰. 随着人工智能时代的来临,编程语言的热度居高不下,随着智能机器人的出现,在2020年更是迎 ...

  8. java 3D 第二章 java 3D基本概念

    java 3D 第二章 java 3D基本概念 java 3D基本概念 java 3D的包及其功能 java 3D 高分辨率大尺度坐标 Java 3D场景图(Scene Graph) VirtualU ...

  9. java学习第二周周记

    JAVA学习第二周周记 **day1.**流程控制语句 ,跳转控制语句 **day2.**什么是方法?方法的重载 **day3.**数组,栈和队列 **day4.**冒泡排序及对象的引入 **day5 ...

最新文章

  1. rabbitmq系列问题解决:406, “PRECONDITION_FAILED - inequivalent arg ‘durable‘
  2. 百度java验证码不显示不出来,Java-使用百度链接时,遇到无法弹出用户登录框的问题...
  3. php post请求后端拿不到值_[精选] uniapp实现多端开发,与PHP是如何结合的
  4. java 反射机制 视频_【视频笔记】Java反射机制笔记
  5. Python str / bytes / unicode 区别详解 - Python零基础入门教程
  6. MakeDAO 推出新漏洞奖励计划,最高赏金1000万美元
  7. L1-041 寻找250-PAT团体程序设计天梯赛GPLT
  8. [logstash-input-log4j]插件使用详解
  9. Fiddler抓包一键生成代码
  10. python爬取热门新闻每日排行_用python查看百度搜索中今日热点事件排行榜
  11. 商(quotient)—— 两数之比
  12. Win32 网络编程基本函数
  13. conda: command not found
  14. [转贴]VB函数的列表
  15. 微信小程序 图片显示不完全
  16. java 按符号分割字符串_JAVA字符串按分隔符号字符串分割
  17. JavaScript之---嵌入JS代码
  18. 英文字体识别在线识别_如何查找和识别字体
  19. 知识星球的规划和落实!
  20. 大数据平台docker一键搭建

热门文章

  1. Sql字符串分组Split函数的两种实现方法
  2. SQL Server 2000从入门到精通3
  3. 【python自动化办公01】word操作-新建文档
  4. 【实战】tensorflow 花卉识别
  5. 详解Java8中流(Stream)的使用
  6. go gin gorm获取harbor项目,镜像,tag代码示例
  7. CDH6.3.2添加Hue服务时,验证数据库连接报错 Unexpected error. Unable to verify database connection.
  8. linux安装phoenix 5.1.0(对应hbase 2.2.6)
  9. Scala函数简化写法
  10. Jenkins安装与启动