Shiro快速入门

Apache Shiro是一个简洁易用的Java安全框架,提供认证、授权、加密和会话管理功能。本文档介绍了Shiro的基本概念、工作流程、核心组件以及如何在项目中快速配置和使用。通过创建一个简单的Maven工程,导入Shiro依赖,配置shiro.ini文件,然后进行用户认证和授权的测试,展示了Shiro的基础用法。接下来的内容将引导读者将Shiro整合到SpringBoot应用中。
摘要由CSDN通过智能技术生成

Shiro快速入门

shiro 的确是个很简单的安全框架,是个很好的轮子,重复造轮子是可耻的,所以就好好学习如何使用和扩展轮子,发挥它的价值。

shiro官网地址

什么是Shiro

Shiro是一个强大的简单易用的Java安全框架,主要用来更便捷的认证,授权,加密,会话管理。Shiro首要的和最重要的目标就是容易使用并且容易理解。

Shiro可以做什么

  • 验证用户验证他们的身份

  • 确定用户是否被分配了一定的安全角色
  • 确定用户是否被允许做某事
  • 在任何环境中使用Session API,即使没有Web或EJB容器。

  • 在身份验证,访问控制或会话生命周期内对事件作出反应。

  • 聚合1个或更多的用户安全数据数据源,并将其全部显示为单个复合用户视图。

  • 启用单点登录(SSO)功能

  • 为用户关联启用“记住我”服务,无需登录

 shiro 功能(官方)

  • Authentication:身份认证/登录,验证用户是不是拥有相应的身份。
  • Authorization:授权,即权限验证,验证某个已认证的用户是否拥有某个权限;即判断用户是否能做事情,常见的如:验证某个用户是否拥有某个角色。或者细粒度的验证某个用户对某个资源是否具有某个权限。
  • Session Manager:会话管理,即用户登录后就是一次会话,在没有退出之前,它的所有信息都在会话中;会话可以是普通 JavaSE 环境的,也可以是如 Web 环境的。
  • Cryptography:加密,保护数据的安全性,如密码加密存储到数据库,而不是明文存储。
  • Web Support:Web支持,可以非常容易的集成到 web 环境。
  • Caching:缓存,比如用户登录后,其用户信息、拥有的角色/权限不必每次去查,这样可以提高效率。
  • Concurrency:shiro 支持多线程应用的并发验证,即如在一个线程中开启另一个线程,能把权限自动传播过去。
  • Testing:提供测试支持。
  • Run As:允许一个用户假装为另一个用户(如果他们允许)的身份进行访问。
  • Remember Me:记住我,这个是非常常见的功能,即一次登录后,下次再来的话不用登录了。

工作流程

  • Subject:主体,代表了当前“用户”。这个用户不一定是一个具体的人,与当前应用交互的任何东西都是 Subject,如网络爬虫,机器人等。所有 Subject 都绑定到 SecurityManager,与 Subject 的所有交互都会委托给 SecurityManager。我们可以把 Subject 认为是一个请求,SecurityManager 才是实际的执行者
  • SecurityManager:安全管理器。即所有与安全有关的操作都会与 SecurityManager 交互,且它管理着所有 Subject。可以看出它是 Shiro 的核心,它负责与后边介绍的其他组件进行交互,例如SpringMVC中的DispatcherServlet 前端控制器。
  • Realm:域。Shiro 从 Realm 获取安全数据(如用户、角色、权限),就是说 SecurityManager 要验证用户身份,那么它需要从 Realm 获取相应的用户进行比较以确定用户身份是否合法,也需要从 Realm 得到用户相应的角色/权限进行验证用户是否能进行操作。我们可以把 Realm 看成 DataSource,即安全数据源。

Shiro 运行原理图及内部架构图

  • Subject:主体,可以与其他应用进行交互。
  • SecurityManager:相当于 SpringMVC 中的 DispatcherServlet 。它是 Shiro 的核心,所有具体的交互都通过 SecurityManager 进行控制。它管理着所有 Subject、且负责进行认证和授权、及会话、缓存的管理。
  • Authenticator:认证器,负责主体认证的,这是一个扩展点,如果用户觉得 Shiro 默认的不好,我们可以自定义实现。其需要认证策略(Authentication Strategy),即什么情况下算用户认证通过了。
  • 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 提高了一些常见的加密组件用于如密码加密/解密的。

过滤器

过滤器名字对应的Java类作用
anonorg.apache.shiro.web.filter.authc.AnonymousFilter不登陆也能访问
authcorg.apache.shiro.web.filter.authc.FormAuthenticationFilter登录才能访问
authcBasicorg.apache.shiro.web.filter.authc.BasicHttpAuthenticationFilterhttpBasic认证才能访问
permsorg.apache.shiro.web.filter.authz.PermissionsAuthorizationFilter对某个资源具有权限才能访问,可以有多个
portorg.apache.shiro.web.filter.authz.PortFilter限定端口才能访问
restorg.apache.shiro.web.filter.authz.HttpMethodPermissionFilter限定请求的方法,相当于/admins/user/=perms[user:method] ,其中method为post,get,delete等
rolesorg.apache.shiro.web.filter.authz.RolesAuthorizationFilter需要认证用户拥有 roles[角色]角色才能访问。可以有多个,但是要全部满足
sslorg.apache.shiro.web.filter.authz.SslFilter需要使用 https 协议
userorg.apache.shiro.web.filter.authc.UserFilter需要登录或通过记住我认证才能访问

 

Shiro快速入门(官网也有教程)

官网教程

创建一个空的maven工程

导入配置文件

pom.xml

dependencies>
        <dependency>
            <groupId>org.apache.shiro</groupId>
            <artifactId>shiro-core</artifactId>
            <version>1.4.1</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.30</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>jcl-over-slf4j</artifactId>
            <version>1.7.25</version>
        </dependency>
    </dependencies>

shiro.ini

[users]
# admin=admin 分别表示账号和密码,administrator 表示逗号前边的账号拥有 administrator 这个角色。
admin=admin,administrator
lzx=lzx,manager
lisi=lisi,guest

[roles]
# administrator 表示角色名称,* 表示这个角色拥有所有权限
administrator=*
manager=user:*,department:*
guest=user:query,department:query

测试

import org.apache.shiro.SecurityUtils;
import org.apache.shiro.authc.IncorrectCredentialsException;
import org.apache.shiro.authc.LockedAccountException;
import org.apache.shiro.authc.UnknownAccountException;
import org.apache.shiro.authc.UsernamePasswordToken;
import org.apache.shiro.config.IniSecurityManagerFactory;
import org.apache.shiro.mgt.SecurityManager;
import org.apache.shiro.session.Session;
import org.apache.shiro.subject.Subject;
import org.junit.Test;

/**
 * @Author 小先生
 * @Date 2021年05月04日13:49
 */
public class QuickstartTest {

    @Test
    public void test(){
        // 读取 shiro.ini 文件内容
        IniSecurityManagerFactory factory = new IniSecurityManagerFactory("classpath:shiro.ini");
        SecurityManager securityManager = factory.getInstance();
        SecurityUtils.setSecurityManager(securityManager);

        Subject currentUser = SecurityUtils.getSubject();

        Session session = currentUser.getSession();
        session.setAttribute("someKey", "aValue");
        String value = (String) session.getAttribute("someKey");
        if (value.equals("aValue")) {
            System.out.println("someKey 的值:" + value);
        }

        // 登陆
        UsernamePasswordToken token = new UsernamePasswordToken("lzx", "lzx");
        token.setRememberMe(true);
        try {
            currentUser.login(token);
        } catch (UnknownAccountException uae) {
            System.out.println("用户名不存在:" + token.getPrincipal());
        } catch (IncorrectCredentialsException ice) {
            System.out.println("账户密码 " + token.getPrincipal()  + " 不正确!");
        } catch (LockedAccountException lae) {
            System.out.println("用户名 " + token.getPrincipal() + " 被锁定 !");
        }

        // 认证成功后
        if (currentUser.isAuthenticated()) {

            System.out.println("用户 " + currentUser.getPrincipal() + " 登陆成功!");

            //测试角色
            System.out.println("是否拥有 manager 角色:" + currentUser.hasRole("manager"));

            //测试权限
            System.out.println("是否拥有 user:create 权限" + currentUser.isPermitted("user:create"));

            //退出
            currentUser.logout();
        }

    }


}

这里只是一个简单的测试,但是测试代码块里用放到方法是经常要使用,需要掌握。

接下来将Spring Boot与Shiro进行整合。

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值