目录

  • 介绍
  • spring架构图
  • spring配置方式
  • spring装配Bean(此处只介绍xml配置)
  • 构造器装配Bean
  • set装配Bean
  • 运行时值注入
  • list、set、map、prop配置
  • 懒加载
  • 混合配置
  • 自动装配
  • bean的作用域
  • AOP面向切面编程
  • spring Data
  • spring mvc
  • 控制器controller
  • 渲染Web视图
  • spring mvc 高级用法
  • 处理异常
  • 保护web应用
  • QA


介绍

  • 在此只介绍如何使用spring IOC、AOP、JDBC等功能。
  • 这个仅是开发参考文档,方便快速查阅使用。

spring架构图

springdatajpa开发手册 spring开发文档_java

spring配置方式

  • xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:c="http://www.springframework.org/schema/c"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!-- service -->
    <bean id="service" class="iocAndDIAndWiring.Service"></bean>
</beans>
  • java config类
@Configuration //配置文件
//组件扫描:扫描此类所在的包。
//xml配置文件中:<context:annotation-config/>
@ComponentScan 
public class JavaConfig {
    @Bean //bean装配
    public Service service() {
        return new Service();
    }
}

spring装配Bean(此处只介绍xml配置)

xml配置比较复杂,所以在这里详细介绍

构造器装配Bean

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:c="http://www.springframework.org/schema/c"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!-- service -->
    <bean id="service" class="iocAndDIAndWiring.Service"></bean>
    
    <!-- 使用构造器装配 -->
    <bean id="handler" class="iocAndDIAndWiring.Handler">
        <constructor-arg ref="service"></constructor-arg>
    </bean>
    
    <!-- c标签创建handler Bean -->
    <bean id="handler" class="iocAndDIAndWiring.Handler" c:service-ref="service"></bean>
</beans>

set装配Bean

<!-- handler -->
<bean id="handler" class="iocAndDIAndWiring.Handler">
    <property name="title" value="hello"></property>
    <property name="art" value="java"></property>
    <property name="hobby">
        <list>
            <value>hello</value>
            <value>world</value>
        </list>
    </property>
</bean>

<!-- p标签  p:参数名-ref装配对象、p:参数名 装配字面量-->
<bean id="handler" class="iocAndDIAndWiring.Handler" p:title="hello" p:art="art">
    <property name="hobby">
        <list>
            <value>hello</value>
            <value>world</value>
        </list>
    </property>
</bean>

运行时值注入

  • 属性占位符:${""}
  1. Environment实现
@Configuration
@PropertySource("iocAndDIAndWiring/prop.properties")
public class SpringConfig {
    @Autowired
    Environment env;
    @Bean
    public Service service() {
        return new Service(env.getProperty("title"), env.getProperty("name"));
    }
}
  1. @Value实现
@Configuration
@PropertySource("iocAndDIAndWiring/prop.properties")
public class SpringConfig {
    @Bean
    public static PropertySourcesPlaceholderConfigurer placeholderConfigurer() {
        return new PropertySourcesPlaceholderConfigurer();
    }
    @Bean
    public Service service(@Value("${title}") String title,@Value("${name}") String name) {
        return new Service(title, name);
    }
}
  1. xml配置
<context:property-placeholder location="iocAndDIAndWiring/prop.properties"/>

<bean id="service" class="iocAndDIAndWiring.Service" c:title="${title}" c:name="${name}"></bean>
  • 使用spEL
    等以后补充。

list、set、map、prop配置

<!--  list、set、map、prop属性赋值 -->
 <list>
    <value>world</value>
    <value>hello</value>
</list>

<set>
    <value>hello</value>
    <value>world</value>
</set>
    
<!-- 对于引入bean -->
<set>
    <ref bean="bean1" />
    <ref bean="bean2" />
    //父xml配置
    <ref parent="parentBean1">
</set>
	
<property name="hobby">
	 <props>
	     <prop key="key">value</prop>
	 </props>
	 
	 <map>
	     <entry key="key" value="value"></entry>
	 </map>
</property>

<!--  util配置集合属性 -->
<bean id="handler" class="iocAndDIAndWiring.Handler" p:title="hello" p:art="art">
    <property name="hobby">
        <util:list>
            <value>hello</value>
            <value>world</value>
        </util:list>
    </property>
</bean>

懒加载

  • ioc容器初始化的时候,不创建懒加载的类,等需要时才会创建。
<bean id="lazy" class="com.something.ExpensiveToCreateBean" lazy-init="true"/>

//ioc容器全部懒加载
<beans default-lazy-init="true">

混合配置

  • java类混合xml
    @ImportResource,导入xml文件即可。
@Configuration
@ImportResource("iocAndDIAndWiring/spring.xml")
public class SpringConfig {
    @Bean
    public Service service() {
        return new Service();
    }
}
  • xml混合java类
    引入配置类即可
<!-- springCOnfig 配置类 -->
<bean class="iocAndDIAndWiring.SpringConfig" />

自动装配

  • 实现方式
@AutoWired
public void setIce(Ice ice) {
}
  • 歧义问题
  1. 标记首选的bean
@component
@Primary
public class Ice{}
  1. @Qualifier - 限定自动装配的bean
@AutoWired
@Qualifier("beanId") //@Qualifier,根据bean id限定使用哪个bean
public void setIce(Ice ice) {
}
  1. 创建自定义的限定符注解
@Target(ElementType.CONSTRUCTOR,ElementType.FIELD,ElementType.METHOD,ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface Ice{}

@AutoWired
@Ice
public void setIce(Ice ice) {
}

bean的作用域

  • 单例(singeton)
  • 原型(prototype)
  • 会话(session): web应用使用
  • 请求(request):web应用使用
@Component
//@Scope("prototype") 不安全
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class Ice{}
  1. 购物车
@Component
@Scope(
    value=WebApplicationContext.SCOPE_SESSION,
    //接口代理模式,另一种是类代理模式
    proxyMode=ScopedProxyMode.INTERFACES
)
public class ShopCart{}

//xml配置
<bean id="shopCart" class="ShopCart" scope="session">
    //默认使用CGLib类代理
    <aop:scoped-proxy />
    //<aop:scoped-proxy proxy-target-class="false"/>
</bean>

AOP面向切面编程

  • 注解实现
  1. @Before、@After、@AfterReturning、@AfterThrowing
@Configuration
@ComponentScan
@EnableAspectJAutoProxy
public class JavaConfig {
    //环绕通知
    @Bean
    public aop.环绕通知.Audience audience() {
        return new aop.环绕通知.Audience();
    }
    //before、afterReturn、afterThrow
    @Bean
    public Audience audience1() {
        return new Audience();
    }
    //引入新功能
    @Bean
    public AddMethod addMethod() {
        return new AddMethod();
    }
    @Bean
    public Performance performance() {
        return new PerformanceImpl();
    }
}
  1. @Around
@Aspect
public class Audience {
    /**
     * 定义切点
     */
    @Pointcut("execution(** aop.Performance.perform(..))")
    public void performamce(){}
    @Around("performamce()")
    public void watchPerformance(ProceedingJoinPoint jp) {
        try {
            System.out.println("手机静音2。。。");
            System.out.println("观众坐下2。。。");
            //连接点,即拦截的方法调用
            jp.proceed();
            System.out.println("观众喝彩2。。。");
        } catch (Throwable e) {
            System.out.println("观众退票2。。。");
            e.printStackTrace();
        }
    }
}
  • xml配置
<!-- aop -->
<aop:config>
    <aop:aspect ref="aop">
        <aop:pointcut expression="execution(** aop.xml.Service.*(..))" id="p"/>
        <aop:before method="before" pointcut-ref="p"/>
        <aop:after method="after" pointcut-ref="p"/>
        <aop:around method="around" pointcut-ref="p"/>
    </aop:aspect>
</aop:config>

spring Data

  • JDBC
  1. xml配置
<bean id="dataSource"
        class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver" />
        <property name="url" value="jdbc:mysql://localhost:3306/yiibaijava" />
        <property name="username" value="root" />
        <property name="password" value="password" />
</bean>
  1. java类配置
@Bean
public DataSource dataSource() {
    DriverManagerDataSource ds = new DriverManagerDataSource();
    ds.setDriverName("org.h2.Driver");
    ds.setUrl("jdbc:h2:tcp://localhost/~/spitter");
    ds.setUserName("");
    ds.setPassWord("");
    return ds;
}
  1. 使用
public class JdbcCustomerDAO implements CustomerDAO
{
    private DataSource dataSource;
    public void setDataSource(DataSource dataSource) {
        this.dataSource = dataSource;
    }
    public void insert(Customer customer){
        String sql = "INSERT INTO CUSTOMER " +
                "(CUST_ID, NAME, AGE) VALUES (?, ?, ?)";
        Connection conn = null;
        try {
            conn = dataSource.getConnection();
            PreparedStatement ps = conn.prepareStatement(sql);
            ps.setInt(1, customer.getCustId());
            ps.setString(2, customer.getName());
            ps.setInt(3, customer.getAge());
            ps.executeUpdate();
            ps.close();
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            if (conn != null) {
                try {
                    conn.close();
                } catch (SQLException e) {}
            }
        }
    }
    public Customer findByCustomerId(int custId){
        String sql = "SELECT * FROM CUSTOMER WHERE CUST_ID = ?";
        Connection conn = null;
        try {
            conn = dataSource.getConnection();
            PreparedStatement ps = conn.prepareStatement(sql);
            ps.setInt(1, custId);
            Customer customer = null;
            ResultSet rs = ps.executeQuery();
            if (rs.next()) {
                customer = new Customer(
                    rs.getInt("CUST_ID"),
                    rs.getString("NAME"), 
                    rs.getInt("Age")
                );
            }
            rs.close();
            ps.close();
            return customer;
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            if (conn != null) {
                try {
                conn.close();
                } catch (SQLException e) {}
            }
        }
    }
}
  • JdbcTemplate
  1. xml配置
<!-- 配置C3P0数据源 -->
    <bean id="dataSource"
        class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="user" value="${jdbc.user}"></property>
        <property name="password" value="${jdbc.password}"></property>
        <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
        <property name="driverClass" value="${jdbc.driverClass}"></property>

        <property name="initialPoolSize" value="${jdbc.initPoolSize}"></property>
        <property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
    </bean>

    <!-- 配置Spring的jdbcTemplate 
        并注入一个dataSource数据源-->
    <bean id="jdbcTemplate"
        class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>    
    </bean>
  1. 使用
  • 添加
String sql="insert into user (name,deptid) values (?,?)";
jdbcTemplate.update(sql, new Object[]{"caoyc",3});
  • 修改
String sql="update user set name=?,deptid=? where id=?";
jdbcTemplate.update(sql,new Object[]{"zhh",5,51});
  • 删除
String sql="delete from user where id=?";
jdbcTemplate.update(sql,51);
  • batchUpdate()批量插入、更新和删除方法
String sql="insert into user (name,deptid) values (?,?)";
List<Object[]> batchArgs=new ArrayList<Object[]>();
batchArgs.add(new Object[]{"caoyc",6});
batchArgs.add(new Object[]{"zhh",8});
batchArgs.add(new Object[]{"cjx",8});
jdbcTemplate.batchUpdate(sql, batchArgs);
  • 读取单个对象
String sql="select id,name,deptid from user where id=?";
RowMapper<User> rowMapper=new BeanPropertyRowMapper<User>(User.class);
User user= jdbcTemplate.queryForObject(sql, rowMapper,52);
System.out.println(user);
  • 读取多个对象
String sql="select id,name,deptid from user";
RowMapper<User> rowMapper=new BeanPropertyRowMapper<User>(User.class);
List<User> users= jdbcTemplate.query(sql, rowMapper);
for (User user : users) {
    System.out.println(user);
}
  • 获取某个记录某列或者count、avg、sum等函数返回唯一值
String sql="select count(*) from user";
int count= jdbcTemplate.queryForObject(sql, Integer.class);
System.out.println(count);
  • NamedParameterJdbcTemplate
    NamedParameterJdbcTemplate和JdbcTemplate功能基本差不多。使用方法也类型。下面具体看下代码。
  1. 方式1
//为变量名称前面加上冒号
String sql="insert into user (name,deptid) values (:name,:deptid)";
//定义map集合,其参数名称为sql语句中变量的名称
Map<String,Object> paramMap=new HashMap<String,Object>();
paramMap.put("name", "caoyc");
paramMap.put("deptid", 2);
namedParameterJdbcTemplate.update(sql, paramMap);
  1. 方式2
//为变量名称前面加上冒号
String sql="insert into user (name,deptid) values (:name,:deptid)";
//定义个实体类
User user=new User();
user.setName("zhh");
user.setDeptid(3);
SqlParameterSource paramSource=new BeanPropertySqlParameterSource(user);
namedParameterJdbcTemplate.update(sql, paramSource);
  • hibernate使用
  1. 通过注入sessionFactory,通过setSessionFactory()方法实现
    编写dao继承HibernateDaoSupport, 注入sessionFactory创建。
@Repository
public class SysUserDaoImpl extends HibernateDaoSupport implements SysUserDao {
    @Autowired
    public void setSessionFactor(SessionFactory sessionFactory) {
        super.setSessionFactory(sessionFactory);
    }

    @Override
    public SysUser findSysUserById(String id) {
        return this.getHibernateTemplate().get(SysUser.class, id);
    }

}

//xml配置
<!-- 数据源 ,使用c3p0 -->
<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}" />
    <!-- 最大连接数,建议在开发环境中设置小一点,够用即可 -->
    <property name="maxPoolSize" value="3" />
    <!-- 最小连接数 -->
    <property name="minPoolSize" value="1" />
</bean>


<!-- sessionFactory -->
<bean id="sessionFactory" class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
    <!-- 数据库源 -->
    <property name="dataSource" ref="dataSource" />
    
    <!-- 设置hibernate属性 -->
    <property name="hibernateProperties">
        <props>
            <!-- 方言 -->
            <prop key="hibernate.dialect">org.hibernate.dialect.Oracle10gDialect</prop>
            <!-- 日志输出显示sql -->
            <prop key="hibernate.show_sql">true</prop>
            <!-- 日志输出中格式化sql -->
            <prop key="hibernate.format_sql">true</prop>
            <!-- 从hbm文件到数据库定义策略,建议设置为none,不检查po类和表结构是否一致 -->
            <prop key="hibernate.hbm2ddl.auto">none</prop>
            <!-- 配置Hibernate的session管理机制,spring与hibernate整合后采用当session与当前线程绑定 -->
            <prop key="hibernate.current_session_context_class">org.springframework.orm.hibernate5.SpringSessionContext</prop>
        </props>
    </property>
    <!-- 加载映射文件 -->
    <property name="mappingLocations">
        <list>
            <value>classpath:com/xxx/xxx/*.hbm.xml</value>
        </list>
    </property>
</bean>
  1. 通过注入hibernateTemplate实现,通过setHibernateTemplate()方法实现
@Repository
public class SysUserDaoImpl extends HibernateDaoSupport implements SysUserDao {
    @Autowired
    public void setHT(HibernateTemplate hibernateTeplate) {
        super.setHibernateTemplate(hibernateTeplate);
    }

    @Override
    public SysUser findSysUserById(String id) {
        return this.getHibernateTemplate().get(SysUser.class, id);
    }
}

//xml配置
<bean id="hibernateTemplate" class="org.springframework.orm.hibernate5.HibernateTemplate">
	<!-- spring与hibernate整合后默认必须在事务下对数据库执行写操作,如果将checkWriteOperations设置为false则没有事务也可写数据库,这样是为了单独测试dao方便 -->
	 <property name="checkWriteOperations" value="false"/>
	 <property name="sessionFactory" ref="sessionFactory"/>
</bean>
  • 事务管理
  1. 事务的隔离级别
  2. 事务的传播行为
  3. 编程事务
  • demo demo
  • 第二种TransactionTemplate实现
@Resource   
private  TransactionTemplate transaction;

private void f(){    
	transaction.execute(new TransactionCallback<Void>() {        
	@Override           
	publicVoid doInTransaction(TransactionStatus status) {              
	 // update1                
	 // update2               
	 returnnull;
	 }        });
}

//xml配置
<!-- Initialization for data source -->
   <bean id="dataSource" 
      class="org.springframework.jdbc.datasource.DriverManagerDataSource">
      <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
      <property name="url" value="jdbc:mysql://localhost:3306/TEST"/>
      <property name="username" value="root"/>
      <property name="password" value="password"/>
   </bean>

   <!-- Initialization for TransactionManager -->
   <bean id="transactionManager" 
      class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
      <property name="dataSource"  ref="dataSource" />    
   </bean>

<!-- Definition for studentJDBCTemplate bean -->
<bean id="transactionTemplate"
   class="org.springframework.transaction.support.TransactionTemplate">
   <property name="dataSource"  ref="dataSource" />
   <property name="transactionManager"  ref="transactionManager" />    
</bean>
  1. 声明式事务
  • 基于 TransactionProxyFactoryBean的声明式事务管理
<bean id="buyStockService" class="transaction.test2.service.BuyStockServiceImpl">
        <property name="accountDao" ref="accountDao"></property>
        <property name="stockDao" ref="stockDao"></property>
    </bean>
  
    <!-- 事务管理器 -->
<bean id="myTracnsactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"></property>
</bean>
    
    <!-- 事务代理工厂 -->
<!-- 生成事务代理对象 -->
<bean id="serviceProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
        <property name="transactionManager" ref="myTracnsactionManager"></property>
        <property name="target" ref="buyStockService"></property>
        <property name="transactionAttributes">
            <props>
                <!-- 主要 key 是方法   
                    ISOLATION_DEFAULT  事务的隔离级别
                    PROPAGATION_REQUIRED  传播行为
                -->
                <prop key="add*">ISOLATION_DEFAULT,PROPAGATION_REQUIRED</prop>
                <!-- -Exception 表示发生指定异常回滚,+Exception 表示发生指定异常提交 -->
                <prop key="buyStock">ISOLATION_DEFAULT,PROPAGATION_REQUIRED,-BuyStockException</prop>
            </props>
        </property>
</bean>
  • 基于 @Transactional 的声明式事务管理
<!-- 事务管理器 -->
<bean id="myTracnsactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"></property>
</bean>
    
<!-- 启用事务注解 -->
 <tx:annotation-driven transaction-manager="myTracnsactionManager"/>
  • 基于Aspectj AOP配置事务
<!-- 事务管理器 -->
<bean id="myTracnsactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"></property>
</bean>
    
 <tx:advice id="txAdvice" transaction-manager="myTracnsactionManager">
     <tx:attributes>
         <!-- 为连接点指定事务属性 -->
         <tx:method name="add*" isolation="DEFAULT" propagation="REQUIRED"/>
         <tx:method name="buyStock" isolation="DEFAULT" propagation="REQUIRED" rollback-for="BuyStockException"/>
     </tx:attributes>
 </tx:advice>
    
 <aop:config>
     <!-- 切入点配置 -->
     <aop:pointcut expression="execution(* *..service.*.*(..))" id="point"/>
     <aop:advisor advice-ref="txAdvice" pointcut-ref="point"/>
 </aop:config>

spring mvc

控制器controller

配置DispatcherServlet到servlet容器内,而不再配置到web.xml中

  • 配置DispatcherServlet
public class SpittrWebAppInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {

    //配置根配置类
    @Override
    protected Class<?>[] getRootConfigClasses() {
        //return new Class[]{RootConfid.class};
        return new Class[0];
    }

    //指定servlet配置类
    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[]{WebConfig.class};
    }

    //将DispatcherServlet映射到 '/' 目录。
    @Override
    protected String[] getServletMappings() {
        return new String[] {"/"};
    }
}
  • 控制器demo
@Controller
@RequestMapping("/spittles") //类级别的请求处理
public class SpittrController {
    @RequestMapping(value = "/test", method= RequestMethod.GET)
    //传递model模型数据到视图中。
    public String test(Model model) {
        model.addAttribute("test", "hello");
        return "test";
    }
}
  • 接受请求参数方式
  1. 查询参数(query parameter)
    @RequestParam
@RequestMapping("/test01")
public void test01(
	@RequestParam(value="max", defaultValue="20") long max,
	@RequestParam("count") long count) { 
}
  1. 路径参数(path parameter)
@RequestMapping("/test02/{id}")
public void test02(@PathVariable("id") long id) {
}
  1. 表单参数(form parameter)
// obj代表任意java对象,表单参数可封装到java对象中去。
@RequestMapping(value = "/test03/save", method = RequestMethod.POST)
public String test03(Object obj) {
	return "redirect:/spitter/name";
	//return "forward:/spitter/name";
}
  1. 校验表单
    这些注解可以放到属性上,限制这些属性的值。
  • 配置demo
public class Spitter {
    @NonNull
    @Size(min=0, max=16)
    private String username;
}

//controller
@RequestMapping(value = "/test04", method = RequestMethod.POST)
public void test04(@Validated Spitter obj) {
}

渲染Web视图

  • ViewResolver视图解析器
public interface ViewResolver {
    View resolveViewName(String var1, Locale var2) throws Exception;
}
  • View
public interface View {
    String RESPONSE_STATUS_ATTRIBUTE = View.class.getName() + ".responseStatus";
    String PATH_VARIABLES = View.class.getName() + ".pathVariables";
    String SELECTED_CONTENT_TYPE = View.class.getName() + ".selectedContentType";

    String getContentType();

    void render(Map<String, ?> var1, HttpServletRequest var2, HttpServletResponse var3) throws Exception;
}
  • 创建jsp视图
  1. demo
public ViewResolver viewResolver() {
	InternalResourceViewResolver internalResourceViewResolver = new InternalResourceViewResolver();
	//前缀
	internalResourceViewResolver.setPrefix("/WEB-INF/views/");
	//后缀
	internalResourceViewResolver.setSuffix(".jsp");
	//解析成jstlView
	internalResourceViewResolver.setViewClass(JstlView.class);
		
	return internalResourceViewResolver;
}
  1. spring下的jsp标签库
  • 表单



    注意:
  1. <sf:errors path=“firstname”>标签的使用,可以验证表单数据输入有误后,展示的错误信息。
  2. 这里不再详细介绍jsp了,使用较少了。
  • 使用apache Tiles视图定义布局
  1. tilesConfigurer + tilesViewResolver 实现页面布局
@Bean
public TilesConfigurer tilesConfigurer() {
  TilesConfigurer tilesConfigurer = new TilesConfigurer();
  tilesConfigurer.setDefinitions(new String[] {
          "/WEB-INF/**/tiles.xml"
  });
  tilesConfigurer.setCheckRefresh(true); //启用刷新

  return tilesConfigurer;
}

@Bean
public TilesViewResolver tilesViewResolver() {
  return new TilesViewResolver();
}
  1. 定义tiles及使用
  • 使用Thymeleaf
  1. demo,此处仅列举一个demo供参考

spring mvc 高级用法

  • 添加其他的servlet和filter
  • springdatajpa开发手册 spring开发文档_spring_02


  • springdatajpa开发手册 spring开发文档_java_03


  • spring mvc 在 web.xml的配置
  • springdatajpa开发手册 spring开发文档_spring_04


  • 处理multipart格式的数据
  1. 基于servlet3以上的实现
@Bean
public MultipartResolver multipartResolver() {
    return new StandardServletMultipartResolver();
}
  1. 配置上传文件
  • 使用servlet初始化类
public class MyInitConfig implements WebApplicationInitializer {
    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
        DispatcherServlet dispatcherServlet = new DispatcherServlet();

        ServletRegistration.Dynamic appServlet = servletContext.addServlet("appServlet", dispatcherServlet);

        appServlet.setLoadOnStartup(1);
        appServlet.setMultipartConfig(new MultipartConfigElement("/tmp/uploads"));
    }
}
  • 不适用servlet初始化类,重载customizeRegistration()方法
@Override
protected void customizeRegistration(ServletRegistration.Dynamic registration) {
    super.customizeRegistration(registration);
    registration.setMultipartConfig(new MultipartConfigElement("/tmp/uploads"));
}
  • 使用传统web.xml配置
  • springdatajpa开发手册 spring开发文档_xml配置_05


  1. 使用FileUpload 处理文件上传
  2. springdatajpa开发手册 spring开发文档_xml_06


  3. 处理文件上传请求
  • @RequestPart
@RequestMapping("/file01")
public void file01(@RequestPart("file") byte[] file) {
 }
  • MultipartFile
@RequestMapping("/file02")
public void file02(MultipartFile file) {
}
  • part形式处理
@RequestMapping("/file03")
public void file03(@RequestPart("file") Part file) {
}

处理异常

  • 将异常映射到Http状态码。以下为自定义异常
@ResponseStatus(value= HttpStatus.NOT_FOUND,
        reason = "Not Found")
public class MyException extends RuntimeException{
}
  • 编写异常处理方法
@ExceptionHandler(MyException.class)
public String handleException() {
     return "error";
 }

注意:异常情况是MyException时,会调用此方法。

  • 为控制器添加通知
  • springdatajpa开发手册 spring开发文档_springdatajpa开发手册_07


保护web应用

后续补充。。。

QA

  • 事务的传播行为
  1. 上图前三个,保证业务在同一个事务管理下。
  2. 中间三个,保证业务不再同一个事务管理
  3. 最后一个,第一个事务执行完后,会创建savepoint保存点,如果第二个事务出现异常,可以选择回滚到保存点,也可以回滚到最初的状态。