SSM整合shiro进行权限控制以及shiro的一些特殊功能实现

项目结构图:


强力推荐《跟我学shiro》!!

一、先新建一个maven项目,配置pom.xml


      
      
  1. <project xmlns=“http://maven.apache.org/POM/4.0.0” xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”
  2. xsi:schemaLocation= “http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd”>
  3. <modelVersion>4.0.0 </modelVersion>
  4. <groupId>com.test </groupId>
  5. <artifactId>Test </artifactId>
  6. <packaging>war </packaging>
  7. <version>0.0.1 </version>
  8. <name>Test </name>
  9. <url>http://maven.apache.org </url>
  10. <properties>
  11. <maven.compiler.source>1.8 </maven.compiler.source>
  12. <maven.compiler.target>1.8 </maven.compiler.target>
  13. </properties>
  14. <dependencies>
  15. <dependency>
  16. <groupId>junit </groupId>
  17. <artifactId>junit </artifactId>
  18. <version>3.8.1 </version>
  19. <scope>test </scope>
  20. </dependency>
  21. <!–1、 spring相关包 –>
  22. <dependency>
  23. <groupId>org.springframework </groupId>
  24. <artifactId>spring-aop </artifactId>
  25. <version>4.3.2.RELEASE </version>
  26. </dependency>
  27. <!– 要使用spring的aop,要么引入aspectj,要么cglib –>
  28. <dependency>
  29. <groupId>org.aspectj </groupId>
  30. <artifactId>aspectjweaver </artifactId>
  31. <version>1.6.11 </version>
  32. <type>jar </type>
  33. <scope>compile </scope>
  34. </dependency>
  35. <dependency>
  36. <groupId>cglib </groupId>
  37. <artifactId>cglib </artifactId>
  38. <version>3.2.4 </version>
  39. </dependency>
  40. <dependency>
  41. <groupId>org.springframework </groupId>
  42. <artifactId>spring-beans </artifactId>
  43. <version>4.3.2.RELEASE </version>
  44. </dependency>
  45. <dependency>
  46. <groupId>org.springframework </groupId>
  47. <artifactId>spring-webmvc </artifactId>
  48. <version>4.3.2.RELEASE </version>
  49. </dependency>
  50. <dependency>
  51. <groupId>org.springframework </groupId>
  52. <artifactId>spring-jdbc </artifactId>
  53. <version>4.3.2.RELEASE </version>
  54. </dependency>
  55. <dependency>
  56. <groupId>org.springframework </groupId>
  57. <artifactId>spring-web </artifactId>
  58. <version>4.3.2.RELEASE </version>
  59. </dependency>
  60. <!– 事务相关spring包 –>
  61. <dependency>
  62. <groupId>org.springframework </groupId>
  63. <artifactId>spring-tx </artifactId>
  64. <version>4.3.2.RELEASE </version>
  65. </dependency>
  66. <!–2、 数据库相关包 –>
  67. <!– 导入dbcp的jar包,用来在applicationContext.xml中配置数据库 –>
  68. <dependency>
  69. <groupId>commons-dbcp </groupId>
  70. <artifactId>commons-dbcp </artifactId>
  71. <version>1.4 </version>
  72. </dependency>
  73. <!– 导入Mysql数据库链接jar包 –>
  74. <dependency>
  75. <groupId>mysql </groupId>
  76. <artifactId>mysql-connector-java </artifactId>
  77. <!– Maven中driverClassName为com.mysql.jdbc.Driver时不可使用6.0.3版本 –>
  78. <version>5.1.18 </version>
  79. </dependency>
  80. <!– mybatis相关包 –>
  81. <dependency>
  82. <groupId>org.mybatis </groupId>
  83. <artifactId>mybatis </artifactId>
  84. <version>3.4.2 </version>
  85. </dependency>
  86. <!– mybatis/spring包 –>
  87. <dependency>
  88. <groupId>org.mybatis </groupId>
  89. <artifactId>mybatis-spring </artifactId>
  90. <version>1.3.0 </version>
  91. </dependency>
  92. <!– Ehcache实现,用于参考 –>
  93. <dependency>
  94. <groupId>org.mybatis </groupId>
  95. <artifactId>mybatis-ehcache </artifactId>
  96. <version>1.0.0 </version>
  97. </dependency>
  98. <!– 3、日志文件管理包 –>
  99. <!– log start –>
  100. <dependency>
  101. <groupId>log4j </groupId>
  102. <artifactId>log4j </artifactId>
  103. <version>1.2.17 </version>
  104. </dependency>
  105. <!– 格式化对象,方便输出日志 –>
  106. <dependency>
  107. <groupId>com.alibaba </groupId>
  108. <artifactId>fastjson </artifactId>
  109. <version>1.1.41 </version>
  110. </dependency>
  111. <!– 简单日志门面 –>
  112. <dependency>
  113. <groupId>org.slf4j </groupId>
  114. <artifactId>slf4j-api </artifactId>
  115. <version>1.7.7 </version>
  116. </dependency>
  117. <dependency>
  118. <groupId>org.slf4j </groupId>
  119. <artifactId>slf4j-log4j12 </artifactId>
  120. <version>1.7.7 </version>
  121. </dependency>
  122. <!– log end –>
  123. <!–4、 使用jackson处理对象与JSON之间相互转换 –>
  124. <dependency>
  125. <groupId>org.codehaus.jackson </groupId>
  126. <artifactId>jackson-mapper-asl </artifactId>
  127. <version>1.9.13 </version>
  128. </dependency>
  129. <!– 5、上传组件包 –>
  130. <!– 需要与commons-io配合使用 –>
  131. <dependency>
  132. <groupId>commons-fileupload </groupId>
  133. <artifactId>commons-fileupload </artifactId>
  134. <version>1.3.1 </version>
  135. </dependency>
  136. <!– 开发IO流功能的工具类库 –>
  137. <dependency>
  138. <groupId>commons-io </groupId>
  139. <artifactId>commons-io </artifactId>
  140. <version>2.4 </version>
  141. </dependency>
  142. <!–6、 用来处理常用的编码方法的工具类包 –>
  143. <dependency>
  144. <groupId>commons-codec </groupId>
  145. <artifactId>commons-codec </artifactId>
  146. <version>1.9 </version>
  147. </dependency>
  148. <!– 7、支持servlet的jar包(HttpServletRequest、HttpServletResponse) –>
  149. <dependency>
  150. <groupId>javax.servlet </groupId>
  151. <artifactId>javax.servlet-api </artifactId>
  152. <version>3.1.0 </version>
  153. </dependency>
  154. <!–8、shiro相关包 –>
  155. <dependency>
  156. <groupId>org.apache.shiro </groupId>
  157. <artifactId>shiro-core </artifactId>
  158. <version>1.3.2 </version>
  159. </dependency>
  160. <dependency>
  161. <groupId>org.apache.shiro </groupId>
  162. <artifactId>shiro-web </artifactId>
  163. <version>1.3.2 </version>
  164. </dependency>
  165. <dependency>
  166. <groupId>org.apache.shiro </groupId>
  167. <artifactId>shiro-ehcache </artifactId>
  168. <version>1.3.2 </version>
  169. </dependency>
  170. <dependency>
  171. <groupId>org.apache.shiro </groupId>
  172. <artifactId>shiro-spring </artifactId>
  173. <version>1.3.2 </version>
  174. </dependency>
  175. <!– shiro整合util方法需要 –>
  176. <dependency>
  177. <groupId>com.google.collections </groupId>
  178. <artifactId>google-collections </artifactId>
  179. <version>1.0 </version>
  180. </dependency>
  181. <!– https://mvnrepository.com/artifact/com.google.guava/guava –>
  182. <dependency>
  183. <groupId>com.google.guava </groupId>
  184. <artifactId>guava </artifactId>
  185. <version>19.0 </version>
  186. </dependency>
  187. <!– 9、lombok 包–>
  188. <dependency>
  189. <groupId>org.projectlombok </groupId>
  190. <artifactId>lombok </artifactId>
  191. <version>1.16.16 </version>
  192. </dependency>
  193. <!– 10、编译jsp –>
  194. <dependency>
  195. <groupId>javax.servlet.jsp </groupId>
  196. <artifactId>jsp-api </artifactId>
  197. <version>2.2 </version>
  198. </dependency>
  199. <!– 11、JSTL标签类 –>
  200. <dependency>
  201. <groupId>jstl </groupId>
  202. <artifactId>jstl </artifactId>
  203. <version>1.2 </version>
  204. </dependency>
  205. <!– https://mvnrepository.com/artifact/javax.mail/mail –>
  206. <dependency>
  207. <groupId>com.fasterxml.jackson.core </groupId>
  208. <artifactId>jackson-core </artifactId>
  209. <version>2.8.3 </version>
  210. </dependency>
  211. <!– https://mvnrepository.com/artifact/org.codehaus.jackson/jackson-core-asl –>
  212. <dependency>
  213. <groupId>org.codehaus.jackson </groupId>
  214. <artifactId>jackson-core-asl </artifactId>
  215. <version>1.9.13 </version>
  216. </dependency>
  217. <!– https://mvnrepository.com/artifact/org.codehaus.jackson/jackson-mapper-asl –>
  218. <dependency>
  219. <groupId>com.fasterxml.jackson.core </groupId>
  220. <artifactId>jackson-databind </artifactId>
  221. <version>2.8.3 </version>
  222. </dependency>
  223. <!– https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-annotations –>
  224. <dependency>
  225. <groupId>com.fasterxml.jackson.core </groupId>
  226. <artifactId>jackson-annotations </artifactId>
  227. <version>2.8.3 </version>
  228. </dependency>
  229. <dependency>
  230. <groupId>org.apache.shiro </groupId>
  231. <artifactId>shiro-quartz </artifactId>
  232. <version>1.2.2 </version>
  233. </dependency>
  234. <!– https://mvnrepository.com/artifact/org.springframework/spring-context-support –>
  235. <dependency>
  236. <groupId>org.springframework </groupId>
  237. <artifactId>spring-context-support </artifactId>
  238. <version>3.2.8.RELEASE </version>
  239. </dependency>
  240. <dependency>
  241. <groupId>net.sf.ehcache </groupId>
  242. <artifactId>ehcache-core </artifactId>
  243. <version>2.6.11 </version>
  244. </dependency>
  245. <dependency>
  246. <groupId>commons-collections </groupId>
  247. <artifactId>commons-collections </artifactId>
  248. <version>3.2.1 </version>
  249. </dependency>
  250. <dependency>
  251. <groupId>org.springframework </groupId>
  252. <artifactId>spring-test </artifactId>
  253. <version> 3.2.4.RELEASE </version>
  254. <scope>provided </scope>
  255. </dependency>
  256. </dependencies>
  257. <build>
  258. <finalName>Test </finalName>
  259. <pluginManagement>
  260. <plugins>
  261. <!– 逆向工程生成po类以及mapper –>
  262. <plugin>
  263. <groupId>org.mybatis.generator </groupId>
  264. <artifactId>mybatis-generator-maven-plugin </artifactId>
  265. <version>1.3.2 </version>
  266. <configuration>
  267. <configurationFile>src/main/resources/generator.xml </configurationFile>
  268. <verbose>true </verbose>
  269. <overwrite>true </overwrite>
  270. </configuration>
  271. <executions>
  272. <execution>
  273. <id>Generate MyBatis Artifacts </id>
  274. <goals>
  275. <goal>generate </goal>
  276. </goals>
  277. </execution>
  278. </executions>
  279. <dependencies>
  280. <dependency>
  281. <groupId>org.mybatis.generator </groupId>
  282. <artifactId>mybatis-generator-core </artifactId>
  283. <version>1.3.2 </version>
  284. </dependency>
  285. </dependencies>
  286. </plugin>
  287. <!– 运用maven指令进行test –>
  288. <plugin>
  289. <groupId>org.apache.maven.plugins </groupId>
  290. <artifactId>maven-surefire-plugin </artifactId>
  291. <version>2.19.1 </version>
  292. <configuration>
  293. <skipTests>true </skipTests>
  294. </configuration>
  295. </plugin>
  296. <!– 默认加载本项目下的resource –>
  297. <plugin>
  298. <groupId>org.apache.maven.plugins </groupId>
  299. <artifactId>maven-resources-plugin </artifactId>
  300. <version>3.0.1 </version>
  301. <configuration>
  302. <encoding>UTF-8 </encoding>
  303. </configuration>
  304. </plugin>
  305. <!– 运行maven指令运行tomcat –>
  306. <plugin>
  307. <groupId>org.apache.tomcat.maven </groupId>
  308. <artifactId>tomcat7-maven-plugin </artifactId>
  309. <version>2.2 </version>
  310. </plugin>
  311. </plugins>
  312. </pluginManagement>
  313. <!– 扫描的resource目录 –>
  314. <resources>
  315. <resource>
  316. <directory>src/main/resources </directory>
  317. <includes>
  318. <include>**/* </include>
  319. </includes>
  320. <filtering>true </filtering>
  321. </resource>
  322. </resources>
  323. </build>
  324. <!– 根据不同环境配置 –>
  325. <profiles>
  326. <profile>
  327. <id>dev </id>
  328. <activation>
  329. <activeByDefault>true </activeByDefault>
  330. </activation>
  331. <properties>
  332. <!– 数据库 –>
  333. <ipPort>192.168.*.*:3306/dev_test </ipPort>
  334. </properties>
  335. <build>
  336. <!– 过滤的配置文件 –>
  337. <filters>
  338. <filter>src/main/resources/jdbc.properties </filter>
  339. <filter>src/main/resources/log4j.properties </filter>
  340. </filters>
  341. </build>
  342. </profile>
  343. <profile>
  344. <id>test </id>
  345. <properties>
  346. <!– 数据库 –>
  347. <ipPort>192.168.*.*:3306/creditloan_ph </ipPort>
  348. </properties>
  349. <build>
  350. <filters>
  351. <filter>src/main/resources/jdbc.properties </filter>
  352. <filter>src/main/resources/log4j.properties </filter>
  353. </filters>
  354. </build>
  355. </profile>
  356. <profile>
  357. <id>product </id>
  358. <properties>
  359. <!– 数据库 –>
  360. <ipPort>192.168.*.*:3306/creditloan_ph </ipPort>
  361. </properties>
  362. <build>
  363. <filters>
  364. <filter>src/main/resources/jdbc.properties </filter>
  365. <filter>src/main/resources/log4j.properties </filter>
  366. </filters>
  367. </build>
  368. </profile>
  369. </profiles>
  370. </project>

附:generator.xml

      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <!DOCTYPE generatorConfiguration PUBLIC “-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN” “http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd”>
  3. <generatorConfiguration>
  4. <!– 数据库驱动包位置 –>
  5. <classPathEntry
  6. location= “E:\apache-maven-3.3.9\repo\mysql\mysql-connector-java\5.1.18\mysql-connector-java-5.1.18.jar” />
  7. <context id=“Tables” targetRuntime=“MyBatis3”>
  8. <commentGenerator>
  9. <!– 是否去除自动生成的注释 true:是 : false:否 –>
  10. <property name=“suppressAllComments” value=“true” />
  11. </commentGenerator>
  12. <!– 数据库链接URL、用户名、密码 –>
  13. <jdbcConnection driverClass=“com.mysql.jdbc.Driver”
  14. connectionURL= “jdbc:mysql://192.168.*.*:3306/dev_test” userId= “用户名”
  15. password= “密码”>
  16. <!–<jdbcConnection driverClass=”oracle.jdbc.driver.OracleDriver” connectionURL=”jdbc:oracle:thin:@localhost:1521:orcl”
  17. userId=”msa” password=”msa”> –>
  18. </jdbcConnection>
  19. <javaTypeResolver>
  20. <!– 默认false,把JDBC DECIMAL 和 NUMERIC 类型解析为 Integer, 为 true时把JDBC DECIMAL和NUMERIC类型解析为java.math.BigDecimal –>
  21. <property name=“forceBigDecimals” value=“true” />
  22. </javaTypeResolver>
  23. <!– 生成实体类的包名和位置,这里配置将生成的实体类放在com.loan.entity这个包下 –>
  24. <javaModelGenerator targetPackage=“com.loan.entity”
  25. targetProject= “.\src\main\java\”>
  26. <!– enableSubPackages:是否让schema作为包的后缀 –>
  27. <property name=“enableSubPackages” value=“true” />
  28. <!– 从数据库返回的值被清理前后的空格 –>
  29. <property name=“trimStrings” value=“true” />
  30. </javaModelGenerator>
  31. <!– 生成的SQL映射文件包名和位置,这里配置将生成的SQL映射文件放在com.loan.dao.xml这个包下 –>
  32. <sqlMapGenerator targetPackage=“com.loan.dao.xml”
  33. targetProject= “.\src\main\java\”>
  34. <!– enableSubPackages:是否让schema作为包的后缀 –>
  35. <property name=“enableSubPackages” value=“true” />
  36. </sqlMapGenerator>
  37. <!– 生成DAO的包名和位置,这里配置将生成的dao类放在com.loan.dao.mapper这个包下 –>
  38. <javaClientGenerator type=“XMLMAPPER”
  39. targetPackage= “com.loan.dao.mapper” targetProject= “.\src\main\java\”>
  40. <!– enableSubPackages:是否让schema作为包的后缀 –>
  41. <property name=“enableSubPackages” value=“true” />
  42. </javaClientGenerator>
  43. <!– 要生成那些表(更改tableName和domainObjectName就可以) –>
  44. <table tableName=“sys_user_role” domainObjectName=“UserRole”
  45. enableCountByExample= “false” enableUpdateByExample= “false”
  46. enableDeleteByExample= “false” enableSelectByExample= “false”
  47. selectByExampleQueryId= “false” />
  48. </context>
  49. </generatorConfiguration>


二、先把ssm的基本配置贴上来,就不详细赘述了:

1、application-context.xml(spring管理的相关配置)


     
     
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi= "http://www.w3.org/2001/XMLSchema-instance" xmlns:context= "http://www.springframework.org/schema/context"
  4. xmlns:p= "http://www.springframework.org/schema/p" xmlns:aop= "http://www.springframework.org/schema/aop"
  5. xmlns:tx= "http://www.springframework.org/schema/tx" xmlns:mvc= "http://www.springframework.org/schema/mvc"
  6. xsi:schemaLocation= "http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
  8. http://www.springframework.org/schema/mvc
  9. http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
  10. http://www.springframework.org/schema/tx
  11. http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
  12. http://www.springframework.org/schema/aop
  13. http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
  14. http://www.springframework.org/schema/context
  15. http://www.springframework.org/schema/context/spring-context-3.0.xsd">
  16. <!-- 第一步:【1.整合dao】 将Mybatis和Spring进行整合MyBatis和Spring整合,通过Spring管理mapper接口。使用mapper的扫描器自动扫描mapper接口在Spring中进行注册。 -->
  17. <!-- 需要配置:a、数据源 b、SqlSessionFactory c、mapper扫描器 -->
  18. <!-- 1、数据源定义 -->
  19. <!-- (1)加载jdbc.properties、redis.properties文件中的内容 -->
  20. <bean id="propertyConfigurer"
  21. class= "org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
  22. <property name="locations">
  23. <list>
  24. <value>classpath:jdbc.properties </value>
  25. <!-- <value>classpath:redis.properties</value> -->
  26. </list>
  27. </property>
  28. </bean>
  29. <!-- (2)mysql数据源配置 -->
  30. <!-- a、数据源 -->
  31. <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
  32. destroy-method= "close">
  33. <property name="driverClassName" value="${connection.driverClassName}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="36"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"url"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.url}" />
  34. <property name="username" value="${connection.username}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="38"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"password"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.password}" />
  35. <property name="maxActive" value="${connection.maxActive}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="40"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"maxIdle"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.maxIdle}" />
  36. <property name="minIdle" value="${connection.minIdle}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="42"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"removeAbandoned"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.removeAbandoned}" />
  37. <property name="removeAbandonedTimeout" value="${connection.removeAbandonedTimeout}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="44"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"logAbandoned"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.logAbandoned}" />
  38. <property name="defaultAutoCommit" value="${connection.defaultAutoCommit}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="46"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"defaultReadOnly"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.defaultReadOnly}" />
  39. <property name="validationQuery" value="${connection.validationQuery}"</span> /&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="48"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">property</span> <span class="hljs-attr">name</span>=<span class="hljs-string">"testOnBorrow"</span> <span class="hljs-attr">value</span>=<span class="hljs-string">"${connection.testOnBorrow}" />
  40. </bean>
  41. <!-- b、sqlSessionFactory:创建sqlSessionFactory,同时指定数据源 -->
  42. <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
  43. <property name="dataSource" ref="dataSource"> </property>
  44. <property name="configLocation" value="classpath:mybatis-config.xml" />
  45. <!-- 自动扫描mapper目录, 省掉mybatis-config.xml里的手工配置 -->
  46. <property name="mapperLocations">
  47. <list>
  48. <value>classpath:com/loan/dao/xml/*.xml </value>
  49. </list>
  50. </property>
  51. </bean>
  52. <!-- c、mapper扫描器:通过扫描的模式,扫描目录在com/loan/mapper目录下 -->
  53. <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
  54. <property name="basePackage" value="com.loan.dao.mapper" />
  55. <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" />
  56. </bean>
  57. <!-- 第二步:通过Spring管理Service接口。使用配置方式将Service接口配置在Spring配置文件中。实现事务控制。 -->
  58. <!-- (事务管理) -->
  59. <bean id="transactionManager"
  60. class= "org.springframework.jdbc.datasource.DataSourceTransactionManager">
  61. <property name="dataSource" ref="dataSource"> </property>
  62. </bean>
  63. <!-- 使用annotation定义数据库事务,这样可以在类或方法中直接使用@Transactional注解来声明事务 -->
  64. <tx:annotation-driven transaction-manager="transactionManager" />
  65. <tx:advice id="txAdvice" transaction-manager="transactionManager">
  66. <tx:attributes>
  67. <tx:method name="save*" propagation="REQUIRED" />
  68. <tx:method name="update*" propagation="REQUIRED" />
  69. <tx:method name="delete*" propagation="REQUIRED" />
  70. <tx:method name="load*" propagation="SUPPORTS" read-only="true" />
  71. <tx:method name="find*" propagation="SUPPORTS" read-only="true" />
  72. <tx:method name="search*" propagation="SUPPORTS" read-only="true" />
  73. <tx:method name="approve" propagation="REQUIRED" />
  74. <tx:method name="undo" propagation="REQUIRED" />
  75. <tx:method name="*" propagation="SUPPORTS" read-only="true" />
  76. </tx:attributes>
  77. </tx:advice>
  78. <aop:config>
  79. <aop:pointcut id="serviceMethod"
  80. expression= "execution(* com.loan.service..*.*(..))" />
  81. <aop:advisor advice-ref="txAdvice" pointcut-ref="serviceMethod" />
  82. </aop:config>
  83. <!-- spring管理:自动搜索注解路径 在xml配置了这个标签后,spring可以自动去扫描base-pack下面或者子包下面的Java文件,如果扫描到有@Component @Controller@Service等这些注解的类,则把这些类注册为bean-->
  84. <context:component-scan base-package="com.loan"> </context:component-scan>
  85. </beans>

附:
a、jdbc.properties

      
      
  1. connection.driverClassName=com.mysql.jdbc.Driver
  2. connection.url=jdbc:mysql://192.168.*.*:3306/dev_test?useUnicode=true&characterEncoding=UTF-8
  3. connection.username=user
  4. connection.password=pwd
  5. connection.initialSize=0
  6. connection.maxActive=100
  7. connection.maxIdle=30
  8. connection.minIdle=5
  9. connection.maxWait=5000
  10. connection.removeAbandoned=true
  11. connection.removeAbandonedTimeout=3000
  12. connection.logAbandoned=false
  13. connection.defaultAutoCommit=true
  14. connection.defaultReadOnly=false
  15. connection.validationQuery=SELECT 1
  16. connection.testOnBorrow=true
b、mybatis-config.xml

      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <!DOCTYPE configuration PUBLIC “-//mybatis.org//DTD Config 3.0//EN” “http://mybatis.org/dtd/mybatis-3-config.dtd”>
  3. <configuration>
  4. <!–整合Spring的时候 只有 settings typeAliases mapper 三个属性有用, 其余的要在spring总配置文件中会覆盖 –>
  5. <settings>
  6. <!– 全局映射器,是否启用缓存 –>
  7. <setting name=“cacheEnabled” value=“true” />
  8. <!– 查询时,关闭关联对象即时加载以提高性能 –>
  9. <!– 设置关联对象加载的形态,此处为按需加载字段(加载字段由SQL指 定),不会加载关联表的所有字段,以提高性能 –>
  10. <setting name=“aggressiveLazyLoading” value=“false” />
  11. <!– 对于未知的SQL查询,允许返回不同的结果集以达到通用的效果 –>
  12. <setting name=“multipleResultSetsEnabled” value=“true” />
  13. <!– 允许使用列标签代替列名 –>
  14. <setting name=“useColumnLabel” value=“true” />
  15. <!– 允许使用自定义的主键值(比如由程序生成的UUID 32位编码作为键值),数据表的PK生成策略将被覆盖 –>
  16. <setting name=“useGeneratedKeys” value=“true” />
  17. <!– 给予被嵌套的resultMap以字段-属性的映射支持 –>
  18. <setting name=“autoMappingBehavior” value=“FULL” />
  19. <!– 对于批量更新操作缓存SQL以提高性能 –>
  20. <!– <setting name=”defaultExecutorType” value=”BATCH” /> –>
  21. <!– 数据库超过25000秒仍未响应则超时 –>
  22. <setting name=“defaultStatementTimeout” value=“25000” />
  23. </settings>
  24. </configuration>




2、spring-mvc.xml(SpringMVC的相关配置)


      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <beans xmlns=“http://www.springframework.org/schema/beans”
  3. xmlns:aop= “http://www.springframework.org/schema/aop” xmlns:context= “http://www.springframework.org/schema/context”
  4. xmlns:mvc= “http://www.springframework.org/schema/mvc” xmlns:tx= “http://www.springframework.org/schema/tx”
  5. xmlns:xsi= “http://www.w3.org/2001/XMLSchema-instance”
  6. xsi:schemaLocation= ”http://www.springframework.org/schema/util
  7. http://www.springframework.org/schema/util/spring-util-3.2.xsd
  8. http://www.springframework.org/schema/beans
  9. http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
  10. http://www.springframework.org/schema/context
  11. http://www.springframework.org/schema/context/spring-context-3.2.xsd
  12. http://www.springframework.org/schema/tx
  13. http://www.springframework.org/schema/tx/spring-tx-3.2.xsd
  14. http://www.springframework.org/schema/jee
  15. http://www.springframework.org/schema/jee/spring-jee-3.2.xsd
  16. http://www.springframework.org/schema/mvc
  17. http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd
  18. http://www.springframework.org/schema/aop
  19. http://www.springframework.org/schema/aop/spring-aop-3.2.xsd”>
  20. <!– 使用 mvc:annotation-driven代替注解映射器和注解适配器配置 mvc:annotation-driven默认加载很多的参数绑定方法,
  21. 比如json转换解析器就默认加载了,如果使用mvc:annotation-driven不用配置上边的RequestMappingHandlerMapping和RequestMappingHandlerAdapter
  22. 实际开发时使用mvc:annotation-driven –>
  23. <mvc:annotation-driven />
  24. <context:component-scan base-package=“com.loan” />
  25. <bean
  26. class= “org.springframework.web.servlet.view.InternalResourceViewResolver”>
  27. <property name=“prefix” value=“/WEB-INF/view/” />
  28. <property name=“suffix” value=“.jsp” />
  29. </bean>
  30. </beans>


4、log4j.properties


      
      
  1. log4j.rootLogger=INFO, stdout, logfile
  2. log4j.appender.stdout=org.apache.log4j.ConsoleAppender
  3. log4j.appender.stdout.threshold=INFO
  4. log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
  5. log4j.appender.stdout.layout.ConversionPattern=%d \u4FE1\u9500\u7F51\u7AD9\u540E\u53F0\u7BA1\u7406 –>%5p{%F:%L}-%m%n
  6. log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender
  7. log4j.appender.logfile.threshold=ERROR
  8. log4j.appender.logfile.File=${catalina.home}/logs/Test/Test
  9. log4j.appender.logfile.DatePattern=’-‘yyyy-MM-dd-HH-mm’.log’
  10. log4j.appender.logfile.layout=org.apache.log4j.PatternLayout
  11. log4j.appender.logfile.layout.ConversionPattern=%d \u4FE1\u9500\u7F51\u7AD9\ –>%5p{%F:%L}-%m%n


5、web.xml


      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <web-app xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”
  3. xmlns= “http://java.sun.com/xml/ns/javaee”
  4. xsi:schemaLocation= “http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd”
  5. version= “3.0”>
  6. <context-param>
  7. <param-name>webAppRootKey </param-name>
  8. <param-value>test </param-value>
  9. </context-param>
  10. <context-param>
  11. <param-name>log4jConfigLocation </param-name>
  12. <param-value>classpath:log4j.properties </param-value>
  13. </context-param>
  14. <listener>
  15. <listener-class>org.springframework.web.util.Log4jConfigListener </listener-class>
  16. </listener>
  17. <filter>
  18. <filter-name>CharacterEncodingFilter </filter-name>
  19. <filter-class>org.springframework.web.filter.CharacterEncodingFilter </filter-class>
  20. <init-param>
  21. <param-name>encoding </param-name>
  22. <param-value>UTF-8 </param-value>
  23. </init-param>
  24. <init-param>
  25. <param-name>forceEncoding </param-name>
  26. <param-value>true </param-value>
  27. </init-param>
  28. </filter>
  29. <filter-mapping>
  30. <filter-name>CharacterEncodingFilter </filter-name>
  31. <url-pattern>/* </url-pattern>
  32. </filter-mapping>
  33. <context-param>
  34. <param-name>contextConfigLocation </param-name>
  35. <param-value>
  36. classpath:application-context.xml
  37. classpath:spring-shiro.xml
  38. </param-value>
  39. </context-param>
  40. <listener>
  41. <listener-class>org.springframework.web.context.ContextLoaderListener </listener-class>
  42. </listener>
  43. <servlet>
  44. <servlet-name>dispatcher </servlet-name>
  45. <servlet-class>org.springframework.web.servlet.DispatcherServlet </servlet-class>
  46. <init-param>
  47. <param-name>contextConfigLocation </param-name>
  48. <param-value>classpath:spring-mvc.xml </param-value>
  49. </init-param>
  50. <load-on-startup>1 </load-on-startup>
  51. </servlet>
  52. <servlet-mapping>
  53. <servlet-name>dispatcher </servlet-name>
  54. <url-pattern>/ </url-pattern>
  55. </servlet-mapping>
  56. <session-config>
  57. <session-timeout>30 </session-timeout>
  58. </session-config>
  59. </web-app>


至此,ssm基本配置完成

三、整合shiro

1、spring-shiro.xml


      
      
  1. <beans xmlns=“http://www.springframework.org/schema/beans”
  2. xmlns:xsi= “http://www.w3.org/2001/XMLSchema-instance” xmlns:p= “http://www.springframework.org/schema/p”
  3. xmlns:context= “http://www.springframework.org/schema/context”
  4. xmlns:util= “http://www.springframework.org/schema/util” xmlns:jee= “http://www.springframework.org/schema/jee”
  5. xmlns:tx= “http://www.springframework.org/schema/tx” xmlns:mvc= “http://www.springframework.org/schema/mvc”
  6. xmlns:aop= “http://www.springframework.org/schema/aop”
  7. xsi:schemaLocation=
  8. http://www.springframework.org/schema/util
  9. http://www.springframework.org/schema/util/spring-util-3.0.xsd
  10. http://www.springframework.org/schema/beans
  11. http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
  12. http://www.springframework.org/schema/context
  13. http://www.springframework.org/schema/context/spring-context-3.0.xsd
  14. http://www.springframework.org/schema/tx
  15. http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
  16. http://www.springframework.org/schema/jee
  17. http://www.springframework.org/schema/jee/spring-jee-3.0.xsd
  18. http://www.springframework.org/schema/mvc
  19. http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
  20. http://www.springframework.org/schema/aop
  21. http://www.springframework.org/schema/aop/spring-aop-3.0.xsd”
  22. default-lazy-init= “false”>
  23. <!– 缓存管理器使用Ehcache实现 –>
  24. <bean id=“cacheManager” class=“org.apache.shiro.cache.ehcache.EhCacheManager”>
  25. <property name=“cacheManagerConfigFile” value=“classpath:ehcache.xml” />
  26. </bean>
  27. <!– 凭证匹配器 –>
  28. <bean id=“credentialsMatcher” class=“com.loan.credentials.RetryLimitHashedCredentialsMatcher”>
  29. <constructor-arg ref=“cacheManager” />
  30. <property name=“hashAlgorithmName” value=“md5” />
  31. <property name=“hashIterations” value=“2” />
  32. <property name=“storedCredentialsHexEncoded” value=“true” />
  33. </bean>
  34. <!– Realm实现 –>
  35. <bean id=“userRealm” class=“com.loan.realm.UserRealm”>
  36. <!– <property name=”userService” ref=”userService” /> –>
  37. <property name=“credentialsMatcher” ref=“credentialsMatcher” />
  38. <property name=“cachingEnabled” value=“true” />
  39. <property name=“authenticationCachingEnabled” value=“true” />
  40. <property name=“authenticationCacheName” value=“authenticationCache” />
  41. <property name=“authorizationCachingEnabled” value=“true” />
  42. <property name=“authorizationCacheName” value=“authorizationCache” />
  43. </bean>
  44. <!– 会话ID 生成器 –>
  45. <bean id=“sessionIdGenerator”
  46. class= “org.apache.shiro.session.mgt.eis.JavaUuidSessionIdGenerator” />
  47. <!– 会话DAO –>
  48. <bean id=“sessionDAO”
  49. class= “org.apache.shiro.session.mgt.eis.EnterpriseCacheSessionDAO”>
  50. <property name=“activeSessionsCacheName” value=“shiro-activeSessionCache” />
  51. <property name=“sessionIdGenerator” ref=“sessionIdGenerator” />
  52. </bean>
  53. <!– 会话验证调度器 sessionValidationInterval:设置调度时间间隔 –>
  54. <bean id=“sessionValidationScheduler”
  55. class= “org.apache.shiro.session.mgt.quartz.QuartzSessionValidationScheduler”>
  56. <property name=“sessionValidationInterval” value=“18000000” />
  57. <property name=“sessionManager” ref=“sessionManager” />
  58. </bean>
  59. <!– 会话Cookie 模板 –>
  60. <bean id=“sessionIdCookie” class=“org.apache.shiro.web.servlet.SimpleCookie”>
  61. <constructor-arg value=“sid” />
  62. <property name=“httpOnly” value=“true” />
  63. <property name=“maxAge” value=“1800” />
  64. </bean>
  65. <!– 记住密码cookie –>
  66. <bean id=“rememberMeCookie” class=“org.apache.shiro.web.servlet.SimpleCookie”>
  67. <constructor-arg value=“rememberMe” />
  68. <property name=“httpOnly” value=“true” />
  69. <property name=“maxAge” value=“2592000” /> <!– 30天 30*24*60*60 –>
  70. </bean>
  71. <!– rememberMe管理器,cipherKey是加密rememberMe Cookie的密钥;默认AES算 –>
  72. <bean id=“rememberMeManager” class=“org.apache.shiro.web.mgt.CookieRememberMeManager”>
  73. <property name=“cipherKey”
  74. value= “#{T(org.apache.shiro.codec.Base64).decode(‘4AvVhmFLUs0KTA3Kprsdag==’)}” />
  75. <property name=“cookie” ref=“rememberMeCookie” />
  76. </bean>
  77. <!– 会话管理器 globalSessionTimeout:设置全局会话超时时间,默认30分钟,即如果30分钟内没有访问会话将过期 sessionValidationSchedulerEnabled:是否开启会话验证器,默认是开启的 –>
  78. <bean id=“sessionManager”
  79. class= “org.apache.shiro.web.session.mgt.DefaultWebSessionManager”>
  80. <property name=“globalSessionTimeout” value=“18000000” />
  81. <property name=“deleteInvalidSessions” value=“true” />
  82. <property name=“sessionValidationSchedulerEnabled” value=“true” />
  83. <property name=“sessionValidationScheduler” ref=“sessionValidationScheduler” />
  84. <property name=“sessionDAO” ref=“sessionDAO” />
  85. <property name=“sessionIdCookieEnabled” value=“true” />
  86. <property name=“sessionIdCookie” ref=“sessionIdCookie” />
  87. <property name=“sessionListeners” ref=“sessionListener1” />
  88. </bean>
  89. <!– 监听会话 –>
  90. <bean id=“sessionListener1” class=“com.loan.util.MySessionListener1”> </bean>
  91. <!– 安全管理器 –>
  92. <bean id=“securityManager” class=“org.apache.shiro.web.mgt.DefaultWebSecurityManager”>
  93. <property name=“realm” ref=“userRealm” />
  94. <property name=“sessionManager” ref=“sessionManager” />
  95. <property name=“cacheManager” ref=“cacheManager” />
  96. <property name=“rememberMeManager” ref=“rememberMeManager” />
  97. </bean>
  98. <!– kickoutSessionControlFilter 用于控制并发登录人数的 –>
  99. <bean id=“kickoutSessionControlFilter”
  100. class= “com.loan.util.KickoutSessionControlFilter”>
  101. <property name=“cacheManager” ref=“cacheManager” />
  102. <property name=“sessionManager” ref=“sessionManager” />
  103. <property name=“kickoutAfter” value=“false” />
  104. <property name=“maxSession” value=“2” />
  105. <property name=“kickoutUrl” value=“/login?kickout=1” />
  106. </bean>
  107. <!– 相当于调用SecurityUtils.setSecurityManager(securityManager) –>
  108. <bean
  109. class= “org.springframework.beans.factory.config.MethodInvokingFactoryBean”>
  110. <property name=“staticMethod”
  111. value= “org.apache.shiro.SecurityUtils.setSecurityManager” />
  112. <property name=“arguments” ref=“securityManager” />
  113. </bean>
  114. <!– Shiro 生命周期处理器 –>
  115. <bean id=“lifecycleBeanPostProcessor” class=“org.apache.shiro.spring.LifecycleBeanPostProcessor” />
  116. <bean id=“formAuthenticationFilter”
  117. class= “org.apache.shiro.web.filter.authc.FormAuthenticationFilter”>
  118. <property name=“usernameParam” value=“username” />
  119. <property name=“passwordParam” value=“password” />
  120. <property name=“rememberMeParam” value=“rememberMe” />
  121. <property name=“loginUrl” value=“/login” />
  122. </bean>
  123. <!– Shiro 的Web过滤器 –>
  124. <bean id=“shiroFilter” class=“org.apache.shiro.spring.web.ShiroFilterFactoryBean”>
  125. <property name=“securityManager” ref=“securityManager” />
  126. <property name=“loginUrl” value=“/login” />
  127. <property name=“successUrl” value=“/index” />
  128. <property name=“unauthorizedUrl” value=“/unauthorized” />
  129. <property name=“filters”>
  130. <util:map>
  131. <entry key=“authc” value-ref=“formAuthenticationFilter” />
  132. <entry key=“kickout” value-ref=“kickoutSessionControlFilter”/>
  133. </util:map>
  134. </property>
  135. <property name=“filterChainDefinitions”>
  136. <value>
  137. /static/** = anon
  138. /index = anon
  139. /unauthorized = anon
  140. /login =authc
  141. /logout = logout
  142. /admin/**=user,kickout
  143. </value>
  144. </property>
  145. </bean>
  146. <bean
  147. class= “org.springframework.web.servlet.handler.SimpleMappingExceptionResolver”>
  148. <property name=“exceptionMappings”>
  149. <props>
  150. <prop key=“org.apache.shiro.authz.UnauthorizedException”>
  151. /unauthorized
  152. </prop>
  153. <prop key=“org.apache.shiro.authz.UnauthenticatedException”>
  154. /login
  155. </prop>
  156. </props>
  157. </property>
  158. </bean>
  159. </beans>



附:
a、ehcache.xml


      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <ehcache name=“es”>
  3. <diskStore path=“java.io.tmpdir”/>
  4. <!– 密码输入错误 锁定1小时 –>
  5. <!– timeToIdleSeconds:设置对象在失效前的允许闲置时间(单位:秒) –>
  6. <!– timeToLiveSeconds:设置对象在失效前允许存活时间(单位:秒)。最大时间介于创建时间和失效时间之间。仅当eternal=false对象不是永久有效时使用,默认是0.,也就是对象存活时间无穷大。 –>
  7. <cache name=“passwordRetryCache”
  8. maxEntriesLocalHeap= “2000”
  9. eternal= “false”
  10. timeToIdleSeconds= “3600”
  11. timeToLiveSeconds= “0”
  12. overflowToDisk= “false”
  13. statistics= “true”>
  14. </cache>
  15. <!– 权限记录缓存 锁定1小时 –>
  16. <cache name=“authorizationCache”
  17. maxEntriesLocalHeap= “2000”
  18. eternal= “false”
  19. timeToIdleSeconds= “3600”
  20. timeToLiveSeconds= “0”
  21. overflowToDisk= “false”
  22. statistics= “true”>
  23. </cache>
  24. <!– 登录认证记录缓存 锁定10分钟 –>
  25. <cache name=“authenticationCache”
  26. maxEntriesLocalHeap= “2000”
  27. eternal= “false”
  28. timeToIdleSeconds= “3600”
  29. timeToLiveSeconds= “0”
  30. overflowToDisk= “false”
  31. statistics= “true”>
  32. </cache>
  33. <!– 会话次数缓存 –>
  34. <cache name=“shiro-activeSessionCache”
  35. maxEntriesLocalHeap= “10000”
  36. overflowToDisk= “false”
  37. eternal= “false”
  38. diskPersistent= “false”
  39. timeToLiveSeconds= “0”
  40. timeToIdleSeconds= “0”
  41. statistics= “true”/>
  42. </ehcache>


b、RetryLimitHashedCredentialsMatcher.java

      
      
  1. package com.loan.credentials;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. import javax.annotation.Resource;
  4. import org.apache.shiro.authc.AuthenticationInfo;
  5. import org.apache.shiro.authc.AuthenticationToken;
  6. import org.apache.shiro.authc.ExcessiveAttemptsException;
  7. import org.apache.shiro.authc.SaltedAuthenticationInfo;
  8. import org.apache.shiro.authc.credential.HashedCredentialsMatcher;
  9. import org.apache.shiro.cache.Cache;
  10. import org.apache.shiro.cache.CacheManager;
  11. import org.apache.shiro.crypto.hash.SimpleHash;
  12. import org.apache.shiro.util.ByteSource;
  13. /**
  14. * <p>
  15. * User: Zhang Kaitao
  16. * <p>
  17. * Date: 14-1-28
  18. * <p>
  19. * Version: 1.0
  20. */
  21. public class RetryLimitHashedCredentialsMatcher extends
  22. HashedCredentialsMatcher {
  23. //AtomicInteger是一个提供原子操作的Integer类,通过线程安全的方式操作加减。
  24. private Cache<String, AtomicInteger> passwordRetryCache;
  25. public RetryLimitHashedCredentialsMatcher(CacheManager cacheManager) {
  26. passwordRetryCache = cacheManager.getCache( “passwordRetryCache”);
  27. }
  28. //控制密码输入错误次数
  29. @Override
  30. public boolean doCredentialsMatch(AuthenticationToken token,
  31. AuthenticationInfo info) {
  32. String username = (String) token.getPrincipal();
  33. // retry count + 1
  34. AtomicInteger retryCount = passwordRetryCache.get(username);
  35. System.out.println( “retryCount:”+retryCount);
  36. if (retryCount == null) {
  37. retryCount = new AtomicInteger( 0);
  38. passwordRetryCache.put(username, retryCount);
  39. }
  40. if (retryCount.incrementAndGet() > 5) {
  41. // if retry count > 5 throw
  42. throw new ExcessiveAttemptsException();
  43. }
  44. boolean matches = super.doCredentialsMatch(token, info);
  45. if (matches) {
  46. // clear retry count
  47. passwordRetryCache.remove(username);
  48. }
  49. return matches;
  50. }
  51. }

c、UserRealm.java


     
     
  1. package com.loan.realm;
  2. import org.apache.log4j.Logger;
  3. import org.apache.shiro.authc.AuthenticationInfo;
  4. import org.apache.shiro.authc.AuthenticationToken;
  5. import org.apache.shiro.authc.SimpleAuthenticationInfo;
  6. import org.apache.shiro.authc.UsernamePasswordToken;
  7. import org.apache.shiro.authz.AuthorizationInfo;
  8. import org.apache.shiro.authz.SimpleAuthorizationInfo;
  9. import org.apache.shiro.realm.AuthorizingRealm;
  10. import org.apache.shiro.subject.PrincipalCollection;
  11. import org.apache.shiro.util.ByteSource;
  12. import com.loan.entity.User;
  13. import com.loan.pojo.UserParams;
  14. import com.loan.service.ResourceService;
  15. import com.loan.service.RoleService;
  16. import com.loan.service.UserRoleService;
  17. import com.loan.service.UserService;
  18. /**
  19. * @ClassName: UserRealm
  20. * @Description: TODO(这里用一句话描述这个类的作用)
  21. * @author jiayq
  22. * @date 2016年9月5日 上午10:09:51
  23. *
  24. */
  25. public class UserRealm extends AuthorizingRealm {
  26. private static final Logger logger = Logger.getLogger(UserRealm.class);
  27. @javax.annotation.Resource
  28. private UserRoleService userRoleService;
  29. @javax.annotation.Resource
  30. private ResourceService resourceService;
  31. @javax.annotation.Resource
  32. private UserService userService;
  33. @javax.annotation.Resource
  34. private RoleService roleService;
  35. @Override
  36. protected AuthorizationInfo doGetAuthorizationInfo(
  37. PrincipalCollection principals) {
  38. //直接调用getPrimaryPrincipal得到之前传入的用户名
  39. User user = (User) principals.getPrimaryPrincipal();
  40. logger.info( "[用户:" + user.getUsername() + "|权限授权]");
  41. SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
  42. //根据用户名调用UserService接口获取角色及权限信息
  43. authorizationInfo.setRoles(roleService
  44. .loadRoleIdByUsername(user.getUsername()));
  45. authorizationInfo.setStringPermissions(resourceService
  46. .loadPermissionsByUsername(user.getUsername()));
  47. logger.info( "[用户:" + user.getUsername() + "|权限授权完成]");
  48. return authorizationInfo;
  49. }
  50. @Override
  51. public AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) {
  52. // 获取基于用户名和密码的令牌
  53. // 实际上这个authcToken是从LoginController里面currentUser.login(token)传过来的
  54. UsernamePasswordToken token = (UsernamePasswordToken) authenticationToken;
  55. String username = (String) token.getPrincipal();
  56. System.out.println( "pwd:"+token.getCredentials().toString());
  57. logger.info( "[用户:" + username + "|系统权限认证]");
  58. User u = new User();
  59. u.setUsername(username);
  60. if (userService.find( new UserParams(u)).size() > 0) {
  61. User sqluser = userService.find( new UserParams(u)).get( 0);
  62. // 交给AuthenticatingRealm使用CredentialsMatcher进行密码匹配,如果觉得人家的不好可以自定义实现
  63. System.out.println( "Realm:"+ByteSource.Util.bytes(sqluser.getCredentialsSalt()));
  64. SimpleAuthenticationInfo authenticationInfo = new SimpleAuthenticationInfo(sqluser, sqluser.getPassword(),
  65. ByteSource.Util.bytes(sqluser.getCredentialsSalt()), this.getName()); // realm
  66. logger.info( "[用户:" + username + "|系统权限认证完成]");
  67. return authenticationInfo;
  68. }
  69. return null;
  70. }
  71. }

d、MySessionListener1(用户测试会话过期时间)


     
     
  1. package com.loan.util;
  2. import java.util.Date;
  3. import org.apache.shiro.session.Session;
  4. import org.apache.shiro.session.SessionListener;
  5. public class MySessionListener1 implements SessionListener {
  6. @Override
  7. public void onStart(Session session) { //会话创建时触发
  8. System.out.println( "会话创建:" + session.getId()+ "》》时间:"+session.getLastAccessTime());
  9. }
  10. @Override
  11. public void onExpiration(Session session) { //会话过期时触发
  12. System.out.println( "会话过期:" + session.getId()+ "》》时间:"+ new Date());
  13. }
  14. @Override
  15. public void onStop(Session session) { //退出/会话过期时触发
  16. System.out.println( "会话停止:" + session.getId()+ "》》时间:"+session.getLastAccessTime());
  17. }
  18. }

e、KickoutSessionControlFilter.java


     
     
  1. package com.loan.util;
  2. import java.io.Serializable;
  3. import java.util.Deque;
  4. import java.util.LinkedList;
  5. import javax.servlet.ServletRequest;
  6. import javax.servlet.ServletResponse;
  7. import org.apache.shiro.cache.Cache;
  8. import org.apache.shiro.cache.CacheManager;
  9. import org.apache.shiro.session.Session;
  10. import org.apache.shiro.session.mgt.DefaultSessionKey;
  11. import org.apache.shiro.session.mgt.SessionManager;
  12. import org.apache.shiro.subject.Subject;
  13. import org.apache.shiro.web.filter.AccessControlFilter;
  14. import org.apache.shiro.web.util.WebUtils;
  15. import com.loan.entity.User;
  16. public class KickoutSessionControlFilter extends AccessControlFilter{
  17. private String kickoutUrl; //踢出后到的地址
  18. private boolean kickoutAfter; //踢出之前登录的/之后登录的用户 默认踢出之前登录的用户
  19. private int maxSession; //同一个帐号最大会话数 默认1
  20. private SessionManager sessionManager;
  21. private Cache<String, Deque<Serializable>> cache;
  22. public void setKickoutUrl(String kickoutUrl) {
  23. this.kickoutUrl = kickoutUrl;
  24. }
  25. public void setKickoutAfter(boolean kickoutAfter) {
  26. this.kickoutAfter = kickoutAfter;
  27. }
  28. public void setMaxSession(int maxSession) {
  29. this.maxSession = maxSession;
  30. }
  31. public void setSessionManager(SessionManager sessionManager) {
  32. this.sessionManager = sessionManager;
  33. }
  34. public void setCacheManager(CacheManager cacheManager) {
  35. this.cache = cacheManager.getCache( "shiro-activeSessionCache");
  36. }
  37. /**
  38. * 是否允许访问,返回true表示允许
  39. */
  40. @Override
  41. protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) throws Exception {
  42. return false;
  43. }
  44. /**
  45. * 表示访问拒绝时是否自己处理,如果返回true表示自己不处理且继续拦截器链执行,返回false表示自己已经处理了(比如重定向到另一个页面)。
  46. */
  47. @Override
  48. protected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {
  49. Subject subject = getSubject(request, response);
  50. if(!subject.isAuthenticated() && !subject.isRemembered()) {
  51. //如果没有登录,直接进行之后的流程
  52. return true;
  53. }
  54. Session session = subject.getSession();
  55. String username = ((User)(subject.getPrincipal())).getUsername();
  56. Serializable sessionId = session.getId();
  57. // 初始化用户的队列放到缓存里
  58. Deque<Serializable> deque = cache.get(username);
  59. if(deque == null) {
  60. deque = new LinkedList<Serializable>();
  61. cache.put(username, deque);
  62. }
  63. //如果队列里没有此sessionId,且用户没有被踢出;放入队列
  64. if(!deque.contains(sessionId) && session.getAttribute( "kickout") == null) {
  65. deque.push(sessionId);
  66. }
  67. //如果队列里的sessionId数超出最大会话数,开始踢人
  68. while(deque.size() > maxSession) {
  69. Serializable kickoutSessionId = null;
  70. if(kickoutAfter) { //如果踢出后者
  71. kickoutSessionId=deque.getFirst();
  72. kickoutSessionId = deque.removeFirst();
  73. } else { //否则踢出前者
  74. kickoutSessionId = deque.removeLast();
  75. }
  76. try {
  77. Session kickoutSession = sessionManager.getSession( new DefaultSessionKey(kickoutSessionId));
  78. if(kickoutSession != null) {
  79. //设置会话的kickout属性表示踢出了
  80. kickoutSession.setAttribute( "kickout", true);
  81. }
  82. } catch (Exception e) { //ignore exception
  83. e.printStackTrace();
  84. }
  85. }
  86. //如果被踢出了,直接退出,重定向到踢出后的地址
  87. if (session.getAttribute( "kickout") != null) {
  88. //会话被踢出了
  89. try {
  90. subject.logout();
  91. } catch (Exception e) {
  92. }
  93. WebUtils.issueRedirect(request, response, kickoutUrl);
  94. return false;
  95. }
  96. return true;
  97. }
  98. }

2、spring-mvc.xml中加入shiro注解配置


      
      
  1. <!– 配置用于开启Shiro Spring AOP 权限注解的支持 –>
  2. <bean
  3. class= “org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator”
  4. depends-on= “lifecycleBeanPostProcessor” />
  5. <aop:config proxy-target-class=“true”> </aop:config>
  6. <bean
  7. class= “org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor”>
  8. <property name=“securityManager” ref=“securityManager” />
  9. </bean>
  10. <mvc:resources mapping=“/static/**” location=“/static/” cache-period=“2592000”/>

3、web.xml中加入过滤器shiroFilter


      
      
  1. <filter>
  2. <filter-name>shiroFilter </filter-name>
  3. <filter-class>org.springframework.web.filter.DelegatingFilterProxy </filter-class>
  4. <init-param>
  5. <param-name>targetFilterLifecycle </param-name>
  6. <param-value>true </param-value>
  7. </init-param>
  8. </filter>
  9. <filter-mapping>
  10. <filter-name>shiroFilter </filter-name>
  11. <url-pattern>/* </url-pattern>
  12. </filter-mapping>



四、基本配置已经完成,接下来通过简单的demo对逻辑进行梳理

1、登录验证逻辑

基本思路:新建用户(密码加密)->利用shiro的credentialsMatcher进行验证->完成登录( 重点:密码加密方式与credentialsMatcher的验证方式一致,当然你也可以自己实现验证)
(1)建立sys_user数据库表


(2)实体类User.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import java.util.List;
  4. import lombok.Data;
  5. /**
  6. * @ClassName: User
  7. * @Description: TODO(这里用一句话描述这个类的作用)
  8. * @author jiayq
  9. * @date 2016年9月5日 下午2:31:58
  10. *
  11. */
  12. @SuppressWarnings( “serial”)
  13. @Data
  14. public class User implements Serializable {
  15. private Long id;
  16. private String username; // 用户名
  17. private String workNo; // 工作编号
  18. private String salt; // 盐(密码安全)
  19. private String password; // 密码
  20. private Integer age; // 年龄
  21. private String state; // 状态
  22. private Long orgId;
  23. private String pic;
  24. private String phone;
  25. private String address;
  26. private String email;
  27. private Integer percent;
  28. /**
  29. * @Title: getCredentialsSalt
  30. * @Description: salt = salt + username
  31. * @param @return 设定文件
  32. * @return String 返回类型
  33. * @author jiayq
  34. * @throws
  35. */
  36. public String getCredentialsSalt() {
  37. return username + salt;
  38. }
  39. }
(3)UserMapper.xml


      
      
  1. <?xml version=“1.0” encoding=“UTF-8”?>
  2. <!DOCTYPE mapper PUBLIC “-//mybatis.org//DTD Mapper 3.0//EN” “http://mybatis.org/dtd/mybatis-3-mapper.dtd”>
  3. <mapper namespace=“com.loan.dao.mapper.UserMapper”>
  4. <!–增加–>
  5. <insert id=“save” parameterType=“com.loan.entity.User” keyColumn=“id” useGeneratedKeys=“true” keyProperty=“id”>
  6. insert into sys_user(id,username,work_no,salt,password,age,state,org_id,pic,phone,address,email,percent) values (#{id},#{username},#{workNo},#{salt},#{password},#{age},#{state},#{orgId},#{pic},#{phone},#{address},#{email},#{percent})
  7. </insert>
  8. </mapper>
(4)UserMapper.java

      
      
  1. package com.loan.dao.mapper;
  2. import java.util.List;
  3. import com.loan.entity.User;
  4. import com.loan.pojo.UserParams;
  5. /**
  6. * @ClassName: UserMapper
  7. * @Description: TODO(这里用一句话描述这个类的作用)
  8. * @author jiayq
  9. * @date 2016年9月29日 下午4:08:43
  10. *
  11. */
  12. public interface UserMapper {
  13. public void save(User user);
  14. }

(5)UserService.java

      
      
  1. package com.loan.service;
  2. import java.util.List;
  3. import com.loan.entity.User;
  4. /**
  5. * @ClassName: UserService
  6. * @Description: TODO(这里用一句话描述这个类的作用)
  7. * @author jiayq
  8. * @date 2016年9月7日 上午10:09:32
  9. *
  10. */
  11. public interface UserService {
  12. public void save(User user);
  13. }

(6)UserServiceImpl.java

      
      
  1. package com.loan.service.impl;
  2. import java.util.List;
  3. import javax.annotation.Resource;
  4. import org.springframework.stereotype.Service;
  5. import com.loan.dao.mapper.UserMapper;
  6. import com.loan.dao.mapper.UserRoleMapper;
  7. import com.loan.entity.User;
  8. import com.loan.pojo.UserParams;
  9. import com.loan.service.UserService;
  10. import com.loan.util.EndecryptUtils;
  11. import com.loan.util.PasswordHelper;
  12. /**
  13. * @ClassName: UserServiceImpl
  14. * @Description: TODO(这里用一句话描述这个类的作用)
  15. * @author jiayq
  16. * @date 2016年9月6日 上午9:24:49
  17. *
  18. */
  19. @Service( "userService")
  20. public class UserServiceImpl implements UserService {
  21. @Resource
  22. private UserMapper userMapper;
  23. @Resource
  24. private UserRoleMapper userRoleMapper;
  25. @Override
  26. public void save(User user) {
  27. User u= new PasswordHelper().encryptPassword(user);
  28. userMapper.save(u);
  29. }
  30. }

(7)PasswordHelper.java

      
      
  1. package com.loan.util;
  2. import org.apache.shiro.authc.SimpleAuthenticationInfo;
  3. import org.apache.shiro.crypto.RandomNumberGenerator;
  4. import org.apache.shiro.crypto.SecureRandomNumberGenerator;
  5. import org.apache.shiro.crypto.hash.SimpleHash;
  6. import org.apache.shiro.util.ByteSource;
  7. import com.loan.entity.User;
  8. public class PasswordHelper {
  9. private RandomNumberGenerator randomNumberGenerator = new SecureRandomNumberGenerator();
  10. private String algorithmName = “md5”;
  11. private final int hashIterations = 2;
  12. public User encryptPassword(User user) {
  13. if(user.getSalt()== null||( “”).equals(user.getSalt())){
  14. user.setSalt(randomNumberGenerator.nextBytes().toHex());
  15. }
  16. String newPassword = new SimpleHash(algorithmName, user.getPassword(),
  17. ByteSource.Util.bytes(user.getCredentialsSalt()), hashIterations).toHex();
  18. user.setPassword(newPassword);
  19. return user;
  20. }
  21. }

重点:

(8)新建用户

      
      
  1. package com.loan.controller;
  2. import javax.annotation.Resource;
  3. import org.junit.Test;
  4. import org.junit.runner.RunWith;
  5. import org.springframework.stereotype.Controller;
  6. import org.springframework.test.context.ContextConfiguration;
  7. import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
  8. import org.springframework.ui.Model;
  9. import org.springframework.web.bind.annotation.RequestMapping;
  10. import com.loan.entity.User;
  11. import com.loan.service.UserService;
  12. @Controller
  13. @RunWith(SpringJUnit4ClassRunner.class)
  14. @ContextConfiguration( “classpath:/application-context.xml”)
  15. @RequestMapping( “/test”)
  16. public class TestController {
  17. @Resource
  18. private UserService userService;
  19. @Test
  20. public void userCreate(){
  21. User user= new User();
  22. user.setAge( 20);
  23. user.setPassword( “123”);
  24. user.setUsername( “shiroUser”);
  25. userService.save(user);
  26. }
  27. }

右键run as JUnit test

(9)简单login.jsp页面进行登录


     
     
  1. <%@ page contentType="text/html;charset=UTF-8" language="java"%>
  2. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  3. <html xmlns="http://www.w3.org/1999/xhtml">
  4. <head>
  5. <meta charset="utf-8" />
  6. <meta http-equiv="X-UA-Compatible" content="IE=edge" />
  7. <meta name="viewport" content="width=device-width, initial-scale=1" />
  8. <title>登录 </title>
  9. <!-- Javascript -->
  10. <script src="${pageContext.request.contextPath}/static/jquery-1.7.2.js"</span>&gt;</span><span class="undefined"></span><span class="hljs-tag">&lt;/<span class="hljs-name">script</span>&gt;</span> </div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="11"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">script</span> <span class="hljs-attr">type</span>=<span class="hljs-string">"text/javascript"</span>&gt;</span><span class="javascript"></span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="12"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-comment">//登录</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="13"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">login</span>(<span class="hljs-params"></span>) </span>{</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="14"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-keyword">if</span> ($(<span class="hljs-string">"#userName"</span>).val() == <span class="hljs-literal">null</span> || $(<span class="hljs-string">"#userName"</span>).val() == <span class="hljs-string">''</span>) {</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="15"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> alert(<span class="hljs-string">"请输入用户名!"</span>);</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="16"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> } <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ($(<span class="hljs-string">"#pwd"</span>).val() == <span class="hljs-literal">null</span> || $(<span class="hljs-string">"#pwd"</span>).val() == <span class="hljs-string">""</span>) {</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="17"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> alert(<span class="hljs-string">"密码不能为空!"</span>);</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="18"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> } <span class="hljs-keyword">else</span> {</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="19"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-built_in">document</span>.getElementById(<span class="hljs-string">"form"</span>).submit();</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="20"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> }</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="21"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> </div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="22"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> }</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="23"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-comment">//keydowm登录</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="24"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">keydown_login</span>(<span class="hljs-params"></span>) </span>{</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="25"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-built_in">document</span>.onkeydown = <span class="hljs-function"><span class="hljs-keyword">function</span> (<span class="hljs-params">e</span>)</span>{</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="26"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-keyword">var</span> theEvent = <span class="hljs-built_in">window</span>.event || e;</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="27"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-keyword">var</span> code = theEvent.keyCode || theEvent.which;</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="28"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-keyword">if</span> (code == <span class="hljs-number">13</span> &amp;&amp; (<span class="hljs-literal">null</span> != $(<span class="hljs-string">"#pwd"</span>).val())) {</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="29"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> theEvent.returnValue = <span class="hljs-literal">false</span>;</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="30"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> theEvent.cancel = <span class="hljs-literal">true</span>;</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="31"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-built_in">document</span>.getElementById(<span class="hljs-string">"form"</span>).submit();</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="32"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> }</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="33"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> }</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="34"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> }</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="35"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;/<span class="hljs-name">script</span>&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="36"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;/<span class="hljs-name">head</span>&gt;</span> </div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="37"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">body</span>&gt;</span> </div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="38"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> <span class="hljs-tag">&lt;<span class="hljs-name">form</span> <span class="hljs-attr">method</span>=<span class="hljs-string">"post"</span> <span class="hljs-attr">id</span>=<span class="hljs-string">"form"</span> <span class="hljs-attr">action</span>=<span class="hljs-string">"${pageContext.request.contextPath}/loginTest">
  11. <div class="form-group">
  12. <label>Username </label>
  13. <input type="text" name="username" id="userName" />
  14. <label style="color: red; font-size: 14px; float: right; margin-right: 35px;">${message} </label>
  15. </div>
  16. <div class="form-group">
  17. <label >Password </label>
  18. <input type="password" name="password" id="pwd" />
  19. </div>
  20. <div class="checkbox">
  21. <label> <input type="checkbox" name="rememberMe"/> RemeberMe </label>
  22. </div>
  23. <button type="button" class="btn" onclick="login()">Sign in! </button>
  24. </form>
  25. </body>
  26. </html>

(10)LoginController.java


     
     
  1. package com.loan.controller;
  2. import javax.annotation.Resource;
  3. import javax.servlet.http.HttpServletRequest;
  4. import org.apache.shiro.SecurityUtils;
  5. import org.apache.shiro.authc.AuthenticationException;
  6. import org.apache.shiro.authc.ExcessiveAttemptsException;
  7. import org.apache.shiro.authc.IncorrectCredentialsException;
  8. import org.apache.shiro.authc.UnknownAccountException;
  9. import org.apache.shiro.authc.UsernamePasswordToken;
  10. import org.apache.shiro.subject.Subject;
  11. import org.springframework.stereotype.Controller;
  12. import org.springframework.ui.Model;
  13. import org.springframework.web.bind.annotation.RequestMapping;
  14. import org.springframework.web.bind.annotation.RequestMethod;
  15. import org.springframework.web.bind.annotation.ResponseBody;
  16. import com.loan.entity.User;
  17. import com.loan.service.UserService;
  18. /**
  19. * @ClassName: LoginController
  20. * @Description: TODO(登录controller)
  21. * @author jiayq
  22. * @date 2016年9月5日 下午5:06:33
  23. *
  24. */
  25. @Controller
  26. @RequestMapping( "/")
  27. public class LoginController {
  28. @Resource
  29. private UserService userService;
  30. /**
  31. * @Title: loginView
  32. * @Description: TODO(转向登录界面)
  33. * @param @return 设定文件
  34. * @return String 返回类型
  35. * @throws
  36. */
  37. @RequestMapping(value = "/login")
  38. public String loginView(){
  39. return "login";
  40. }
  41. /**
  42. * @Title: login1
  43. * @Description: TODO(shiro+EndecryptUtils进行认证)
  44. * @param @param request
  45. * @param @param model
  46. * @param @param username
  47. * @param @param password
  48. * @param @return 设定文件
  49. * @return String 返回类型
  50. * @throws
  51. */
  52. @RequestMapping(value = "/loginTest")
  53. public String login(HttpServletRequest request, Model model, String username, String password,boolean rememberMe) {
  54. System.out.println( "rememberMe:"+rememberMe);
  55. //将form中的用户名密码传入Realm 的doGetAuthenticationInfo
  56. UsernamePasswordToken token = new UsernamePasswordToken(username, password.toCharArray());
  57. token.setRememberMe(rememberMe);
  58. Subject currentUser = SecurityUtils.getSubject();
  59. String error = "";
  60. try {
  61. currentUser.login(token);
  62. } catch (UnknownAccountException ex) { // 用户名没有找到
  63. error = "您输入的用户名不存在!";
  64. } catch (IncorrectCredentialsException ex) { // 用户名密码不匹配
  65. error = "用户名密码不匹配 !";
  66. }
  67. catch(ExcessiveAttemptsException e){
  68. error= "密码错误次数已超五次,账号锁定1小时!";
  69. }
  70. catch (AuthenticationException ex) { // 其他的登录错误
  71. ex.printStackTrace();
  72. error = "其他的登录错误 !";
  73. }
  74. // 验证是否成功登录的方法
  75. if (currentUser.isAuthenticated()) {
  76. return "redirect:/admin/index";
  77. } else {
  78. model.addAttribute( "message", error);
  79. currentUser.logout();
  80. return "login";
  81. }
  82. }
  83. }


结果:



重点逻辑:


五、密码输入错误账号锁定功能

前面给的代码中已经实现,主要梳理下逻辑:

六、并发登录人数控制

代码上头已经给出,主要梳理流程:


七、记住密码

spring-shiro.xml中相关配置:


jsp:



LoginController.java


八、shiro的权限授权

1、建立用户、角色、资源以及之间的相关表
(1)sys_user:

User.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import java.util.List;
  4. import lombok.Data;
  5. /**
  6. * @ClassName: User
  7. * @Description: TODO(这里用一句话描述这个类的作用)
  8. * @author jiayq
  9. * @date 2016年9月5日 下午2:31:58
  10. *
  11. */
  12. @SuppressWarnings( “serial”)
  13. @Data
  14. public class User implements Serializable {
  15. private Long id;
  16. private String username; // 用户名
  17. private String workNo; // 工作编号
  18. private String salt; // 盐(密码安全)
  19. private String password; // 密码
  20. private Integer age; // 年龄
  21. private String state; // 状态
  22. private Long orgId;
  23. private String pic;
  24. private String phone;
  25. private String address;
  26. private String email;
  27. private Integer percent;
  28. /**
  29. * @Title: getCredentialsSalt
  30. * @Description: salt = salt + username
  31. * @param @return 设定文件
  32. * @return String 返回类型
  33. * @author jiayq
  34. * @throws
  35. */
  36. public String getCredentialsSalt() {
  37. return username + salt;
  38. }
  39. }


(2)sys_user_role:

UserRole.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import lombok.Data;
  4. /**
  5. * @ClassName: UserRole
  6. * @Description: TODO(这里用一句话描述这个类的作用)
  7. * @author jiayq
  8. * @date 2016年9月20日 上午10:17:50
  9. *
  10. */
  11. @SuppressWarnings( “serial”)
  12. @Data
  13. public class UserRole implements Serializable {
  14. private Long id;
  15. private Long roleId; //角色id
  16. private Long userId; //用户id
  17. }


(3)sys_role:

Role.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import lombok.Data;
  4. /**
  5. * @ClassName: Role
  6. * @Description: TODO(这里用一句话描述这个类的作用)
  7. * @author jiayq
  8. * @date 2016年9月20日 上午10:15:29
  9. *
  10. */
  11. @SuppressWarnings( “serial”)
  12. @Data
  13. public class Role implements Serializable{
  14. private Long id;
  15. private String name; //名称
  16. private String description; //描述
  17. private String state; //状态
  18. private String code; //编码
  19. private Long pid ; //父id
  20. private String remark; //备注
  21. //关联资源列表
  22. private String resources;
  23. }


(4)sys_resource:

Resource.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import java.util.ArrayList;
  4. import java.util.List;
  5. import lombok.Data;
  6. /**
  7. * @ClassName: Resource
  8. * @Description: TODO(这里用一句话描述这个类的作用)
  9. * @author jiayq
  10. * @date 2016年9月19日 下午4:57:11
  11. *
  12. */
  13. @SuppressWarnings( “serial”)
  14. @Data
  15. public class Resource implements Serializable{
  16. private Long id;
  17. private String name; //名称
  18. //private ResourceType type = ResourceType.menu;//资源类型
  19. private String type;
  20. private Integer leaf; //0表示是叶子节点
  21. private Long priority; //顺序
  22. private Long pid; //父id
  23. private String permission; //权限
  24. private String status; //状态
  25. private String url; //路径
  26. private String outUrl; //站外路径
  27. private String pic;
  28. //关联的角色列表
  29. private List<Role> roles;
  30. private List<Resource> children = new ArrayList<Resource>();
  31. }


(5)sys_role_resource:

RoleResource.java

      
      
  1. package com.loan.entity;
  2. import java.io.Serializable;
  3. import lombok.Data;
  4. /**
  5. * @ClassName: RoleResource
  6. * @Description: TODO(这里用一句话描述这个类的作用)
  7. * @author jiayq
  8. * @date 2016年9月20日 上午10:16:57
  9. *
  10. */
  11. @Data
  12. public class RoleResource implements Serializable{
  13. private Long id;
  14. private Long roleId; //角色id
  15. private Long resourceId; //资源id
  16. }

2、新建角色Role为角色分配资源Resource(权限)

(1)新建资源:


      
      
  1. @Test
  2. public void resourceCreate(){
  3. //建立三个资源
  4. for( int i= 1;i< 4;i++){
  5. com.loan.entity.Resource resource= new com.loan.entity.Resource();
  6. resource.setName( “resourceTest_”+i);
  7. resource.setPermission( “test:permission_”+i);
  8. resourceService.save(resource);
  9. }
  10. }

(2)新建role


     
     
  1. @Test
  2. public void roleCreate(){
  3. Role role= new Role();
  4. role.setName( "testRole");
  5. roleService.save(role);
  6. for( int i= 143;i< 146;i++){
  7. RoleResource roleResource= new RoleResource();
  8. roleResource.setRoleId(role.getId());
  9. roleResource.setResourceId(( long)i);
  10. roleResourceService.save(roleResource);
  11. }
  12. }


(3)新建user


     
     
  1. @Test
  2. public void UserCreate(){
  3. User user= new User();
  4. user.setAge( 20);
  5. user.setPassword( "123");
  6. user.setUsername( "shiroUser1");
  7. userService.save(user);
  8. UserRole ur= new UserRole();
  9. ur.setRoleId(( long) 25);
  10. ur.setUserId(user.getId());
  11. userRoleService.save(ur);
  12. }


至此,所以关系逻辑建立。

3、UserRealm权限授权


     
     
  1. @Override
  2. protected AuthorizationInfo doGetAuthorizationInfo(
  3. PrincipalCollection principals) {
  4. //直接调用getPrimaryPrincipal得到之前传入的用户名
  5. User user = (User) principals.getPrimaryPrincipal();
  6. logger.info( "[用户:" + user.getUsername() + "|权限授权]");
  7. SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
  8. //根据用户名调用UserService接口获取角色及权限信息
  9. authorizationInfo.setRoles(roleService
  10. .loadRoleIdByUsername(user.getUsername()));
  11. authorizationInfo.setStringPermissions(resourceService
  12. .loadPermissionsByUsername(user.getUsername()));
  13. logger.info( "[用户:" + user.getUsername() + "|权限授权完成]");
  14. return authorizationInfo;
  15. }
roleMapper.xml


     
     
  1. <select id="loadRoleIdByUsername" parameterType="java.lang.String" resultMap="Role_resultMap">
  2. select sr.* from sys_role sr,sys_user su,sys_user_role sur WHERE su.username=#{username} and su.id=sur.user_id and sr.id=sur.role_id
  3. </select>
resourceMapper.xml


     
     
  1. <select id="loadPermissionByUsername" parameterType="java.lang.String" resultMap="Resource_resultMap">
  2. SELECT sres.* from sys_user su,sys_user_role sur,sys_role sr,sys_role_resource srr,sys_resource sres WHERE su.username=#{name} and su.id=sur.user_id and sr.id=sur.role_id AND srr.role_id=sr.id and srr.resource_id=sres.id
  3. </select>




3、通过jsp/注释测试

(1)jsp


     
     
  1. <%@ page language="java" contentType="text/html; charset=utf-8"
  2. pageEncoding= "utf-8"%>
  3. <%@taglib prefix="shiro" uri="http://shiro.apache.org/tags" %>
  4. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
  5. <html>
  6. <head>
  7. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  8. <title>test </title>
  9. <script src="${pageContext.request.contextPath}/static/jquery-1.7.2.js"</span>&gt;</span><span class="undefined"></span><span class="hljs-tag">&lt;/<span class="hljs-name">script</span>&gt;</span> </div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="10"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-tag">&lt;<span class="hljs-name">script</span> <span class="hljs-attr">type</span>=<span class="hljs-string">"text/javascript"</span>&gt;</span><span class="javascript"></span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="11"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-function"><span class="hljs-keyword">function</span> <span class="hljs-title">logout</span>(<span class="hljs-params"></span>)</span>{</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="12"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> $.post(<span class="hljs-string">'${pageContext.request.contextPath}/logout'</span>,<span class="hljs-function"><span class="hljs-keyword">function</span>(<span class="hljs-params"></span>)</span>{</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="13"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> location.reload();</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="14"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"> });</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="15"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line">}</div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="16"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-tag">&lt;/<span class="hljs-name">script</span>&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="17"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-tag">&lt;/<span class="hljs-name">head</span>&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="18"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-tag">&lt;<span class="hljs-name">body</span>&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="19"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line"><span class="hljs-tag">&lt;<span class="hljs-name">shiro:guest</span>&gt;</span></div></div></li><li><div class="hljs-ln-numbers"><div class="hljs-ln-line hljs-ln-n" data-line-number="20"></div></div><div class="hljs-ln-code"><div class="hljs-ln-line">欢迎游客访问,<span class="hljs-tag">&lt;<span class="hljs-name">a</span> <span class="hljs-attr">href</span>=<span class="hljs-string">"${pageContext.request.contextPath}/login">登录 </a>
  10. </shiro:guest>
  11. <shiro:user>
  12. 欢迎 <shiro:principal property="username"/>登录, <a href="javascript:logout();">退出 </a>
  13. </shiro:user>
  14. <br/>
  15. <shiro:authenticated>
  16. 用户[ <shiro:principal property="username"/>]已身份验证通过
  17. </shiro:authenticated>
  18. <br/>
  19. <shiro:hasRole name="testRole">
  20. 用户[ <shiro:principal property="username"/>]拥有testRole角色 <br/>
  21. </shiro:hasRole>
  22. <br/>
  23. <shiro:hasPermission name="test:permission_1">
  24. 用户[ <shiro:principal property="username"/>]拥有权限test:permission_1 <br/>
  25. </shiro:hasPermission>
  26. </body>
  27. </html>

登录结果:


(2)通过注解控制方法的访问:


     
     
  1. @RequiresPermissions( "test:permission_1")
  2. @RequestMapping( "/testAnnotation")
  3. public String testAnnotation(){
  4. return "/index";
  5. }
  6. @RequiresPermissions( "test:nopermission")
  7. @RequestMapping( "/testAnnotation1")
  8. public String testAnnotation1(){
  9. return "index";
  10. }


测试结果:




终于完了!!!

源代码下载:http://download.csdn.net/download/dreamer_8399/9959021


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值