springboot学习(一)——helloworld
以下内容,如有问题,烦请指出,谢谢
springboot出来也很久了,以前零散地学习了不少,不过很长时间了都没有在实际中使用过了,忘了不少,因此要最近准备抽时间系统的学习积累下springboot,给自己留个根。
因为以前学过一些,这里就主要根据官方文档来学习了,可能会根据自己的理解来选择一些知识点的学习顺序。官方文档地址:https://docs.spring.io/spring-boot/docs/1.5.8.RELEASE/reference/htmlsingle/
官方文档有十个大章节,第一章是综述,就不用太细看了,这里直接从第二章开始,也是介绍hello world的地方。
一开始就介绍了spring的四个目标:
- 更快地开发spring程序
- 提供默认配置,开箱即用
- 提供大量插件(主要是各种starter)
- 避免代码生成以及xml配置
上面这四个也是普通spring程序的弊病,简单来说就是入门门槛高了,spring的各种配置是很重要的,但是项目搭建后就很少管,然后新启动一个项目又要花很长时间去配置,配置得还是跟原来的差不多。
系统要求就是jdk1.7/servlet3.0及以上,1.6版本如何使用可以看很文档后面的说明。官方内嵌支持三种servlet容器,最常见的tomcat,然后是jetty以及undertow(jboss默认的),tomcat是默认的,如何servlet容器替换后面会有内容讲解。
第10小章是讲解如何生成springboot的基础项目,这个是给缺少基础的初学者看的,可以不用太关注,如果想生成,可以直接用https://start.spring.io/ 这个。
11小章开始正篇。
基础的helloworld很简单,先配置maven文件,强烈建议使用springboot提供的parent,不要自己一个个去单独依赖,这个parent本身写的比较好,可以作为很多项目的参考。如果你有一定的基础,可以不照着官方的started文档的代码来写,本身这段代码也比较少比较简单。这里我构造的项目结果如下,因为后续就主要用这个项目来学习springboot,因此就取名main,包名也尽量做到规范
具体代码如下
pom.xml
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>pr.study.springboot</groupId>
<artifactId>main</artifactId>
<version>1.0.0</version>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.8.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
Main,java
package pr.study.springboot;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class Main {
public static void main(String[] args) {
SpringApplication.run(Main.class, args);
}
}
HelloWorldController.class
package pr.study.springboot.controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class HelloWorldController {
@RequestMapping("/hello")
public String hello() {
return "hello world";
}
}
几点说明:
1、因为springboot-parent中的dependencies是使用dependencyManagement的形式,这种形式的依赖不会强制继承,必须在子pom中显式声明才会继承,但是不用声明版本,声明了版本会提示一个warning。
2、继承spring-boot-maven-plugin是为了利用springboot的fat-jar打包功能,和dependency一样,也是使用Management的形式,需要显式声明才会依赖。
3、Main.class我们编写的代码的启动入口类,使用spring-boot-maven-plugin打包成jar并用java -jar启动时,这个类并不是真正应用程序入口类Main-Class。
4、官方的代码中controller和main函数写在一个类中,因此可以不需要SpringBootApplication这个注解,写在不同的类中就需要这个注解,否则扫描不到controller。通过源码可以知道@SpringBootApplication = @SpringBootConfiguration + @EnableAutoConfiguration + @ComponentScan,它具有ComponentScan的作用,因此不需要像spring程序那样显式指定包扫描.
5、@EnableAutoConfiguration,这个注解后面代表的一系列功能是springboot的核心之一,自动配置,这个注解解决了很多配置问题。他的基本原理就是通过你的maven依赖来猜出你需要注入那些bean,它会帮你启动相关配置并注入这个bean。这里我们依赖了spring-boot-starter-web,它就会帮我们开启springmvc相关的配置并自动注入相关的必须的bean。因为会自动配置并注入,所以springboot程序的依赖管理很重要没用的依赖尽量都去掉,特别是那些数据库相关的依赖。关于springboot的自动配置,这个是重点之一,后面细说。
6、@RestController = @Controller + @@ResponseBody,也是一个复合注解,减少代码量,类似的还有@GetMapping、@PostMapping等等。
上面这六点,1-3可以通过maven相关的知识更进一步了解,具体就是maven依赖管理以及maven打包,本人没有系统的学习,知识简单了解,所以这里就不细说了。4-6在后续的springboot学习中会慢慢学习到。
运行程序可以直接在Main上运行java application作为普通java程序运行,这时候Main中的main函数就是真正的启动函数。这里是使用嵌入式tomcat作为servlet容器来运行servlet程序,所以并不需要我们部署到tomcat。
红色标记的就是已经扫描到了并初始化成功了HelloWorldController。
打开浏览器请求 http://localhost:8080/hello 可以看到运行结果
如果运行出现这种问题
The Bean Validation API is on the classpath but no implementation could be found
那么请删除下springboot相关的本地jar包,有时候jar包下载会抽风。本身springmvc里面是依赖有hibernate-validator作为参数校验的(@Valid @Validated使用),是不应该出现这个问题的。
如果你想打包部署,因为引用了spring-boot-maven-plugin,所以直接运行 mvn clean package就可以打包,最后打包的target是一个fat-jar,也就是包含了所有依赖的可运行jar包。
打包好后,在路径下命令行输入 java -jar main-1.0.0.jar 就可以启动
你可以把 java -jar main-1.0.0.jar 写到脚本文件sh or bat中,这样就可以作为一个简单的启动脚本,然后和jar一起发送到目标机器上,运行启动脚本启动程序,这样就差不多就算是完成了一次简单的springboot程序的部署发布。
把这个打包好的jar解压下,简单看下结构
BOOT-INF是应用程序的主体,classes下面的就是我们的写的代码,lib里面的就是是我们的程序依赖的jar包
META-INF是一些基础数据,其中比较重要的就是MANIFEST.MF
Manifest-Version: 1.0
Implementation-Title: main
Implementation-Version: 1.0.0
Archiver-Version: Plexus Archiver
Built-By: pengrui
Implementation-Vendor-Id: pr.study.springboot
Spring-Boot-Version: 1.5.8.RELEASE
Implementation-Vendor: Pivotal Software, Inc.
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: pr.study.springboot.Main
Spring-Boot-Classes: BOOT-INF/classes/
Spring-Boot-Lib: BOOT-INF/lib/
Created-By: Apache Maven 3.3.9
Build-Jdk: 1.8.0_111
Implementation-URL: http://projects.spring.io/spring-boot/main/
这个文件描述了jar的基本结构
- Main-Class:通过java -jar 启动时应用程序启动的真正主类,java程序可以有多个main函数,但是这个类中的main函数的第一个启动的函数
- Start-Class:我们的springboot程序启动的类,直接运行main方法时,这个类中的main函数最先运行,通过 java -jar 运行时,这个类中main并不是最先运行的,只是从这个类开始,我们的写的那部分代码才开始运行
- Spring-Boot-Classes:我们写的代码存放的路径
- Spring-Boot-Lib:我们的依赖包存放的路径
关于JarLauncher以及org.springframework.boot.loader.*里面的内容,我还没有研究过,这个后续再细说。
到这里,一个简单的springboot程序的编码、本地测试运行、打包、启动就都简单过了一遍,后面再一个个补充丰满。
代码相关:
https://gitee.com/page12/study-springboot
https://github.com/page12/study-springboot