这里我首先对我上一篇博文的第三个实例做一下讲解,下面是applicationcontext-security.xml内容如下:
xml version="1.0" encoding="utf-8"?>
beans:beans xmlns="http://www.springframework.org/schema/security"
xmlns:beans="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/xmlschema-instance"
xsi:schemalocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/security
http://www.springframework.org/schema/security/spring-security-3.0.xsd">
自动配置模式,拦截所有请求,有role_user才可以通过 -->
http auto-config="true">
intercept-url pattern="/login.jsp*"access="is_authenticated_anonymously" />
增加 role_admin角色-->
intercept-url pattern="/admin.jsp" access="role_admin"/>
intercept-url pattern="/**" access="role_user"/>
form-login login-page="/login.jsp" authentication-failure-url="/login.jsp?login_error=1"/>
http>
认证管理器。用户名密码都集成在配置文件中 -->
authentication-manager>
authentication-provider>
user-service>
添加role_admin角色 -->
user name="admin" password="admin" authorities="role_user,role_admin"/>
user name="sharp" password="sharp" authorities="role_user"/>
user-service>
authentication-provider>
authentication-manager>
指定中文资源 。默认命名空间是security,所以要加前缀beans: -->
beans:bean id="messagesource"
class="org.springframework.context.support.reloadableresourcebundlemessagesource">
beans:property name="basename"value="classpath:org/springframework/security/messages_zh_cn"/>
beans:bean>
beans:beans>
该配置文件的注解如下:
,这一部分是配置如何拦截用户请求,auto-config="true"将自动配置几个常用的权限机制,包括form,anonymous,rememberme.(这几种机制我也不太不明白) 。
我们利用intercept-url来判断用户需要何种权限才能访问对应的url资源,可以在pattern中指定一个特定的url资源,也可以使用通配符来指定一组类似的url资源。例子中定义的两个intercepter-url,第一个用来控制对/admin.jsp的访问,第二个使用了通配符/**,说明它将控制对系统中所有url资源的访问。在实际使用中,spring security采用的是一种就近原则,就是说当用户访问的url资源满足多个intercepter-url时,系统将使用第一个符合条件的intercept-url进行权限控制。在我们这个例子中就是,当用户访问/admin.jsp时,虽然两个intercept-url都满足要求,但因为第一个intercept-url排在上面,所以spring security会使用第一个intercept-url中的配置处理对/admin.jsp的请求,也就是说,只有那些拥有了role_admin权限的用户才能访问/admin.jsp。access指定的权限部分比较有趣,大家可以注意到这些权限标示符都是以role_开头的,实际上这与spring security中的voter机制有着千丝万缕的联系,只有包含了特定前缀的字符串才会被spring security处理。目前来说我们只需要记住这一点就可以了,在我文章以后的部分中我们会详细讲解voter的内容。(这里就解释了我在上面一个文章里,sharp拥有/**所有页面权限,为什么sharp用户却不能访问admin.jsp的原因)
user-service中定义了两个用户,admin和user。为了简便起见,我们使用明文定义了两个用户对应的密码,这只是为了当前演示的方便,之后的例子中我们会使用spring security提供的加密方式,避免用户密码被他人窃取。最最重要的部分是authorities,这里定义了这个用户登陆之后将会拥有的权限,它与上面intercept-url中定义的权限内容一一对应。每个用户可以同时拥有多个权限,例子中的admin用户就拥有role_admin和role_user两种权限,这使得admin用户在登陆之后可以访问role_admin和role_user允许访问的所有资源。与之对应的是,user用户就只拥有role_user权限,所以他只能访问role_user允许访问的资源,而不能访问role_admin允许访问的资源。
第四个实例:
这个实例里面用户信息存入到数据库里,我使用的是数据库是
create table users
(
username varchar2(64) not null,
password varchar2(64),
enablednumber
)
comment on column users.username
is '用户名';
comment on column users.password
is '密码';
comment on column users.enabled
is '0--不可用;1--可用';
alter table users
add constraint pk_users primary key (username)
create table authorities
(
usernamevarchar2(64) not null,
authority varchar2(64) not null
)
tablespace androidxiajun
pctfree 10
initrans 1
maxtrans 255
storage
(
initial 64k
minextents 1
maxextents unlimited
);
alter table authorities
add constraint index_user_auth unique (username, authority)
alter table authorities
add constraint user_authorities_fk foreign key (username)
references users (username);
数据库表创建好后,我们插入一些测试数据:
insert into users(username,password,enabled) values('admin','admin',1);
insert into users(username,password,enabled) values('sharp','sharp',1);
insert into authorities(username,authority) values('admin','role_admin');
insert into authorities(username,authority) values('admin','role_user');
insert into authorities(username,authority) values('sharp','role_user');
数据库的表建好,里面数据也有了,接下来就是修改我们原来的程序了。前面的三个实例都是在配置文件里配置好用户信息,而且还是明文的,如果换成数据库我们只要把这个部分的内容更改下:
认证管理器。用户名密码都集成在配置文件中 -->
authentication-manager>
authentication-provider>
user-service>
添加role_admin角色 -->
user name="admin" password="admin" authorities="role_user,role_admin"/>
user name="sharp" password="sharp" authorities="role_user"/>
user-service>
authentication-provider>
authentication-manager>
修改的新内容是:
认证管理器。用户名密码从数据库里读取 -->
authentication-manager>
authentication-provider>
jdbc-user-service data-source-ref="datasource"/>
authentication-provider>
authentication-manager>
另外我们还要增加一个数据源
datasource,下面是我修改后完整的applicationcontext-security.xml
xml version="1.0" encoding="utf-8"?>
beans:beans xmlns="http://www.springframework.org/schema/security"
xmlns:beans="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/xmlschema-instance"
xsi:schemalocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/security
http://www.springframework.org/schema/security/spring-security-3.0.xsd">
自动配置模式,拦截所有请求,有role_user才可以通过 -->
http auto-config="true">
intercept-url pattern="/login.jsp*"access="is_authenticated_anonymously" />
增加 role_admin角色-->
intercept-url pattern="/admin.jsp" access="role_admin"/>
intercept-url pattern="/**" access="role_user"/>
form-login login-page="/login.jsp" authentication-failure-url="/login.jsp?login_error=1"/>
http>
认证管理器。用户名密码从数据库里读取 -->
authentication-manager>
authentication-provider>
jdbc-user-service data-source-ref="datasource"/>
authentication-provider>
authentication-manager>
指定中文资源 。默认命名空间是security,所以要加前缀beans: -->
beans:bean id="messagesource"
class="org.springframework.context.support.reloadableresourcebundlemessagesource">
beans:property name="basename"value="classpath:org/springframework/security/messages_zh_cn"/>
beans:bean>
配置数据源信息-->
beans:bean id="datasource" class="com.mchange.v2.c3p0.combopooleddatasource">
beans:property name="driverclass" value="${db.driverclass}"/>
beans:property name="jdbcurl" value="${db.jdbcurl}"/>
beans:property name="user" value="${db.user}"/>
beans:property name="password" value="${db.password}"/>
beans:bean>
读取资源文件 -->
beans:bean id="propertyconfigurer" class="org.springframework.beans.factory.config.propertyplaceholderconfigurer">
beans:property name="locations">
beans:list>
beans:value>classpath:constants.propertiesbeans:value>
beans:list>
beans:property>
beans:bean>
beans:beans>
按照我以前开发的习惯的,我会把spring里面很多配置信息放置到constants.properties里面,便于统一管理,constants.properties内容如下:
db.driverclass = oracle.jdbc.driver.oracledriver
db.user= sharpxiajun
db.password= sharpxiajun
db.jdbcurl= jdbc:oracle:thin:@127.0.0.1:1521:orcl
为了和数据库链接我又增加了一些jar包
至于测试和前面三个实例一样,这里不做过多表述了。
上一篇文章里,我们做的第二个实例是“自定义登录界面”,对里面一些特别的配置没有进行讲解,下面的内容就是对这个功能的具体讲解了:
http auto-config="true">
intercept-url pattern="/login.jsp"access="is_authenticated_anonymously" />
增加 role_admin角色-->
intercept-url pattern="/admin.jsp" access="role_admin"/>
intercept-url pattern="/**" access="role_user"/>
form-login login-page="/login.jsp" authentication-failure-url="/login.jsp?login_error=1" default-target-url="/index.jsp"/>
http>
1.intercept-url pattern="/login.jsp"access="is_authenticated_anonymously" />让没登陆的用户也可以访问login.jsp。这是因为配置文件中的“/**”配置,要求用户访问任意一个系统资源时,必须拥有role_user角色,/login.jsp也不例外,如果我们不为/login.jsp单独设置访问权限,会造成用户连登录权限都没有,这个是不正确的
2. form-login login-page="/login.jsp" authentication-failure-url="/login.jsp?login_error=1" default-target-url="/index.jsp"/>login-page表示用户登陆时显示我们自定义的login.jsp。这时我们访问系统显示的登陆页面将是我们上面创建的login.jsp。authentication-failure-url表示用户登陆失败时,跳转到哪个页面。当用户输入的登录名和密码不正确时,系统将再次跳转到/login.jsp,并添加一个error=true参数作为登陆失败的标示。default-target-url表示登陆成功时,跳转到哪个页面。
自制的登录页面里也有很多没有做注解的地方,现在补上页面里的注解,页面内容如下:
body οnlοad="document.f.j_username.focus();">
c:if test="${not empty param.login_error}">
font color="red">
登录失败,请重试.br/>br/>
原因:c:out value="${spring_security_last_exception.message}"/>
font>
c:if>
form name="f" action="" method="post">
table>
tr>
td>用户名:td>
td>
input type='text' name='j_username' value='test="${not empty param.login_error}">c:out value="${spring_security_last_username}"/>c:if>'/>
td>
tr>
tr>
td>密码:td>
td>input type='password' name='j_password'>td>
tr>
tr>
td>
input type="checkbox" name="_spring_security_remember_me">td>td>两周内自动登录
td>
tr>
tr>
td colspan='2' align="center">
input name="submit" type="submit">
input name="reset" type="reset">
td>
tr>
table>
form>
body>
注解如下:
1.
表示在页面装载时onload调用函数'document.f.j_username.focus();该函数的意思是让表单中的j_username获得焦点,即把光标移动到该控件上;
2. action="c:url value='j_spring_security_check'/>" /j_spring_security_check,提交登陆信息的url地址。自定义form时,要把form的action设置为/j_spring_security_check。注意这里要使用绝对路径,避免登陆页面存放的页面可能带来的问题。
3. j_username,输入登陆名的参数名称。
4. j_password,输入密码的参数名称
5. _spring_security_remember_me,选择是否允许自动登录的参数名称。
可以直接把这个参数设置为一个checkbox,无需设置value,spring security
总结下了:springsecurity应用讲解暂时告一段落,这个系列主要是为了应付实际工作,并不是我的研究重点,不过我很想在项目里面使用到它,到那时我一定会更加什么的讲解。另外我手头上springsecurity的资料并不太好,没有做深入研究的,如果哪位童鞋有更好的相关资料,希望能跟我分享下。
======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/