一、shiro简介

     shiro是apache旗下一个开源框架,它将软件系统的安全认证相关的功能抽取出来,实现用户身份认证、权限授权、加密、会话管理等功能,组成了一个通用的安全认证框架。

以下是你可以用 Apache Shiro所做的事情:

(1)验证用户。

(2)对用户执行访问控制,如: 判断用户是否拥有角色admin,判断用户是否拥有访问的权限。

(3)在任何环境下使用Session API。例如CS程序。

(4)可以使用多个用户数据源。例如一个是oracle用户库,另外一个是mysql用户库。

(5)单点登录(SSO)功能。

(6)"Remember Me"服务,类似购物车的功能,shiro官方建议开启。

 

二、shiro架构

    

 

(1)Subject

    

     Subject 是与程序进行交互的对象,可以是人也可以是服务或者其他,通常就理解为用户。

     所有Subject 实例都必须绑定到一个SecurityManager上。我们与一个 Subject 交互,运行时shiro会自动转化为与 SecurityManager交互的特定 subject的交互。

     Subject在shiro中是一个接口,接口中定义了很多认证授权相关的方法,外部程序通过subject进行认证授,而subject是通过SecurityManager安全管理器进行认证授权。

 

(2)SecurityManager

     SecurityManager即安全管理器,对全部的subject进行安全管理,它是shiro的核心,负责对所有的subject进行安全管理。通过SecurityManager可以完成subject的认证、授权等,实质上SecurityManager是通过Authenticator进行认证,通过Authorizer进行授权,通过SessionManager进行会话管理等。

     SecurityManager是一个接口,继承了Authenticator, Authorizer, SessionManager这三个接口。

 

(3)Authenticator

     Authenticator即认证器,对用户身份进行认证,Authenticator是一个接口,shiro提供ModularRealmAuthenticator实现类,通过ModularRealmAuthenticator基本上可以满足大多数需求,也可以自定义认证器。

 

(4)Authorizer

    Authorizer即授权器,用户通过认证器认证通过,在访问功能时需要通过授权器判断用户是否有此功能的操作权限。

 

(5)Realm

     Realm即领域,相当于datasource数据源,securityManager进行安全认证需要通过Realm获取用户权限数据,比如:如果用户身份数据在数据库那么realm就需要从数据库获取用户身份信息。

注意:不要把realm理解成只是从数据源取数据,在realm中还有认证授权校验的相关的代码。

 

(6)sessionManager

     sessionManager即会话管理,shiro框架定义了一套会话管理,它不依赖web容器的session,所以shiro可以使用在非web应用上,也可以将分布式应用的会话集中在一点管理,此特性可使它实现单点登录。

 

(7)SessionDAO

     SessionDAO会话dao,是对session会话操作的一套接口,比如要将session存储到数据库,可以通过jdbc将会话存储到数据库。

 

(8)CacheManager

     CacheManager即缓存管理,将用户权限数据存储在缓存,这样可以提高性能。

 

(9)Cryptography

     Cryptography即密码管理,shiro提供了一套加密/解密的组件,方便开发。比如提供常用的散列、加/解密等功能。

 

三、基本程序

1.认证流程

      

 

2.入门程序

(1)工程架构

 

(2)shiro-first.ini

     通过shiro-first.ini配置文件初始化SecurityManager环境。

   

 

(3)代码实现

public class AuthenticationTest{

	// 用户登录和退出
@Test
public void testLogin(){ // 创建SecurityManager工厂,通过init配置文件构造
IniSecurityManagerFactory factory = new IniSecurityManagerFactory("classpath:shiro-first.ini"); // 创建SecurityManager
SecurityManager securityManager = factory.createInstance(); // 将SecurityManager设置到当前运行环境
SecurityUtils.setSecurityManager(securityManager); // 从SecurityUtils里创建Subject
Subject subject = SecurityUtils.getSubject(); // 认证提交前准备token
UsernamePasswordToken token = new UsernamePasswordToken("zhangsan","123"); try{
// 执行认证提交
subject.login(token);
}catch(Exception e){
e.printStackTrace();
} // 是否认证通过
boolean isAuthenticated = subject.isAuthenticated();
System.out.println("是否认证通过: " + isAuthenticated); // 退出操作
subject.logout(); isAuthenticated = subject.isAuthenticated();
System.out.println("是否认证通过: " + isAuthenticated);
}
}

结果:

      是否认证通过: true

      是否认证通过: false

 

(4)认证流程

1、通过ini配置文件创建securityManager。

2、调用subject.login方法主体提交认证,提交的token。

3、securityManager进行认证,securityManager最终由ModularRealmAuthenticator进行认证。

4、ModularRealmAuthenticator调用IniRealm(给realm传入token) 去ini配置文件中查询用户信息。

5、IniRealm根据输入的token(UsernamePasswordToken)从shiro-first.ini查询用户信息,根据账号查询用户

    信息(账号和密码)如果查询到用户信息,就给ModularRealmAuthenticator返回用户信息(账号和密码)如果查

    询不到,就给ModularRealmAuthenticator返回null。

6、ModularRealmAuthenticator接收IniRealm返回Authentication认证信息如果返回的认证信息是null,

    ModularRealmAuthenticator抛出异常(org.apache.shiro.authc.UnknownAccountException)如果返回的认

    证信息不是null(说明inirealm找到了用户),对IniRealm返回用户密码(在ini文件中存在)和 token中的密码进行对

    比,如果不一致抛出异常(org.apache.shiro.authc.IncorrectCredentialsException)。

 

三、自定义Realm

1.自定义Realm代码实现

/*
* 自定义Realm
*/
public class CustomRealm extends AuthorizingRealm{ //设置Realm名称
@Override
public void setName(String name){
super.setName("customRealm");
} //用于认证
@Override
protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException{ //1.从token取出用户身份信息
String userCode = (String)token.getPrincipal(); //2.根据用户userCode查询数据库
//模拟从数据库查询到的密码
String password = "123"; //查询不到返回null //查询到返回认证信息
SimpleAuthenticationInfo authenticationInfo = new SimpleAuthenticationInfo(userCode,password,this.getName()); return authenticationInfo;
} //用于授权
@Override
protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals){ return null;
} }

 

2.配置Realm

 

3.测试代码

	// 用户登录和退出
@Test
public void testCustomRealm(){ // 创建SecurityManager工厂,通过init配置文件构造
IniSecurityManagerFactory factory = new IniSecurityManagerFactory("classpath:shiro-realm.ini"); // 创建SecurityManager
SecurityManager securityManager = factory.createInstance(); // 将SecurityManager设置到当前运行环境
SecurityUtils.setSecurityManager(securityManager); // 从SecurityUtils里创建Subject
Subject subject = SecurityUtils.getSubject(); // 认证提交前准备token
UsernamePasswordToken token = new UsernamePasswordToken("zhangsan","123"); try{
// 执行认证提交
subject.login(token);
}catch(Exception e){
e.printStackTrace();
} // 是否认证通过
boolean isAuthenticated = subject.isAuthenticated(); System.out.println("是否认证通过: " + isAuthenticated); // 退出操作
subject.logout(); isAuthenticated = subject.isAuthenticated();
System.out.println("是否认证通过: " + isAuthenticated);
}

shiro基础学习(二)—shiro认证的更多相关文章

  1. shiro基础学习(四)—shiro与项目整合

    一.认证 1.配置web.xml   2.配置applicationContext.xml      在applicationContext.xml中配置一个bean,ID和上面的过滤器的名称一致. ...

  2. shiro基础学习(三)—shiro授权

    一.入门程序 1.授权流程        2.授权的三种方式 (1)编程式: 通过写if/else 授权代码块完成. Subject subject = SecurityUtils.getSubjec ...

  3. Python入门基础学习 二

    Python入门基础学习 二 猜数字小游戏进阶版 修改建议: 猜错的时候程序可以给出提示,告诉用户猜测的数字偏大还是偏小: 没运行一次程序只能猜测一次,应该提供多次机会给用户猜测: 每次运行程序,答案 ...

  4. Python基础学习二

    Python基础学习二 1.编码 utf-8编码:自动将英文保存为1个字符,中文3个字符.ASCll编码被囊括在内. unicode:将所有字符保存为2给字符,容纳了世界上所有的编码. 2.字符串内置 ...

  5. Go基础学习(二)

    数组[array] 数组定义[定义后长度不可变] 12 symbol := [...]string{USD: "$", EUR: "€", GBP: " ...

  6. Django基础学习二

    今天继续学习django的基础 学习用户提交url如何获得返回值 1.首先需要在工程的urls文件定义指定的urls要路由给哪个函数 在这个例子中,我们定义home的urls路由给views里的tes ...

  7. Shiro基础学习(一)—权限管理

    一.基本概念 1.权限管理      只要有用户参与的系统一般都要有权限管理,权限管理实现对用户访问系统的控制,按照安全规则或者安全策略控制用户可以访问而且只能访问自己被授权的资源.     权限管理 ...

  8. salesforce lightning零基础学习(二) lightning 知识简单介绍----lightning事件驱动模型

    看此篇博客前或者后,看一下trailhead可以加深印象以及理解的更好:https://trailhead.salesforce.com/modules/lex_dev_lc_basics 做过cla ...

  9. CSS入门基础学习二

    我们下午继续学习CSS的入门基础,搬上你的小板凳赶快进入吧! 一.背景(background) Background-color:背景颜色 background-image (背景图片) backgr ...

随机推荐

  1. 初识html5——试试博文编辑器

    1.html5简介 HTML5 将成为 HTML.XHTML 以及 HTML DOM 的新标准. HTML 的上一个版本诞生于 1999 年.自从那以后,Web 世界已经经历了巨变. HTML5 仍处 ...

  2. asp.net中listview下嵌套gridview

    最近在上软件工程实践课程,想做一个类似于QQ空间或者朋友圈一样的效果.即显示所有好友发送的动态以及动态下回复的信息. 自己YY了一种方法,一开始以为不能达到效果,研究了2个小时终于实现了,感觉效果还是 ...

  3. Docker存储驱动之AUFS简介

    简介 AUFS是曾是Docker默认的首选存储驱动.它非常稳定.有很多真实场景的部署.很强的社区支持.它有以下主要优点: 极短的容器启动时间. 有效的存储利用率. 有效的内存利用率. 虽然如此,但由于 ...

  4. Material Design学习-----SnackBar

    SnackBar是一个和Toast类似的空间,用于弹出提示作用,但是相比于Toast而已,SnackBar会有一个不错的动画效果,同时当手指完成屏幕中其他操作的时候,SnackBar会立即消失.同时可 ...

  5. [数据挖掘] - 聚类算法:K-means算法理解及SparkCore实现

    聚类算法是机器学习中的一大重要算法,也是我们掌握机器学习的必须算法,下面对聚类算法中的K-means算法做一个简单的描述: 一.概述 K-means算法属于聚类算法中的直接聚类算法.给定一个对象(或记 ...

  6. mongoDB & Nodejs 访问mongoDB (一)

    最近的毕设需要用到mongoDB数据库,又把它拿出来再学一学,下盘并不是很稳,所以做一些笔记,不然又忘啦. 安装 mongoDB & mongoVUE mongoDB: https://www ...

  7. android开发艺术探索读书笔记之-------view的事件分发机制

    View的点击事件的分发,其实就是对MotionEvent事件的分发过程,即当一个MotionEvent产生后,系统需要把这个事件传递给一个具体的View,而这个过程就是分发过程. 分发过程主要由以下 ...

  8. Linux命令语句秘籍

    系统管理命令 stat              显示指定文件的详细信息,比ls更详细 who               显示在线登陆用户 whoami          显示当前操作用户 host ...

  9. js数组,字符串常用方法汇总(面试必备)

    字符串: 1.concat() – 将两个或多个字符的文本组合起来,返回一个新的字符串.  2.indexOf() – 返回字符串中一个子串第一处出现的索引.如果没有匹配项,返回 -1 .  3.ch ...

  10. JAVA基础知识(2)--关键字final的使用

    /***Final关键字的使用*@author lihaiming*Email:912547587@qq.com*关键字Final可以修饰属性,方法,类*修饰属性的时候,final修饰的变量是一个常量 ...