1 Spring环境搭建步骤

①创建工程

②导入静态页面

③导入需要坐标

④创建包结构

⑤导入数据库脚本

⑥创建POJO类

⑦创建配置文件

 

对于⑦重点说明:

配置文件主要是对applicationContext和spring-mvc以及web.xml进行配置。

web.xml:
1)配置Spring监听器

<!--  全局的初始化参数  -->
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:applicationContext.xml</param-value>
    </context-param>

    <!--  Spring监听器  -->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

  

作用:ContextLoaderListener会读取这些XML文件并产生 WebApplicationContext对象,然后将这个对象放置在ServletContext的属性

里,这样我们只要可以得到Servlet就可以得到WebApplicationContext对象,并利用这个对象访问spring 容器管理的bean。

2)配置Spring-mvc的前端控制器

<!--  Spring-mvc的前端控制器  -->
    <servlet>
        <servlet-name>DispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:spring-mvc.xml</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>DispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

  

所有请求地址都将经过这个servlet,且优先度为1,由spring-mvc来完成地址的映射和类的注入

关于初始化参数contextConfigLocation的含义,在Spring MVC的学习笔记中有提到。

spring-mvc.xml:

1)mvc注解驱动

<!--  mvc注解驱动  -->
    <mvc:annotation-driven/>

  

在业务方法返回对象时,会自动将该对象转换成json格式的字符串

2)视图解析器

<!--  配置视图解析器  -->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/pages/"/>
        <property name="suffix" value=".jsp"/>
    </bean>

  

在进行页面跳转时,会自动拼接prefix+xxx+suffix。

3)静态资源开放

<!--  静态资源开放 如果不开放相关的资源,那么所有的地址请求都将通过Dispatcher适配器找RequestMapping映射,有的资源就无法找到  -->
    <mvc:default-servlet-handler/>

  

在访问非响应体请求时使用默认的适配器。

4)组件扫描

<!--  组件扫描 扫描Controller  -->
    <context:component-scan base-package="com.xc.controller"/>

  

applicationContext.xml:

1)加载分离出的数据库配置属性

<!--  加载jdbc.properties资源  -->
    <context:property-placeholder location="classpath:jdbc.properties"/>

  

2)配置数据源对象

<!--  配置数据源对象   -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="${jdbc.driver}"/>
        <property name="jdbcUrl" value="${jdbc.url}"/>
        <property name="user" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
    </bean>

  

3)配置JdbcTemplate对象

<!-- 配置JdbcTemplate对象   -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>

  

4)配置Service层+Dao层的对象

<!--  配置Service对象  -->
    <bean id="roleService" class="com.xc.service.impl.RoleServiceImpl">
        <property name="roleDao" ref="roleDao"/>
    </bean>

    <!--  配置Dao对象  -->
    <bean id="roleDao" class="com.xc.dao.impl.RoleDaoImpl">
        <property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean>

    <bean id="userService" class="com.xc.service.impl.UserServiceImpl">
        <property name="userDao" ref="userDao"/>
        <property name="roleDao" ref="roleDao"/>
    </bean>

    <bean id="userDao" class="com.xc.dao.impl.UserDaoImpl">
        <property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean>

  

web层使用注解的方式配置映射和注入相应的Service类,而Service层和Dao层对象则是通过ApplicationContext.xml配置。

2. 角色列表的展示步骤分析

1)点击角色管理菜单发送请求到服务器端

<li><a
  href="${pageContext.request.contextPath}/role/list"> 
  <i class="fa fa-circle-o"></i> 角色管理
</a></li>

  

该请求会通过MVC,扫描Controller下的/role/list,执行对应的业务方法。

然后由该业务方法返回的ViewName来显示页面,该业务方法返回的Model来返回要显示的数据。

2)创建RoleController和showList()方法

@RequestMapping("/list")
    public ModelAndView list(){
        ModelAndView modelAndView = new ModelAndView();
        List<Role> roleList = roleService.list();
        //设置模型
        modelAndView.addObject("roleList",roleList);
        //设置视图
        modelAndView.setViewName("role-list");
        return modelAndView;
    }

  

RoleController层都是/role子目录下的业务方法,该业务方法是/list。

该方法既要回传参数,又要跳转界面,所以返回ModelAndView对象。

该业务方法调用了roleService对象的list方法,所以要在类中配置一个RoleService服务类,并用注解自动注入,

由于类中仅有且仅会一个RoleService类,所以只用加注解@AutoWired。

@Autowired
    private RoleService roleService;

  

3)创建RoleService和showList()方法

到了Service层以及Dao层之后,一般采用接口+实现类的模式。

采用接口的原因,是为了让上层类不依赖于下层类。

如果下层类内部改变的话,不必修改上层类,这对更新和维护提供了方便。

还有一个我觉得方便的点,别的层的开发人员只需要对接口方法的注释进行查看,不用考虑方法的具体实现,可以直接拿来使用。

public class RoleServiceImpl implements RoleService {

    private RoleDao roleDao;

    public void setRoleDao(RoleDao roleDao) {
        this.roleDao = roleDao;
    }

    public List<Role> list() {

        List<Role> roleList = roleDao.findAll();
        return roleList;
    }
}

  

这个服务方法构成简单,只要接收Dao层对角色的查询结果,所以仅仅在方法内调用了Dao层的findAll方法。

因为该类的配置已经在xml中配置,所以不需要加注解配置。

4)创建RoleDao和findAll()方法

接口和实现类的配置类似3)。

public class RoleDaoImpl implements RoleDao {

    private JdbcTemplate jdbcTemplate;


    public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
        this.jdbcTemplate = jdbcTemplate;
    }

    public List<Role> findAll() {

        List<Role> roleList = jdbcTemplate.query("select * from sys_role", new BeanPropertyRowMapper<Role>(Role.class));
        return roleList;
    }
}

  

Dao层采用JDbc模板对数据库进行访问,方法中调用了query方法,通过映射注入Role实体类中然后放入集合,并以List<Role>接收。

具体用法在JdbcTemplate中已有说明。

5)使用JdbcTemplate完成查询操作

在4)中已有体现。

6)将查询数据存储到Model中

在2)中已有体现。

7)转发到role-list.jsp页面进行展示

在2)中已有体现。

 

3. 角色添加的步骤分析

1)点击列表页面新建按钮跳转到角色添加页面

<button type="button" class="btn btn-default" title="新建" onclick="location.href='${pageContext.request.contextPath}/pages/role-add.jsp'">
  <i class="fa fa-file-o"></i> 新建
</button>

  

通过location.href进行页面跳转。

2)输入角色信息,点击保存按钮,表单数据提交服务器

<form action="${pageContext.request.contextPath}/role/save"
  method="post">
  <!-- 正文区域 -->
  <section class="content"> <!--产品信息-->

  <div class="panel panel-default">
  <div class="panel-heading">角色信息</div>
  <div class="row data-type">

  <div class="col-md-2 title">角色名称</div>
  <div class="col-md-4 data">
  <input type="text" class="form-control" name="roleName"
    placeholder="角色名称" value="">
  </div>
  <div class="col-md-2 title">角色描述</div>
  <div class="col-md-4 data">
  <input type="text" class="form-control" name="roleDesc"
    placeholder="角色描述" value="">
  </div>
										
 </div>
</div>
  <!--订单信息/--> <!--工具栏-->
<div class="box-tools text-center">
  <button type="submit" class="btn bg-maroon">保存</button>
  <button type="button" class="btn bg-default"
  onclick="history.back(-1);">返回</button>
</div>
  <!--工具栏/--> </section>
  <!-- 正文区域 /-->
</form>

  

form表单以post形式提交至/role/save业务方法上,通过回传相应参数,有roleName,roleDesc。

3)编写RoleController的save()方法

类似之前业务的方法,映射一个/save地址。

@RequestMapping("/save")
    public String save(Role role){
        roleService.save(role);
        return "redirect:/role/list";
    }

  

此时的参数为实体类Role,类中的属性要与表单中的name属性一致,这样mvc才会将相应的属性一一对应注入。

返回必须是重定向的获取list的请求地址,如果跳过此步直接返回显示页面,页面中的数据会是空的。

4)编写RoleService的save()方法

public void save(Role role) {
        roleDao.save(role);
    }

  

5)编写RoleDao的save()方法

public void save(Role role) {
        jdbcTemplate.update("insert into sys_role values(?,?,?)",null,role.getRoleName(),role.getRoleDesc());

    }

6)使用JdbcTemplate保存Role数据到sys_role

已经在5)中有体现。

7)跳转返回角色列表页面

 

4.用户增加的步骤分析

步骤大体上与角色添加类似,但是有地方需要说明。

因为在添加用户的同时,需要指定用户的角色,而角色列表需要通过前端向后端请求获得,

并且添加用户的信息需要分成2部分保存。

1)用户本身的信息存放在sys_user表。

2)用户的角色信息存放在sys_user_role表。

此处有一个知识点,用户本身可以拥有多个角色身份,而一个角色身份也可以对应多个用户。

所以仅仅是用户表和角色表2个表并不能完全的符合这两者之间的关系。

一般这种多对多的关系需要第三张表,称为中间表。

本例中的中间表分为2列,一列是userId,代表用户的id,一列是roleId,代表角色的id。

每一条信息代表着,用户对应着某一角色。

Spring项目有main吗_spring

 

 

 例如上图,用户1有2个角色,一个是角色1,一个是角色2。

4.1 加载添加用户的界面

controller层

@RequestMapping("/saveUI")
    public ModelAndView saveUI(){
        ModelAndView modelAndView = new ModelAndView();
        List<Role> roleList = roleService.list();
        modelAndView.addObject("roleList",roleList);
        modelAndView.setViewName("user-add");
        return modelAndView;
    }

  

查询角色的功能之前已经写过,所以直接复用就行,也是非常的方便。

其他层也是类似,不多赘述。

4.2 添加用户的功能

controller层

@RequestMapping("/save")
    public String save(User user,Long[] roleIds){
        userService.save(user,roleIds);
        return "redirect:/user/list";
    }

  

上文说到需要将信息分为2部分保存,参数User用来保存sys_user表的信息,roleIds数据用来保存sys_user_role表的信息。

Service层

public void save(User user, Long[] roleIds) {
        //第一步 向sys_user 表中存储数据
        long userId = userDao.save(user);
        //第二步 向sys_user_role 关系表中存储多条数据
        userDao.saveUserRoleRel(userId,roleIds);
    }

  

Dao层

public long save(final User user) {

        //创建PreparedStatementCreator
        PreparedStatementCreator creator = new PreparedStatementCreator() {
            public PreparedStatement createPreparedStatement(Connection con) throws SQLException {
                //使用原始jdbc完成一个PreparedStatement的组建
                PreparedStatement preparedStatement = con.prepareStatement("insert into sys_user values(?,?,?,?,?)", PreparedStatement.RETURN_GENERATED_KEYS);
                preparedStatement.setObject(1,null);
                preparedStatement.setString(2,user.getUsername());
                preparedStatement.setString(3,user.getEmail());
                preparedStatement.setString(4,user.getPassword());
                preparedStatement.setString(5,user.getPhoneNum());
                return preparedStatement;
            }
        };
        //创建keyHolder

        GeneratedKeyHolder keyHolder = new GeneratedKeyHolder();

        jdbcTemplate.update(creator,keyHolder);

        long userId = keyHolder.getKey().longValue();
        //jdbcTemplate.update("insert into sys_user values(?,?,?,?,?)",null,user.getUsername(),user.getEmail(),user.getPassword(),user.getPhoneNum());
        return userId;
    }

  

这边有一点需要说明,如果将update函数用sql语句来实现的话,传入的参数应该是user.getId(),但user中的id并不是前端传到后端去的,而是数据库表自增的id,所以并没有存放在实体类中。

jdbc有一个方法可以返回这个id值,第一个参数为PreparedStatementCreator对象,第二个为keyHolder对象,因为不常用加上现在有更好用的mybatis,所以只做练习,试一遍就好。

 

5 删除用户的功能

删除用户需要注意的点是,需要在请求地址上携带需要删除的用户id。

之前在进行列表展示的时候已经获取了用户的id,而且删除的按钮在同一循环里,所以要获取用户的id也非常的容易。

controller层

@RequestMapping("/del/{userId}")
    public String del(@PathVariable("userId") Long userId){
        userService.del(userId);
        return "redirect:/user/list";
    }

  

参数上的注解之前的文章也有说过,是获取请求地址中的变量。

 

Service层

public void del(Long userId) {
        //1、删除sys_user_role关系表
        userDao.delUserRoleRel(userId);
        //2、删除sys_user表
        userDao.del(userId);
    }

  

删除同样分为2部分进行删除,一个是删除从属表用户角色表,另一个是删除主表角色表。

 

Dao层

public void delUserRoleRel(Long userId) {
        jdbcTemplate.update("delete from sys_user_role where userId=?",userId);
    }

    public void del(Long userId) {
        jdbcTemplate.update("delete from sys_user where id=?",userId);
    }