现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

一、大纲

  1. 了解Spring的发展
  2. 掌握Spring的java配置方式
  3. 学习Spring Boot
  4. 使用Spring Boot来改造购物车系统

二、Spring的发展

2.1 Spring1.x 时代

在Spring1.x时代,都是通过xml文件配置bean,随着项目的不断扩大,需要将xml配置分放到不同的配置文件中,需要频繁的在java类和xml配置文件中切换。

2.2 Spring2.x时代

随着JDK 1.5带来的注解支持,Spring2.x可以使用注解对Bean进行申明和注入,大大的减少了xml配置文件,同时也大大简化了项目的开发。

那么,问题来了,究竟是应该使用xml还是注解呢?

最佳实践:

  1. 应用的基本配置用xml,比如:数据源、资源文件等;
  2. 业务开发用注解,比如:Service中注入bean等;

2.3 Spring3.x到Spring4.x

从Spring3.x开始提供了Java配置方式,使用Java配置方式可以更好的理解你配置的Bean,现在我们就处于这个时代,并且Spring4.x和Spring boot都推荐使用java配置的方式。

三、Spring的Java配置方式

Java配置是Spring4.x推荐的配置方式,可以完全替代xml配置。

3.1 @Configuration 和 @Bean

Spring的Java配置方式是通过 @Configuration 和 @Bean 这两个注解实现的:

  1. @Configuration 作用于类上,相当于一个xml配置文件;
  2. @Bean 作用于方法上,相当于xml配置中的;

3.2 示例

该示例演示了通过Java配置的方式进行配置Spring,并且实现了Spring IOC功能。

3.2.1 创建工程以及导入依赖

<code><project>\txsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
\t<modelversion>4.0.0/<modelversion>
\t<groupid>cn.itcast.springboot/<groupid>
\t<artifactid>itcast-springboot/<artifactid>
\t<version>1.0.0-SNAPSHOT/<version>
\t<packaging>war/<packaging>
\t<dependencies>
\t\t<dependency>
\t\t\t<groupid>org.springframework/<groupid>
\t\t\t<artifactid>spring-webmvc/<artifactid>
\t\t\t<version>4.3.7.RELEASE/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>com.jolbox/<groupid>
\t\t\t<artifactid>bonecp-spring/<artifactid>
\t\t\t<version>0.8.0.RELEASE/<version>
\t\t/<dependency>
\t/<dependencies>
\t<build>
\t\t<finalname>${project.artifactId}/<finalname>
\t\t<plugins>
\t\t\t
\t\t\t<plugin>
\t\t\t\t<groupid>org.apache.maven.plugins/<groupid>
\t\t\t\t<artifactid>maven-resources-plugin/<artifactid>
\t\t\t\t<configuration>
\t\t\t\t\t<encoding>UTF-8/<encoding>
\t\t\t\t/<configuration>
\t\t\t/<plugin>
\t\t\t
\t\t\t<plugin>
\t\t\t\t<groupid>org.apache.maven.plugins/<groupid>
\t\t\t\t<artifactid>maven-compiler-plugin/<artifactid>
\t\t\t\t<configuration>
\t\t\t\t\t<source>1.7/<source>
\t\t\t\t\t<target>1.7/<target>
\t\t\t\t\t<encoding>UTF-8/<encoding>
\t\t\t\t/<configuration>
\t\t\t/<plugin>
\t\t/<plugins>
\t\t<pluginmanagement>
\t\t\t<plugins>
\t\t\t\t
\t\t\t\t<plugin>

\t\t\t\t\t<groupid>org.apache.tomcat.maven/<groupid>
\t\t\t\t\t<artifactid>tomcat7-maven-plugin/<artifactid>
\t\t\t\t\t<version>2.2/<version>
\t\t\t\t/<plugin>
\t\t\t/<plugins>
\t\t/<pluginmanagement>
\t/<build>
/<project>/<code>

3.2.2 编写User对象

<code>public class User {
\tprivate String username;
\tprivate String password;
\tprivate Integer age;
\tpublic String getUsername() {
\t\treturn username;
\t}
\tpublic void setUsername(String username) {
\t\tthis.username = username;
\t}
\tpublic String getPassword() {
\t\treturn password;
\t}
\tpublic void setPassword(String password) {
\t\tthis.password = password;
\t}
\tpublic Integer getAge() {
\t\treturn age;
\t}
\tpublic void setAge(Integer age) {
\t\tthis.age = age;
\t}
}/<code>

3.2.3 编写UserDAO 用于模拟与数据库的交互

<code>public class UserDAO {
\tpublic List<user> queryUserList(){
\t\tList<user> result = new ArrayList<user>();
\t\t// 模拟数据库的查询
\t\tfor (int i = 0; i < 10; i++) {
\t\t\tUser user = new User();
\t\t\tuser.setUsername("username_" + i);

\t\t\tuser.setPassword("password_" + i);
\t\t\tuser.setAge(i + 1);
\t\t\tresult.add(user);
\t\t}
\t\treturn result;
\t}
}/<user>/<user>/<user>/<code>

3.2.4 编写UserService 用于实现User数据操作业务逻辑

<code>@Service
public class UserService {
\t@Autowired // 注入Spring容器中的bean对象
\tprivate UserDAO userDAO;
\tpublic List<user> queryUserList() {
\t\t// 调用userDAO中的方法进行查询
\t\treturn this.userDAO.queryUserList();
\t}
}/<user>/<code>

3.2.5 编写SpringConfig 用于实例化Spring容器

<code>@Configuration //通过该注解来表明该类是一个Spring的配置,相当于一个xml文件
@ComponentScan(basePackages = "cn.itcast.springboot.javaconfig") //配置扫描包
public class SpringConfig {
\t@Bean // 通过该注解来表明是一个Bean对象,相当于xml中的<bean>
\tpublic UserDAO getUserDAO(){
\t\treturn new UserDAO();
\t\t// 直接new对象做演示
\t}
}/<bean>/<code>

3.2.6 编写测试方法 用于启动Spring容器

<code>public class Main {
\tpublic static void main(String[] args) {
\t\t// 通过Java配置来实例化Spring容器
\t\tAnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);
\t\t// 在Spring容器中获取Bean对象
\t\tUserService userService = context.getBean(UserService.class);
\t\t// 调用对象中的方法
\t\tList<user> list = userService.queryUserList();
\t\tfor (User user : list) {
\t\t\tSystem.out.println(user.getUsername() + ", " + user.getPassword() + ", " + user.getPassword());
\t\t}
\t\t// 销毁该容器
\t\tcontext.destroy();
\t}
}/<user>/<code>

3.2.7 测试效果

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

3.2.8 小结

从以上的示例中可以看出,使用Java代码就完美的替代xml配置文件,并且结构更加的清晰。

3.3 实战

3.3.1 读取外部的资源配置文件

通过@PropertySource可以指定读取的配置文件,通过@Value注解获取值,具体用法:

<code>@Configuration //通过该注解来表明该类是一个Spring的配置,相当于一个xml文件
@ComponentScan(basePackages = "cn.itcast.springboot.javaconfig") //配置扫描包
@PropertySource(value= {"classpath:jdbc.properties"})
public class SpringConfig {
\t@Value("${jdbc.url}")
\t private String jdbcUrl;
\t@Bean // 通过该注解来表明是一个Bean对象,相当于xml中的<bean>
\tpublic UserDAO getUserDAO(){
\t\treturn new UserDAO();
\t\t// 直接new对象做演示
\t}
}/<bean>/<code>

思考:

①. 如何配置多个配置文件?

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 如果配置的配置文件不存在会怎么样?

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

3.3.2 配置数据库连接池

导入依赖:

<code>
\t\t<dependency>
\t\t\t<groupid>com.jolbox/<groupid>
\t\t\t<artifactid>bonecp-spring/<artifactid>
\t\t\t<version>0.8.0.RELEASE/<version>

\t\t/<dependency>/<code>

之前的Spring xml配置:

<code>
\t<bean>\t\tdestroy-method="close">
\t\t
\t\t<property>
\t\t
\t\t<property>
\t\t
\t\t<property>
\t\t
\t\t<property>
\t\t
\t\t<property>
\t\t
\t\t<property>
\t\t
\t\t
\t\t<property>
\t\t
\t\t<property>
\t/<bean>
/<code>

参考xml配置改造成java配置方式:

<code>@Value("${jdbc.url}")
private String jdbcUrl;
@Value("${jdbc.driverClassName}")
private String jdbcDriverClassName;
@Value("${jdbc.username}")
private String jdbcUsername;
@Value("${jdbc.password}")
private String jdbcPassword;
@Bean(destroyMethod = "close")
public DataSource dataSource() {
\tBoneCPDataSource boneCPDataSource = new BoneCPDataSource();
\t// 数据库驱动
\tboneCPDataSource.setDriverClass(jdbcDriverClassName);
\t// 相应驱动的jdbcUrl
\tboneCPDataSource.setJdbcUrl(jdbcUrl);
\t// 数据库的用户名
\tboneCPDataSource.setUsername(jdbcUsername);
\t// 数据库的密码
\tboneCPDataSource.setPassword(jdbcUsername);
\t// 检查数据库连接池中空闲连接的间隔时间,单位是分,默认值:240,如果要取消则设置为0
\tboneCPDataSource.setIdleConnectionTestPeriodInMinutes(60);
\t// 连接池中未使用的链接最大存活时间,单位是分,默认值:60,如果要永远存活设置为0
\tboneCPDataSource.setIdleMaxAgeInMinutes(30);
\t// 每个分区最大的连接数
\tboneCPDataSource.setMaxConnectionsPerPartition(100);
\t// 每个分区最小的连接数
\tboneCPDataSource.setMinConnectionsPerPartition(5);
\treturn boneCPDataSource;
}/<code>

思考: 如何使用该DataSource对象?

四、Spring Boot

4.1 什么是Spring Boot

随着动态语言的流行(Ruby、Groovy、 Scala、 Node.js),Java 的开发显得格外的笨重:繁多的配置、低下的开发效率、复杂的部署流程以及第三方技术集成难度大。

在上述环境下,Spring Boot应运而生。它使用“习惯优于配置”(项目中存在大量的配置,此外还内置一个习惯性的配置,让你无须手动进行配置)的理念让你的项目快速运行起来。使用SpringBoot很容易创建一个独立运行(运行jar,内嵌Servlet容器)、准生产级别的基于Spring框架的项目,使用Spring Boot你可以不用或者只需要很少的Spring配置。

4.2 Spring Boot的优缺点

优点

  • 快速构建项目;
  • 对主流开发框架的无配置集成;
  • 项目可独立运行,无须外部依赖Servlet容器;
  • 提供运行时的应用监控;
  • 极大地提高了开发、部署效率;
  • 与云计算的天然集成。

缺点

  • 书籍文档较少且不够深入,这是直接促使我写这本书的原因;
  • 如果你不认同Spring框架,这也许是它的缺点,但建议你一定要使用Spring框架。

4.3 快速入门

4.3.1 设置spring boot的parent

<code><parent>
\t\t<groupid>org.springframework.boot/<groupid>
\t\t<artifactid>spring-boot-starter-parent/<artifactid>
\t\t<version>1.5.2.RELEASE/<version>
\t/<parent>/<code>

说明:Spring boot的项目必须要将parent设置为spring boot的parent,该parent包含了大量默认的配置,大大简化了我们的开发。

4.3.2 导入spring boot的web支持

<code><dependency>
\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t<artifactid>spring-boot-starter-web/<artifactid>
\t\t/<dependency>/<code>

4.3.3 添加Spring boot的插件

<code><plugin>
\t\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t\t<artifactid>spring-boot-maven-plugin/<artifactid>
\t\t\t/<plugin>/<code>

4.3.4 编写第一个Spring Boot的应用

<code>@Controller
@SpringBootApplication
@Configuration
public class HelloApplication {
\t@RequestMapping("hello")
\t @ResponseBody
\t public String hello(){
\t\treturn "hello world!";
\t}
\tpublic static void main(String[] args) {
\t\tSpringApplication.run(HelloApplication.class, args);
\t}
}/<code>

代码说明:

  1. @SpringBootApplication:Spring Boot项目的核心注解,主要目的是开启自动配置。;
  2. @Configuration:这是一个配置Spring的配置类;
  3. @Controller:标明这是一个SpringMVC的Controller控制器;
  4. main方法:在main方法中启动一个应用,即:这个应用的入口;

4.3.5 启动应用

在Spring Boot项目中,启动的方式有两种,一种是直接run Java Application另外一种是通过Spring Boot的Maven插件运行。

第一种:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

第二种:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

启动效果:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

看到如下信息就说明启动成功了:

<code>INFO 6188 --- [           main] c.i.springboot.demo.HelloApplication     : Started HelloApplication in 3.281 seconds (JVM running for 3.601)
/<code>

4.3.6 测试

打开浏览器,输入地址:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

效果:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

是不是很Easy?

4.4 Spring Boot的核心

4.4.41 入口类和@SpringBootApplication

Spring Boot的项目一般都会有*Application的入口类,入口类中会有main方法,这是一个标准的Java应用程序的入口方法。

@SpringBootApplication注解是Spring Boot的核心注解,它其实是一个组合注解:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

该注解主要组合了以下注解:

①. @SpringBootConfiguration:这是Spring Boot项目的配置注解,这也是一个组合注解:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

在Spring Boot项目中推荐使用@ SpringBootConfiguration替代@Configuration

②. @EnableAutoConfiguration:启用自动配置,该注解会使Spring Boot根据项目中依赖的jar包自动配置项目的配置项:

a) 如:我们添加了spring-boot-starter-web的依赖,项目中也就会引入SpringMVC的依赖,Spring Boot就会自动配置tomcat和SpringMVC

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

③. @ComponentScan:默认扫描@SpringBootApplication所在类的同级目录以及它的子目录。

4.4.2 关闭自动配置

通过上述,我们得知,Spring Boot会根据项目中的jar包依赖,自动做出配置,Spring Boot支持的自动配置如下(非常多):

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

如果我们不需要Spring Boot自动配置,想关闭某一项的自动配置,该如何设置呢?

比如:我们不想自动配置Redis,想手动配置。

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

当然了,其他的配置就类似了。

4.4.3 自定义Banner

启动Spring Boot项目后会看到这样的图案:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

这个图片其实是可以自定义的:

①. 打开网站:http://patorjk.com/software/taag/#p=display&h=3&v=3&f=4Max&t=itcast Spring Boot

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 拷贝生成的字符到一个文本文件中,并且将该文件命名为banner.txt

③. 将banner.txt拷贝到项目的resources目录中:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

④. 重新启动程序,查看效果:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

好像没有默认的好看啊!!!

如果不想看到任何的banner,也是可以将其关闭的:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

4.4.4 全局配置文件

Spring Boot项目使用一个全局的配置文件application.properties或者是application.yml,在resources目录下或者类路径下的/config下,一般我们放到resources下。

①. 修改tomcat的端口为8088

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

重新启动应用,查看效果:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 修改进入DispatcherServlet的规则为:*.html

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

测试:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

4.4.5 Starter pom

Spring Boot为我们提供了简化企业级开发绝大多数场景的sarter pom,只要使用了应用场景所需要的sater pom,相关的技术配置将会消除,就可以得到Sprig Boot为我们提供的自动配置的Bean。

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

4.4.6 Xml 配置文件

Spring Boot 提倡零配置,即无xml配置,但是在实际项目中,可能有一些特殊要求你必须使用xml配置,这时我们可以通过Spring提供的@ImportResource来加载xml配置,例如:

<code>CImportResource({
\t"clas spath: some-context. xml" ,"classpath: another-context. xml"
}
)/<code>

4.4.7 日志

Spring Boot对各种日志框架都做了支持,我们可以通过配置来修改默认的日志的配置:

<code>#设置日志级别
logging.level.org.springframework=DEBUG/<code>

格式:

<code>logging.level.*= # Log levels severity mapping. For instance `logging.level.org.springframework=DEBUG`/<code>

4.5 Spring Boot的自动配置的原理

Spring Boot在进行SpringApplication对象实例化时会加载META-INF/spring.factories文件,将该配置文件中的配置载入到Spring容器。

4.5.1 Maven下载源码

通过 dependency:sources 该命令可以下载该项目中所有的依赖的包的源码。

4.5.2 源码分析

org.springframework.boot.SpringApplication:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

org.springframework.core.io.support.SpringFactoriesLoader:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

由此可见,读取该配置文件来加载内容。

4.5.3 举例:Redis的自动配置

从上述的配置中可以看出,org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration是Redis的自动配置。

内容:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

4.5.4 条件注解

  • @ConditionalOnBean:当容器里有指定的Bean的条件下
  • @ConditionalOnClass:当类路径下有指定的类的条件下。
  • @ConditionalOnExpression: 基于SpEL表达式作为判断条件。
  • @ConditionalOnJava:基于JVM版本作为判断条件。
  • @ConditionalOnJndi:在JNDI存在的条件下查找指定的位置。
  • @ConditionalOnMissingBean:当容器里没有指定Bean的情况下。
  • @ConditionalOnMissingClass:当类路径下没有指定的类的条件下。
  • @ConditionalOnNotWebApplication:当前项目不是Web项目的条件下。
  • @ConditionalOnProperty:指定的属性是否有指定的值。
  • @ConditionalOnResource:类路径是否有指定的值。
  • @ConditionalOnSingleCandidate:当指定Bean 在容器中只有一个,或者虽然有多个但是指定首选的Bean。
  • @ConditionalOn WebA pplication:当前项目是Web项目的条件下

五、Spring Boot的web开发

Web开发的自动配置类:org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration

5.1 自动配置的ViewResolver

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

视图的配置mvcProperties对象中:

org.springframework.boot.autoconfigure.web.WebMvcProperties.View

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

5.2 自动配置静态资源

5.2.1 进入规则为 /

如果进入SpringMVC的规则为/时,Spring Boot的默认静态资源的路径为: spring.resources.static-locations=classpath:/META-INF/resources/,classpath:/resources/,classpath:/static/,classpath:/public/

测试:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

5.2.2. 进入规则为*.xxx 或者 不指定静态文件路径时 将静态资源放置到webapp下的static目录中即可通过地址访问:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了


测试:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

5.3 自定义消息转化器

自定义消息转化器,只需要在@Configuration的类中添加消息转化器的@bean加入到Spring容器,就会被Spring Boot自动加入到容器中。

<code>@Bean 

public StringHttpMessageConverter stringHttpMessageConverter(){
\tStringHttpMessageConverter converter = new StringHttpMessageConverter(Charset.forName("UTF-8"));
\treturn converter;
}/<code>

默认配置:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

5.4 自定义SpringMVC的配置

有些时候我们需要自已配置SpringMVC而不是采用默认,比如说增加一个拦截器,这个时候就得通过继承WebMvcConfigurerAdapter然后重写父类中的方法进行扩展。

<code>import java.nio.charset.Charset;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.StringHttpMessageConverter;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
@Configuration //申明这是一个配置
public class MySrpingMVCConfig extends WebMvcConfigurerAdapter{
\t// 自定义拦截器
\t@Override
\t public void addInterceptors(InterceptorRegistry registry) {
\t\tHandlerInterceptor handlerInterceptor = new HandlerInterceptor() {
\t\t\t@Override
\t\t\t public Boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
\t\t\t throws Exception {
\t\t\t\tSystem.out.println("自定义拦截器............");
\t\t\t\treturn true;
\t\t\t}
\t\t\t@Override
\t\t\t public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
\t\t\t ModelAndView modelAndView) throws Exception {
\t\t\t}
\t\t\t@Override
\t\t\t public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,
\t\t\t Exception ex) throws Exception {
\t\t\t}
\t\t}
\t\t;
\t\tregistry.addInterceptor(handlerInterceptor).addPathPatterns("/**");

\t}
\t// 自定义消息转化器的第二种方法
\t@Override
\t public void configureMessageConverters(List<httpmessageconverter>> converters) {
\t\tStringHttpMessageConverter converter = new StringHttpMessageConverter(Charset.forName("UTF-8"));
\t\tconverters.add(converter);
\t}
}/<httpmessageconverter>/<code>

六、改造购物车系统

6.1 创建购物车的Spring Boot工程

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.2 导入依赖

<code><project>\txsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
\t<modelversion>4.0.0/<modelversion>
\t<parent>
\t\t<groupid>org.springframework.boot/<groupid>
\t\t<artifactid>spring-boot-starter-parent/<artifactid>
\t\t<version>1.5.2.RELEASE/<version>
\t/<parent>
\t<groupid>com.taotao.cart/<groupid>
\t<artifactid>taotao-cart-springboot/<artifactid>
\t<version>1.0.0-SNAPSHOT/<version>
\t<packaging>war/<packaging>
\t<dependencies>
\t\t<dependency>
\t\t\t<groupid>com.taotao.common/<groupid>
\t\t\t<artifactid>taotao-common/<artifactid>
\t\t\t<version>1.0.0-SNAPSHOT/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>com.taotao.sso/<groupid>
\t\t\t<artifactid>taotao-sso-interface/<artifactid>
\t\t\t<version>1.0.0-SNAPSHOT/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>junit/<groupid>
\t\t\t<artifactid>junit/<artifactid>
\t\t\t<scope>test/<scope>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.springframework/<groupid>
\t\t\t<artifactid>spring-jdbc/<artifactid>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.springframework/<groupid>
\t\t\t<artifactid>spring-aspects/<artifactid>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t<artifactid>spring-boot-starter-web/<artifactid>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>org.mybatis/<groupid>
\t\t\t<artifactid>mybatis/<artifactid>
\t\t\t<version>3.2.8/<version>

\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.mybatis/<groupid>
\t\t\t<artifactid>mybatis-spring/<artifactid>
\t\t\t<version>1.2.2/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>com.github.pagehelper/<groupid>
\t\t\t<artifactid>pagehelper/<artifactid>
\t\t\t<version>3.7.5/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>com.github.jsqlparser/<groupid>
\t\t\t<artifactid>jsqlparser/<artifactid>
\t\t\t<version>0.9.1/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>com.github.abel533/<groupid>
\t\t\t<artifactid>mapper/<artifactid>
\t\t\t<version>2.3.4/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>mysql/<groupid>
\t\t\t<artifactid>mysql-connector-java/<artifactid>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.slf4j/<groupid>
\t\t\t<artifactid>slf4j-log4j12/<artifactid>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>com.jolbox/<groupid>
\t\t\t<artifactid>bonecp-spring/<artifactid>
\t\t\t<version>0.8.0.RELEASE/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>org.apache.httpcomponents/<groupid>
\t\t\t<artifactid>httpclient/<artifactid>
\t\t/<dependency>
\t\t

\t\t<dependency>
\t\t\t<groupid>jstl/<groupid>
\t\t\t<artifactid>jstl/<artifactid>
\t\t\t<version>1.2/<version>
\t\t/<dependency>
\t\t
\t\t<dependency>
\t\t\t<groupid>org.apache.commons/<groupid>
\t\t\t<artifactid>commons-lang3/<artifactid>
\t\t\t<version>3.3.2/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.apache.commons/<groupid>
\t\t\t<artifactid>commons-io/<artifactid>
\t\t\t<version>1.3.2/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>commons-codec/<groupid>
\t\t\t<artifactid>commons-codec/<artifactid>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.springframework.amqp/<groupid>
\t\t\t<artifactid>spring-rabbit/<artifactid>
\t\t\t<version>1.4.0.RELEASE/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>com.alibaba/<groupid>
\t\t\t<artifactid>dubbo/<artifactid>
\t\t\t<version>2.5.3/<version>
\t\t\t<exclusions>
\t\t\t\t<exclusion>
\t\t\t\t\t
\t\t\t\t\t<artifactid>spring/<artifactid>
\t\t\t\t\t<groupid>org.springframework/<groupid>
\t\t\t\t/<exclusion>
\t\t\t/<exclusions>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>org.apache.zookeeper/<groupid>
\t\t\t<artifactid>zookeeper/<artifactid>
\t\t\t<version>3.3.3/<version>
\t\t/<dependency>
\t\t<dependency>
\t\t\t<groupid>com.github.sgroschupf/<groupid>
\t\t\t<artifactid>zkclient/<artifactid>
\t\t\t<version>0.1/<version>
\t\t/<dependency>

\t/<dependencies>
\t<build>
\t\t<plugins>
\t\t\t
\t\t\t<plugin>
\t\t\t\t<groupid>org.apache.maven.plugins/<groupid>
\t\t\t\t<artifactid>maven-resources-plugin/<artifactid>
\t\t\t\t<configuration>
\t\t\t\t\t<encoding>UTF-8/<encoding>
\t\t\t\t/<configuration>
\t\t\t/<plugin>
\t\t\t
\t\t\t<plugin>
\t\t\t\t<groupid>org.apache.maven.plugins/<groupid>
\t\t\t\t<artifactid>maven-compiler-plugin/<artifactid>
\t\t\t\t<configuration>
\t\t\t\t\t<source>1.7/<source>
\t\t\t\t\t<target>1.7/<target>
\t\t\t\t\t<encoding>UTF-8/<encoding>
\t\t\t\t/<configuration>
\t\t\t/<plugin>
\t\t\t<plugin>
\t\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t\t<artifactid>spring-boot-maven-plugin/<artifactid>
\t\t\t/<plugin>
\t\t/<plugins>
\t/<build>
/<project>/<code>

6.3 将taotao-cart中的java代码拷贝到taotao-car-springboot

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

拷贝完成后:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

并且将properties文件也拷贝过来:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

将页面也拷贝过来:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.3.1 编写Spring配置类TaotaoApplication

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.3.2 设置tomcat端口

application.properties:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.3.3 读取外部的配置文件

<code>@Configuration
@PropertySource(value = { "classpath:jdbc.properties", "classpath:env.properties",
"classpath:httpclient.properties", "classpath:redis.properties", "classpath:rabbitmq.properties" }, ignoreResourceNotFound = true)
public class TaotaoApplication {
}
/<code>

6.3.4 设置扫描包

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.3.5 定义数据源

<code>@Value("${jdbc.url}")
private String jdbcUrl;
@Value("${jdbc.driverClassName}")
private String jdbcDriverClassName;
@Value("${jdbc.username}")
private String jdbcUsername;
@Value("${jdbc.password}")
private String jdbcPassword;
@Bean(destroyMethod = "close")

public DataSource dataSource() {
\tBoneCPDataSource boneCPDataSource = new BoneCPDataSource();
\t// 数据库驱动
\tboneCPDataSource.setDriverClass(jdbcDriverClassName);
\t// 相应驱动的jdbcUrl
\tboneCPDataSource.setJdbcUrl(jdbcUrl);
\t// 数据库的用户名
\tboneCPDataSource.setUsername(jdbcUsername);
\t// 数据库的密码
\tboneCPDataSource.setPassword(jdbcUsername);
\t// 检查数据库连接池中空闲连接的间隔时间,单位是分,默认值:240,如果要取消则设置为0
\tboneCPDataSource.setIdleConnectionTestPeriodInMinutes(60);
\t// 连接池中未使用的链接最大存活时间,单位是分,默认值:60,如果要永远存活设置为0
\tboneCPDataSource.setIdleMaxAgeInMinutes(30);
\t// 每个分区最大的连接数
\tboneCPDataSource.setMaxConnectionsPerPartition(100);
\t// 每个分区最小的连接数
\tboneCPDataSource.setMinConnectionsPerPartition(5);
\treturn boneCPDataSource;
}/<code>

6.3.6 设置Mybatis和Spring Boot整合

Mybatis和Spring Boot的整合有两种方式:

  • 第一种:使用mybatis官方提供的Spring Boot整合包实现,地址:https://github.com/mybatis/spring-boot-starter
  • 第二种:使用mybatis-spring整合的方式,也就是我们传统的方式

这里我们推荐使用第二种,因为这样我们可以很方便的控制Mybatis的各种配置。

首先,创建一个Mybatis的配置类:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

代码:

<code>import javax.sql.DataSource;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternResolver;
@Configuration

public class MyBatisConfig {
\t@Bean
\t @ConditionalOnMissingBean //当容器里没有指定的Bean的情况下创建该对象
\tpublic SqlSessionFactoryBean sqlSessionFactory(DataSource dataSource) {
\t\tSqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
\t\t// 设置数据源
\t\tsqlSessionFactoryBean.setDataSource(dataSource);
\t\t// 设置mybatis的主配置文件
\t\tResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
\t\tResource mybatisConfigXml = resolver.getResource("classpath:mybatis/mybatis-config.xml");
\t\tsqlSessionFactoryBean.setConfigLocation(mybatisConfigXml);
\t\t// 设置别名包
\t\tsqlSessionFactoryBean.setTypeAliasesPackage("com.taotao.cart.pojo");
\t\treturn sqlSessionFactoryBean;
\t}
}/<code>

然后,创建Mapper接口的扫描类MapperScannerConfig:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

代码:

<code>import org.mybatis.spring.mapper.MapperScannerConfigurer;
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
@AutoConfigureAfter(MyBatisConfig.class) //保证在MyBatisConfig实例化之后再实例化该类
public class MapperScannerConfig {
\t// mapper接口的扫描器
\t@Bean
\t public MapperScannerConfigurer mapperScannerConfigurer() {
\t\tMapperScannerConfigurer mapperScannerConfigurer = new MapperScannerConfigurer();
\t\tmapperScannerConfigurer.setBasePackage("com.taotao.cart.mapper");
\t\treturn mapperScannerConfigurer;
\t}
}/<code>

6.3.7 设置事务管理

在Spring Boot中推荐使用@Transactional注解来申明事务。

首先需要导入依赖:

<code><dependency>
\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t<artifactid>spring-boot-starter-jdbc/<artifactid>
\t\t/<dependency>/<code>

当引入jdbc依赖之后,Spring Boot会自动默认分别注入DataSourceTransactionManager或JpaTransactionManager,所以我们不需要任何额外配置就可以用@Transactional注解进行事务的使用。

在Service中添加@Transactional注解:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

@Transactional不仅可以注解在方法上,也可以注解在类上。当注解在类上的时候意味着此类的所有public方法都是开启事务的。如果类级别和方法级别同时使用了@Transactional注解,则使用在类级别的注解会重载方法级别的注解。

6.3.8 设置Redis和Spring的整合

在Spring Boot中提供了RedisTempplate的操作,我们暂时不做学习,先按照我们之前的实现来完成。

代码:

<code>import java.util.ArrayList;
import java.util.List;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.JedisShardInfo;
import redis.clients.jedis.ShardedJedisPool;
@Configuration
@PropertySource(value = "classpath:redis.properties")
public class RedisSpringConfig {
\t@Value("${redis.maxTotal}")
\t private Integer redisMaxTotal;
\t@Value("${redis.node1.host}")
\t private String redisNode1Host;
\t@Value("${redis.node1.port}")
\t private Integer redisNode1Port;
\tprivate JedisPoolConfig jedisPoolConfig() {
\t\tJedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
\t\tjedisPoolConfig.setMaxTotal(redisMaxTotal);
\t\treturn jedisPoolConfig;
\t}
\t@Bean
\t public ShardedJedisPool shardedJedisPool() {
\t\tList<jedisshardinfo> jedisShardInfos = new ArrayList<jedisshardinfo>();

\t\tjedisShardInfos.add(new JedisShardInfo(redisNode1Host, redisNode1Port));
\t\treturn new ShardedJedisPool(jedisPoolConfig(), jedisShardInfos);
\t}
}/<jedisshardinfo>/<jedisshardinfo>/<code>

6.3.9 设置Httpclient和Spring的整合

<code>import org.apache.http.client.config.RequestConfig;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import org.springframework.context.annotation.Scope;
import com.taotao.common.httpclient.IdleConnectionEvictor;
@Configuration
@PropertySource(value = "classpath:httpclient.properties")
public class HttpclientSpringConfig {
\t@Value("${http.maxTotal}")
\t private Integer httpMaxTotal;
\t@Value("${http.defaultMaxPerRoute}")
\t private Integer httpDefaultMaxPerRoute;
\t@Value("${http.connectTimeout}")
\t private Integer httpConnectTimeout;
\t@Value("${http.connectionRequestTimeout}")
\t private Integer httpConnectionRequestTimeout;
\t@Value("${http.socketTimeout}")
\t private Integer httpSocketTimeout;
\t@Value("${http.staleConnectionCheckEnabled}")
\t private Boolean httpStaleConnectionCheckEnabled;
\t@Autowired
\t private PoolingHttpClientConnectionManager manager;
\t@Bean
\t public PoolingHttpClientConnectionManager poolingHttpClientConnectionManager() {
\t\tPoolingHttpClientConnectionManager poolingHttpClientConnectionManager = new PoolingHttpClientConnectionManager();
\t\t// 最大连接数
\t\tpoolingHttpClientConnectionManager.setMaxTotal(httpMaxTotal);
\t\t// 每个主机的最大并发数
\t\tpoolingHttpClientConnectionManager.setDefaultMaxPerRoute(httpDefaultMaxPerRoute);
\t\treturn poolingHttpClientConnectionManager;
\t}
\t// 定期关闭无效连接
\t@Bean

\t public IdleConnectionEvictor idleConnectionEvictor() {
\t\treturn new IdleConnectionEvictor(manager);
\t}
\t// 定义Httpclient对
\t@Bean
\t @Scope("prototype")
\t public CloseableHttpClient closeableHttpClient() {
\t\treturn HttpClients.custom().setConnectionManager(this.manager).build();
\t}
\t// 请求配置
\t@Bean
\t public RequestConfig requestConfig() {
\t\treturn RequestConfig.custom().setConnectTimeout(httpConnectTimeout) // 创建连接的最长时间
\t\t.setConnectionRequestTimeout(httpConnectionRequestTimeout) // 从连接池中获取到连接的最长时间
\t\t.setSocketTimeout(httpSocketTimeout) // 数据传输的最长时间
\t\t.setStaleConnectionCheckEnabled(httpStaleConnectionCheckEnabled) // 提交请求前测试连接是否可用
\t\t.build();
\t}
}/<code>

6.3.10 设置RabbitMQ和Spring的整合

我们之前使用的Spring-Rabbit的xml方式,现在我们要改造成java方式,并且Spring Boot对RabbitMQ的使用做了自动配置,更加的简化了我们的使用。

①. 在导入spring-boot-starter-amqp的依赖;

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 在application.properties文件中配置RabbitMQ的连接信息

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

③. 编写Rabbit的Spring配置类

<code>import org.springframework.amqp.core.Queue;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import org.springframework.amqp.rabbit.core.RabbitAdmin;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class RabbitMQSpringConfig {
\t@Autowired
\t private ConnectionFactory connectionFactory;
\t// 管理
\t@Bean

\t public RabbitAdmin rabbitAdmin() {
\t\treturn new RabbitAdmin(connectionFactory);
\t}
\t// 声明队列
\t@Bean
\t public Queue taotaoCartLoginQueue() {
\t\t// 默认就是自动声明的
\t\treturn new Queue("TAOTAO-CART-LOGIN-QUEUE", true);
\t}
\t// 声明队列
\t@Bean
\t public Queue taotaoCartOrderSuccessQueue() {
\t\t// 默认就是自动声明的
\t\treturn new Queue("TAOTAO-CART-ORDER-SUCCESS-QUEUE", true);
\t}
}/<code>

④. 设置监听

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.3.11 设置SpringMVC的配置

原有配置:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

具体实现: 视图解析器配置:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

自定义拦截器:

<code>import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
import com.taotao.cart.interceptors.UserLoginHandlerInterceptor;
@Configuration
public class SpringMVCConfig extends WebMvcConfigurerAdapter {
\t@Override
\t public void addInterceptors(InterceptorRegistry registry) {
\t\t// 判断用户是否登录的拦截器
\t\tregistry.addInterceptor(new UserLoginHandlerInterceptor()).addPathPatterns("/cart/**");

\t}
}/<code>

6.3.12 设置dubbo的配置

Dubbo目前只能使用xml配置的方式,所以我们需要保留xml,并且需要将该xml加入到现有的Spring容器中才能生效。

①. 将dubbo目录以及下面的xml配置文件拷贝到taotao-cat-springboot中

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 将dubbo的xml文件加入到spring容器

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.4 编写入口类

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

编写main方法:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.4.1 启动错误1

关键错误(丢失了web容器的工厂,也就是说我们并没有把它作为一个web应用来启动):

<code>org.springframework.context.ApplicationContextException: Unable to start embedded container; nested exception is org.springframework.context.ApplicationContextException: Unable to start EmbeddedWebApplicationContext due to missing EmbeddedServletContainerFactory bean.
/<code>

解决:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

让Spring Boot来自动选择并且完成web的相关加载工作。

6.4.2 Slf4j日志警告

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

提示我们当前的项目中slf4j引入了2个,导致了jar冲突。

解决:

①. 删除自己引入到slf4j的依赖

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

②. 将taotao-common中传递的依赖排除掉

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

再次启动,发现警告没了:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.4.3 解决jsp访问404的问题

由于Spring boot使用的内嵌的tomcat,而内嵌的tamcat是不支持jsp页面的,所有需要导入额外的包才能解决。

<code><dependency>
\t\t\t<groupid>org.apache.tomcat.embed/<groupid>
\t\t\t<artifactid>tomcat-embed-jasper/<artifactid>
\t\t\t<scope>provided/<scope>
\t\t/<dependency>
/<code>

重新启动进行测试:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.4.4 拦截器中的UserService空指针异常

分析:由于添加拦截器时,直接对UserLoginHandlerInterceptor进行new操作,导致UserService无法注入,所以有空指针异常。

解决:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

6.4.5 路径问题

现在我们进入Servlet的路径为”/”,访问*.html页面没问题,但是,访问 /service/* 就会有问题,所以需要改一下js,将原有的/service/ 改为 /

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

测试,功能一切ok。

七、发布到独立的tomcat中运行

在开发阶段我们推荐使用内嵌的tomcat进行开发,因为这样会方便很多,但是到生成环境,我希望在独立的tomcat容器中运行,因为我们需要对tomcat做额外的优化,这时我们需要将工程打包成war包发进行发布。

7.1 工程的打包方式为war


现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

7.2 将spring-boot-starter-tomcat的范围设置为provided

设置为provided是在打包时会将该包排除,因为要放到独立的tomcat中运行,是不需要的。

<code><dependency> 

\t\t\t<groupid>org.springframework.boot/<groupid>
\t\t\t<artifactid>spring-boot-starter-tomcat/<artifactid>
\t\t\t<scope>provided/<scope>
\t\t/<dependency>
/<code>

7.3 修改代码,设置启动配置

需要集成SpringBootServletInitializer,然后重写configure,将Spring Boot的入口类设置进去。

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

7.4 打war包

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

打包成功:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

7.5 部署到tomcat

解压apache-tomcat-7.0.57.tar.gz,将war包解压到webapps下的ROOT目录中,启动:

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

现在搞Java还不会SpringBoot?看完这篇两万字笔记精髓就够了

写在最后:

  • 针对于Java程序员,笔者最近整理了一些面试真题,思维导图,程序人生等PDF学习资料;
  • 关注私信我"86",即可获取!
  • 希望读到这的您能点个小赞和关注下我,以后还会更新技术干货,谢谢您的支持!


分享到:


相關文章: