Shiro概述
Shiro是一款Apache提供的权限校验框架,Shiro同时也是一个强大且易用的Java安全框架,执行身份验证、授权、密码学和会话管理。使用Shiro的易于理解的API,您可以快速、轻松地获得任何应用程序,从最小的移动应用程序到最大的网络和企业应用程序,特别是今天对权限校验和管理特别严格,大家有必要对shiro有一个基本的认识和学习。
Shiro的三大核心组件
1、Subject:当前用户的操作
2、SecurityManager:用于管理所有的Subject3、Realms:用于进行权限信息的验证
如下图:各组件之间的调用关系:
Subject:翻译为主角,当前参与应用安全部分的主角。可以是用户,可以试第三方服务,可以是cron任务,或者任何东西。主要指一个正在与当前软件交互的东西。所有Subject都需要SecurityManager,当你与Subject进行交互,这些交互行为实际上被转换为与SecurityManager的交互
SecurityManager:安全管理员,Shiro架构的核心,它就像Shiro内部所有原件的保护伞。然而一旦配置了SecurityManager,SecurityManager就用到的比较少,开发者大部分时间都花在Subject上面请记得,当你与Subject进行
交互的时候,实际上是SecurityManager在背后帮你举起Subject来做一些安全操作。
Realms:Realms作为Shiro和你的应用的连接桥,当需要与安全数据交互的时候,像用户账户,或者访问控制,Shiro就从一个或多个Realms中查找。Shiro提供了一些可以直接使用的Realms,如果默认的Realms不能满足你的需求,你也可以定制自己的Realms.
Shiro的十大功能特点:
功能特点
Shiro包含10个内容,如下图:
Authentication:身份认证/登录,验证用户是不是拥有相应的身份。Authorization:授权,即权限验证,验证某个已认证的用户是否拥有某个权限;即判断用户是否能做事情,常见的如:验证某个用户是否拥有某个角色。或者细粒度的验证某个用户对某个资源是否具有某个权限。SessionManager:会话管理,即用户登录后就是一次会话,在没有退出之前,它的所有信息都在会话中;会话可以是普通JavaSE环境的,也可以是如Web环境的。Cryptography:加密,保护数据的安全性,如密码加密存储到数据库,而不是明文存储。WebSupport:Web支持,可以非常容易的集成到web环境。Caching:缓存,比如用户登录后,其用户信息、拥有的角色/权限不必每次去查,这样可以提高效率。Concurrency:shiro支持多线程应用的并发验证,即如在一个线程中开启另一个线程,能把权限自动传播过去。Testing:提供测试支持。RunAs:允许一个用户假装为另一个用户(如果他们允许)的身份进行访问。RememberMe:记住我,这个是非常常见的功能,即一次登录后,下次再来的话不用登录了。Shiro的运行原理
Subject:主体,可以看到主体可以是任何与应用交互的“用户”。
SecurityManager:相当于SpringMVC中的DispatcherServlet或者Struts2中的FilterDispatcher。它是Shiro的核心,所有具体的交互都通过
SecurityManager进行控制。它管理着所有Subject、且负责进行认证和授权、及会话、缓存的管理。
Authenticator:认证器,负责主体认证的,这是一个扩展点,如果用户觉得Shiro默认的不好,我们可以自定义实现。其需要认证策略(AuthenticationStrategy),即什么情况下算用户认证通过了。
Authrizer:授权器,或者访问控制器。它用来决定主体是否有权限进行相应的操作,即控制着用户能访问应用中的哪些功能。
Realm:可以有1个或多个Realm,可以认为是安全实体数据源,即用于获取安全实体的。它可以是JDBC实现,也可以是LDAP实现,或者内存实现等。
SessionManager:如果写过Servlet就应该知道Session的概念,Session需要有人去管理它的生命周期,这个组件就是SessionManager。而Shiro并不仅仅可以用在Web环境,也可以用在如普通的JavaSE环境。
SessionDAO:DAO大家都用过,数据访问对象,用于会话的CRUD。我们可以自定义SessionDAO的实现,控制session存储的位置。如通过JDBC写到数据库或通过jedis写入redis中。另外SessionDAO中可以使用Cache进行缓存,以提高性能。
CacheManager:缓存管理器。它来管理如用户、角色、权限等的缓存的。因为这些数据基本上很少去改变,放到缓存中后可以提高访问的性能。
Cryptography:密码模块,Shiro提高了一些常见的加密组件用于如密码加密/解密的。
Shiro的基本入门
今天我们演示Shiro的入门,没有整合任何框架,只是单纯的演示Shiro运行原理,所以无需创建经典的五张表
使用Idea创建一个Maven项目
在pom.xml中添加如下依赖:
!—配置shiro依赖--
dependencies
dependency
groupIdorg.apache.shiro/groupId
artifactIdshiro-core/artifactId
version1.2.3/version
/dependency
dependency
groupIdorg.slf4j/groupId
artifactIdslf4j-simple/artifactId
version1.6.1/version
/dependency
/dependencies
任意创建一个包,在里面创建一个测试的Demo类:
packageme.aihe;importorg.slf4j.Logger;importorg.slf4j.LoggerFactory;
publicclassDemo{
privatestaticfinaltransientLoggerlog=LoggerFactory.getLogger(Tutorial.class);publicstaticvoidmain(String[]args){
log.info(MyFirstApacheShiroApplication);System.exit(0);
}
}创建一个shiro.ini配置文件
Shiro提供了一个通用的方案通过INI进行配置,当然也可以通过XML,YMAL,JSON等进行配置。在resource目录下面,创建一个shiro.ini的文件。内容如下:
#-------------------------------------------------------------------------
#Usersandtheir(optional)assignedroles
#username=password,role1,role2,...,roleN
#-------------------------------------------------------------------------
[users]
root=secret,adminguest=guest,guest
presidentskroob=,president
darkhelmet=ludicrousspeed,darklord,schwartzaihe=aihe,goodguy,client
#-------------------------------------------------------------------------
#Roleswithassignedpermissions
#roleName=perm1,perm2,...,permN
#-------------------------------------------------------------------------
[roles]admin=*
client=look:*
goodguy=winnebago:drive:eagle5
引用Shiro.ini配置进行测试
现在改变我们的Demo类文件,内容如下
importorg.apache.shiro.authc.*;
importorg.apache.shiro.config.IniSecurityManagerFactory;
importorg.apache.shiro.mgt.SecurityManager;
importorg.apache.shiro.session.Session;
importorg.apache.shiro.subject.Subject;
importorg.apache.shiro.util.Factory;
importorg.slf4j.Logger;
importorg.slf4j.LoggerFactory;
publicclassDemo{
privatestaticfinaltransientLoggerlog=LoggerFactory.getLogger(Tutorial.class);
publicstaticvoidmain(String[]args){log.info(MyFirstApacheShiroApplication);
//1.这里的SecurityManager是org.apache.shiro.mgt.SecurityManager,而不是
//java.lang.SecurityManager加载配置文件
FactorySecurityManagerfactory=newIniSecurityManagerFactory(classpath:shiro.ini);
//2.解析配置文件,并且返回一些SecurityManger实例
SecurityManagersecurityManager=factory.getInstance();
//3.设置SecurityManager到静态内存区,单例模式
SecurityUtils.setSecurityManager(securityManager);
//安全操作
SubjectcurrentUser=SecurityUtils.getSubject();
//在应用的当前会话中设置属性
Sessionsession=currentUser.getSession();session.setAttribute(key,value);
//当前我们的用户是匿名的用户,我们尝试进行登录,
if(!currentUser.isAuthenticated()){
UsernamePasswordTokentoken=newUsernamePasswordToken(aihe,aihe);
token.setRememberMe(true);
//尝试进行登录用户,如果登录失败了,我们进行一些处理
try{currentUser.login(token);
//当我们获登录用户之后
log.info(User[+currentUser.getPrincipal()+]loggedinsuccessfully.);
//查看用户是否有指定的角色
if(currentUser.hasRole(client)){
log.info(Lookisinyourrole);}else{log.info(. );
}
//查看用户是否有某个权限
if(currentUser.isPermitted(look:desk)){
log.info(Youcanlook.Useitwisely.);
}else{
log.info(Sorry,youcantlook.);
}
if(currentUser.isPermitted(winnebago:drive:eagle5)){
log.info(Youarepermittedtodrivethewinnebagowithlicenseplate(id)eagle5.+Hereare
thekeys-havefun!);
}else{
log.info(Sorry,youarentallowedtodrivetheeagle5winnebago!);
}
//登出currentUser.logout();
}catch(UnknownAccountExceptionuae){
//账户不存在的操作
}catch(IncorrectCredentialsExceptionice){
//密码不正确
}catch(LockedAccountExceptionlae){
//用户被锁定了
}catch(AuthenticationExceptionae){
//无法判断的情形
}
}
System.exit(0);}
}
通过shiro演示,我们学到了什么
这个相对来说是一个简单的程序,但也证明了一些shiro的基本用法,我们可以通过shiro进行认证,权限控制等。
本文对Shiro进行了一个基本介绍,Shiro具体在开发中的实际应用,以及更多在开发中的应用,我们并没有具体说太多,比如和其它框架的整合等等,如果想对Shiro有更多的了解,请前往