zoukankan      html  css  js  c++  java
  • JAVA配置&注解方式搭建简单的SpringMVC前后台交互系统

    前面两篇文章介绍了 基于XML方式搭建SpringMVC前后台交互系统的方法,博文链接如下:
      http://www.cnblogs.com/hunterCecil/p/8252060.html
      http://www.cnblogs.com/hunterCecil/p/6924935.html

    本文重点介绍一下基于java配置+注解方式搭建系统的过程步骤。


    一、摘要
      1、所需软件列表:
        1) tomcat : apache-tomcat-7.0.54  服务端容器
        2) Intellij: Intellij IDEA 2016.3.1    开发工具  
        3) Syslog: SQLyog Community    数据库工具
      2. 步骤简述:
       1) 新建一个 Java 项目, 按下图依次选择:
         
       2) 项目目录如下:
        
    各目录存储文件如下:

    (1)java.com : 目录下用于存放后台代码:

        config: 配置启动类
        controller: 应用层实现类
        dao: 数据访问层的bean
        entity: 对象实体类
        service: 业务层bean, 具体的数据库操作实现类
    (2)resource:
        jdbc.properties: 数据库配置
        log4j.properties: 日志配置相关属性
        users.sql: 建表语句

    (3)webapp: 客户端展示相关文件

    二、 相关配置文件
      1. web.xml配置(用于配置Spring 相关配置项:上下文、日志、字符集、监听器等通过配置文件加载,这里只配置welcome-file-list及session-config)

    <?xml version="1.0" encoding="UTF-8"?>
    <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    	xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
    	version="3.0" metadata-complete="false">
    
    	<!-- session超时 -->
    	<session-config>
    		<session-timeout>60</session-timeout>
    	</session-config>
    	<!-- 欢迎页面 -->;
    	<welcome-file-list>
    		<welcome-file>index.jsp</welcome-file>
    	</welcome-file-list>
    	
    </web-app>
    

      2. jdbc. properties

    jdbc.driver = com.mysql.jdbc.Driver
    db.url = jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=UTF-8
    db.username = root
    db.password = *****
    

      3. log4j.properties

    #日志输出级别
    log4j.rootLogger=INFO,stdout,other
    
    #设置stdout的日志输出控制台
    log4j.appender.stdout=org.apache.log4j.ConsoleAppender
    #输出日志到控制台的方式,默认为System.out
    log4j.appender.stdout.Target = System.out
    #设置使用灵活布局
    log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
    #灵活定义输出格式
    log4j.appender.stdout.layout.ConversionPattern=[%p][%d{yyyy-MM-dd HH:mm:ss}] %l %m %n
    
    #设置other的日志输出控制台
    log4j.appender.other=org.apache.log4j.RollingFileAppender
    #设置other的输出日志
    log4j.appender.other.File=/WEB-INF/logs/log.log
    #设置other的日志最大限制
    log4j.appender.other.MaxFileSize=1024KB
    #最多只保存20个备份文件
    log4j.appender.other.MaxBackupIndex=1000
    #输出INFO级别以上的日志
    og4j.appender.other.Threshold=INFO
    #设置使用灵活布局
    log4j.appender.other.layout=org.apache.log4j.PatternLayout
    #灵活定义输出格式
    log4j.appender.other.layout.ConversionPattern=[%p][%d{yyyy-MM-dd HH:mm:ss}] %l %t %m %n
    

      4. uses.sql

    -- ----------------------------
    -- Table structure for user
    -- ----------------------------
    DROP TABLE IF EXISTS `user`;
    CREATE TABLE `user` (
      `id` tinyint (20),
    	`name` varchar (90),
    	`age` tinyint (100),
      PRIMARY KEY (`id`),
    ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;
    
    
    -- ----------------------------
    -- Records of user
    -- ----------------------------
    insert into `user` (`id`, `name`, `age`) values('2','zhangsan','32');
    insert into `user` (`id`, `name`, `age`) values('3','lisi','33');
    insert into `user` (`id`, `name`, `age`) values('1','wangwu','12'); 

     5. config目录下文件:
      DatabaseConfig 

    @Configuration
    @EnableTransactionManagement
    public class DatabaseConfig {
        private static final Logger logger = Logger.getLogger(DatabaseConfig.class.getName());
    
        @Value("${jdbc.driver}")
        private String jdbcDriver;
    
        @Value("${db.url}")
        private String dbUrl;
    
        @Value("${db.username}")
        private String username;
    
        @Value("${db.password}")
        private String password;
    
        @Bean(destroyMethod = "close")
        public DataSource dataSource() {
            logger.info("mysql url:"+dbUrl);
            BasicDataSource dataSource = new BasicDataSource();
            dataSource.setDriverClassName(jdbcDriver);
            dataSource.setUrl(dbUrl);
            dataSource.setUsername(username);
            dataSource.setPassword(password);
            return dataSource;
        }
    
        @Bean
        public DataSourceTransactionManager txManager() {
            return new DataSourceTransactionManager(dataSource());
        }
    
        @Bean
        public SqlSessionFactory sqlSessionFactory() throws Exception {
            SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
            sessionFactory.setDataSource(dataSource());
            return sessionFactory.getObject();
        }
    }
    

     PropertiesConfig

    @Configuration
    @PropertySource(value={"classpath:jdbc.properties"})
    public class PropertiesConfig {
        @Bean
        public static PropertySourcesPlaceholderConfigurer propertyConfigInDev() {
            return new PropertySourcesPlaceholderConfigurer();
        }
    }
    

     RootConfig

    @Configuration
    @ComponentScan(basePackages = {"com"})
    public class RootConfig {
    }
    

     WebConfig

    @Configuration
    @EnableWebMvc
    @EnableAspectJAutoProxy
    @EnableTransactionManagement
    @ComponentScan("com.controller")
    @MapperScan("com.dao")
    public class WebConfig extends WebMvcConfigurerAdapter {
        @Bean
        public ViewResolver viewResolver() {
            InternalResourceViewResolver resolver = new InternalResourceViewResolver();
            resolver.setPrefix("/WEB-INF/views/");
            resolver.setSuffix(".jsp");
            resolver.setExposeContextBeansAsAttributes(true);
            return resolver;
        }
    
        @Bean
        public StringHttpMessageConverter stringHttpMessageConverter(){
            return new StringHttpMessageConverter();
        }
    
        @Bean
        public MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter(){
            return new MappingJackson2HttpMessageConverter();
        }
    
        @Override
        public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
            converters.add(stringHttpMessageConverter());
            converters.add(mappingJackson2HttpMessageConverter());
            super.configureMessageConverters(converters);
        }
    
        /**
         * 启用spring mvc 的注解
         */
        public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
            configurer.enable();
        }
    }
    

      WebAppInitializer

    /**
     * 配置SpringMVC需要添加DispatchServlet ,
     * DispatcherServlet主要负责前端调用URL的分发,
     * 它在Web容器初始化的时候被注册。
     * 这里配置中继承了DispatchServlet 的一个抽象类AbstractAnnotationConfigDispatcherServletInitializer ,
     * 此抽象类的父类在实例化的时候会注册一个DispatchServlet到容器中
     * 这里配置了容器初始化时需要加载的配置类 RootConfig,PropertiesConfig  和 WebConfig
     */
    public class WebAppInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
        /** 应用上下文、配置文件读取,数据库配置*/
        @Override
        protected Class<?>[] getRootConfigClasses() {
            System.out.println("======1.应用上下文、配置文件读取,数据库配置 load ================");
            return new Class<?>[]{RootConfig.class, PropertiesConfig.class, DatabaseConfig.class};
        }
    
        /* web上下文  */
        @Override
        protected Class<?>[] getServletConfigClasses() {
            System.out.println("======2.web上下文 load ================");
            return new Class<?>[]{WebConfig.class};
        }
    
        /*  DispatcherServlet的映射路径 */
        @Override
        protected String[] getServletMappings() {
            System.out.println("======3.DispatcherServlet的映射路径 load ================");
            return new String[]{"/"};
        }
    
        /* 注册过滤器,映射路径与DispatcherServlet一致,路径不一致的过滤器需要注册到另外的WebApplicationInitializer中*/
        @Override
        protected Filter[] getServletFilters() {
            CharacterEncodingFilter characterEncodingFilter = new CharacterEncodingFilter();
            characterEncodingFilter.setEncoding("UTF-8");
            characterEncodingFilter.setForceEncoding(true);
            return new Filter[]{characterEncodingFilter};
        }
    }
    

      6.  controller-TestController

    @Controller
    public class TestController {
    
        @Autowired
        @Qualifier("userService")
        private UserService userService;
    
        @RequestMapping(value = "/test", method = RequestMethod.GET)
        public String test() {
            return "test";
        }
    
        @RequestMapping(value = "/getuser/{username}", method = RequestMethod.GET)
        @ResponseBody
        public User getUser(@PathVariable String username) {
            return  userService.loadUserByUsername(username);
        }
    }
    

      7. dao-UserDao

    @Controller
    public class TestController {
    
        @Autowired
        @Qualifier("userService")
        private UserService userService;
    
        @RequestMapping(value = "/test", method = RequestMethod.GET)
        public String test() {
            return "test";
        }
    
        @RequestMapping(value = "/getuser/{username}", method = RequestMethod.GET)
        @ResponseBody
        public User getUser(@PathVariable String username) {
            return  userService.loadUserByUsername(username);
        }
    }
    

      8. entity-User

    public class User {
        private Integer id;
        private String name;
        private Integer age;
    
        public Integer getId() {
            return id;
        }
    
        public void setId(Integer id) {
            this.id = id;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    }
    

      9. log-logAspect

    @Component
    @Aspect
    public class LogAspect {
        private static final Logger logger = Logger.getLogger(LogAspect.class.getName());
    
        @Pointcut("execution(* com.controller.*.*(..))")
        public void performance() {
        }
    
        @Before("performance()")
        public void optionBefore(JoinPoint joinPoint){
            System.out.println("@Before开始执行"+joinPoint.getSignature().getDeclaringTypeName() +
                    "." + joinPoint.getSignature().getName());
        }
    
        @After("performance()")
        public void optionAfter(JoinPoint joinPoint){
            System.out.println("@After结束执行"+joinPoint.getSignature().getDeclaringTypeName() +
                    "." + joinPoint.getSignature().getName());
        }
        @Around("performance()")
        public Object optionAround(ProceedingJoinPoint pjp) throws Throwable {
            Object retVal = null;
            System.out.println("@Around start..");
            try {
                retVal = pjp.proceed();
            } catch (Throwable ex) {
                System.out.println("error in @Around");
                throw ex;
            }
            System.out.println("@Around end");
            return retVal;
        }
    
        @AfterReturning(pointcut = "performance()", returning = "returnValue")
        public void optionAfterReturn(JoinPoint joinPoint, Object returnValue) {
            System.out.println("@AfterReturning:目标方法为:" +
                    joinPoint.getSignature().getDeclaringTypeName() +
                    "." + joinPoint.getSignature().getName());
            System.out.println("@AfterReturning:参数为:" +
                    Arrays.toString(joinPoint.getArgs()));
            System.out.println("@AfterReturning:返回值为:" + returnValue);
            System.out.println("@AfterReturning:被织入的目标对象为:" + joinPoint.getTarget());
    
        }
    
        @AfterThrowing(pointcut = "performance()", throwing = "e")
        public void optionError(JoinPoint joinPoint, Throwable e) {
            logger.log(Level.WARNING,"异常方法:{" + joinPoint.getTarget().getClass().getName() +
                    "." + joinPoint.getSignature().getName() + "}" +
                    "异常类型:{" + e.getClass().getName() + "}" +
                    "异常信息:{" + e.getMessage() + "}" +
                    "参数:{" + Arrays.toString(joinPoint.getArgs()) + "}", e);
        }
    }
    

      10. service-UserService

    public interface UserService {
    
        User loadUserByUsername(String username);
    
        void saveUser(User user);
    }
    

       11. service-impl-UserServiceImpl

    @Service("userService")
    public class UserServiceImpl implements UserService {
    
        @Autowired
        private UserDao userDao;
    
        @Transactional(readOnly = true)
        public User loadUserByUsername(String username) {
            return userDao.loadUserByUsername(username);
        }
    
        @Transactional
        public void saveUser(User user) {
            userDao.saveUser(user);
        }
    }
    

      12. maven 对应的pom文件如下:

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
        <modelVersion>4.0.0</modelVersion>
        <groupId>com</groupId>
        <artifactId>springJavaConfig</artifactId>
        <packaging>war</packaging>
        <version>1.0-SNAPSHOT</version>
        <name>springJavaConfig Maven Webapp</name>
        <url>http://maven.apache.org</url>
        <properties>
            <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
            <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
            <java.version>1.6</java.version>
        </properties>
        <dependencies>
            <dependency>
                <!--Unit Test -->
                <groupId>junit</groupId>
                <artifactId>junit</artifactId>
                <version>4.10</version>
                <scope>test</scope>
            </dependency>
            <!-- Servlet -->
            <dependency>
                <groupId>javax.servlet</groupId>
                <artifactId>javax.servlet-api</artifactId>
                <version>3.1.0</version>
                <scope>provided</scope>
            </dependency>
            <!-- SpringMVC -->
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-webmvc</artifactId>
                <version>4.3.2.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-jdbc</artifactId>
                <version>4.3.2.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>mysql</groupId>
                <artifactId>mysql-connector-java</artifactId>
                <version>5.1.6</version>
            </dependency>
        <!--Mybatis--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.3.0</version> </dependency> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis-spring</artifactId> <version>1.2.3</version> </dependency> <dependency> <groupId>org.apache.commons</groupId> <artifactId>commons-dbcp2</artifactId> <version>2.1.1</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.8.0</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>2.8.5</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>2.8.5</version> </dependency> </dependencies> <build> <finalName>springJavaConfig</finalName> </build> </project>

     三. 启动过程打印日志:

    Connected to server
    [2018-01-17 10:47:20,950] Artifact springJavaConfig:war exploded: Artifact is being deployed, please wait...
    一月 17, 2018 10:47:23 上午 org.apache.catalina.startup.TldConfig execute
    信息: At least one JAR was scanned for TLDs yet contained no TLDs. Enable debug logging for this logger for a complete list of JARs that were scanned but no TLDs were found in them. Skipping unneeded JARs during scanning can improve startup time and JSP compilation time.
    ======1.应用上下文、配置文件读取,数据库配置 load ================
    ======2.web上下文 load ================
    ======3.DispatcherServlet的映射路径 load ================
    一月 17, 2018 10:47:24 上午 org.springframework.web.context.ContextLoader initWebApplicationContext
    信息: Root WebApplicationContext: initialization started
    一月 17, 2018 10:47:24 上午 org.springframework.web.context.support.AnnotationConfigWebApplicationContext prepareRefresh
    信息: Refreshing Root WebApplicationContext: startup date [Wed Jan 17 10:47:24 CST 2018]; root of context hierarchy
    一月 17, 2018 10:47:24 上午 org.springframework.web.context.support.AnnotationConfigWebApplicationContext loadBeanDefinitions
    信息: Registering annotated classes: [class com.config.RootConfig,class com.config.PropertiesConfig,class com.config.DatabaseConfig]
    一月 17, 2018 10:47:25 上午 com.config.DatabaseConfig dataSource
    信息: mysql url:jdbc:mysql://10.42.120.201:3306/test?useUnicode=true&characterEncoding=UTF-8
    一月 17, 2018 10:47:25 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping register
    信息: Mapped "{[/test],methods=[GET]}" onto public java.lang.String com.controller.TestController.test()
    一月 17, 2018 10:47:25 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping register
    信息: Mapped "{[/getuser/{username}],methods=[GET]}" onto public com.entity.User com.controller.TestController.getUser(java.lang.String)
    一月 17, 2018 10:47:25 上午 org.springframework.web.servlet.handler.SimpleUrlHandlerMapping registerHandler
    信息: Mapped URL path [/**] onto handler of type [class org.springframework.web.servlet.resource.DefaultServletHttpRequestHandler]
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter initControllerAdviceCache
    信息: Looking for @ControllerAdvice: Root WebApplicationContext: startup date [Wed Jan 17 10:47:24 CST 2018]; root of context hierarchy
    一月 17, 2018 10:47:26 上午 org.springframework.web.context.ContextLoader initWebApplicationContext
    信息: Root WebApplicationContext: initialization completed in 2370 ms
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.DispatcherServlet initServletBean
    信息: FrameworkServlet 'dispatcher': initialization started
    一月 17, 2018 10:47:26 上午 org.springframework.web.context.support.AnnotationConfigWebApplicationContext prepareRefresh
    信息: Refreshing WebApplicationContext for namespace 'dispatcher-servlet': startup date [Wed Jan 17 10:47:26 CST 2018]; parent: Root WebApplicationContext
    一月 17, 2018 10:47:26 上午 org.springframework.web.context.support.AnnotationConfigWebApplicationContext loadBeanDefinitions
    信息: Registering annotated classes: [class com.config.WebConfig]
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping register
    信息: Mapped "{[/test],methods=[GET]}" onto public java.lang.String com.controller.TestController.test()
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping register
    信息: Mapped "{[/getuser/{username}],methods=[GET]}" onto public com.entity.User com.controller.TestController.getUser(java.lang.String)
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.handler.SimpleUrlHandlerMapping registerHandler
    信息: Mapped URL path [/**] onto handler of type [class org.springframework.web.servlet.resource.DefaultServletHttpRequestHandler]
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter initControllerAdviceCache
    信息: Looking for @ControllerAdvice: WebApplicationContext for namespace 'dispatcher-servlet': startup date [Wed Jan 17 10:47:26 CST 2018]; parent: Root WebApplicationContext
    一月 17, 2018 10:47:26 上午 org.springframework.web.servlet.DispatcherServlet initServletBean
    信息: FrameworkServlet 'dispatcher': initialization completed in 356 ms
    [2018-01-17 10:47:26,774] Artifact springJavaConfig:war exploded: Artifact is deployed successfully
    [2018-01-17 10:47:26,775] Artifact springJavaConfig:war exploded: Deploy took 5,825 milliseconds 

    可以看到,WebAppInitializer类中我们的配置类已经按顺序加载过了。

    四、 测试

    url 输入 http://localhost:8080/getuser/zhangsan,可以查看切面部分打印日志如下:

    @Around start..
    @Before开始执行com.controller.TestController.getUser
    @Around end
    @After结束执行com.controller.TestController.getUser
    @AfterReturning:目标方法为:com.controller.TestController.getUser
    @AfterReturning:参数为:[zhangsan]
    @AfterReturning:返回值为:com.entity.User@182bab1a
    @AfterReturning:被织入的目标对象为:com.controller.TestController@24b0c989
    

     

  • 相关阅读:
    Java CompletableFuture:allOf等待所有异步线程任务结束
    多线程分批处理数据
    ListSplitUtil (批量数据处理)
    mysql性能优化相关
    JsonUtil
    批量数据数据处理
    多集合取交集公用方法
    ALSA:Linux下声卡编程
    word
    安卓使用自己导入的db3数据库文件
  • 原文地址:https://www.cnblogs.com/hunterCecil/p/8302783.html
Copyright © 2011-2022 走看看