10 分钟手把手教你快速入门 SpringBoot!,字节跳动 java 研发面试题社招
@SpringBootApplication
public class Springroad01Application {
public static void main(String[] args) {
SpringApplication.run(Springroad01Application.class, args);
}
}
// 测试数据库访问,容器启动后会执行 run 方法
@Component
class DataWriter implements ApplicationRunner {
private Logger log = LoggerFactory.getLogger(DataWriter.class);
private DatabaseClient client;
private UsersDao usersDao;
public DataWriter(DatabaseClient client, UsersDao usersDao) {
this.client = client;
this.usersDao = usersDao;
}
@Override
public void run(ApplicationArguments args) {
List<String> statements = Arrays.asList(
"DROP TABLE IF EXISTS USERS;",
"CREATE TABLE IF NOT EXISTS USERS ( id SERIAL PRIMARY KEY, name VARCHAR(100) NOT NULL);");
statements.forEach(sql -> client.sql(sql).fetch().rowsUpdated()
.doOnSuccess(count -> log.info("Schema created, rows updated: {}", count))
.doOnError(error -> log.error("got error : {}",error.getMessage(),error))
.subscribe()
);
Flux.just("sevenluo","tonyzhu","jameschen").flatMap(name -> usersDao.save(new Users(null,name))).subscribe(user -> log.info("User saved: {}",user));
}
}
// DAO 接口,不需要加注解,继承了 ReactiveCrudRepository 会自动生成实例的
interface UsersDao extends ReactiveCrudRepository<Users, String> {
}
@Data
@AllArgsConstructor
@NoArgsConstructor
class Users {
@Id
private Integer id;
private String name;
}
执行 main 方法,输入结果;
就是这么赤鸡,结束了,啥配置没有,直接写了几行代码就可以访问数据库了。
REST 接口开发
=========
你应该还关心 Spring Boot 如何开发一个 REST 风格的 WEB 接口吧?别慌,我们直接用 Spring 支持的响应式编程来搞一个 REST 应用服务。但是对于 WEB 访问用户来说是看不出来我们使用了非阻塞的响应式编程的,来一手润物细无声。
将下面的代码放到启动类中,就是一个响应式的 REST 接口就开发好了。
测试一下,打开命令行,curl 测试一下(不是非要装逼不用浏览器访问,公司的大佬说程序员要多用 shell 环境,嗯!从小事开始练);
温馨提示,在 mac 上安装 jq,直接使用 brew install jq 就会自动帮你安装好了。
好了,打完手工,没啥搞头,Spring Boot 永远滴神!
监控和管理应用
=======
我们的应用上线后,那肯定都是要上监控的,不然哪一天神不知鬼不觉死翘翘了,麻烦就大了。这个 Spring Boot 它就天生带了可以替我们监控和管理的 Spring 应用的模块工具:spring-boot-actuator 。
像什么 健康检查、审计、统计和 HTTP 追踪等该有的它都有。值得一提的是 spring-boot-a
ctuator 它还很 open ,支持与其它外部监控系统做整合。补充了它自身没有一些好看的仪表盘、图表、分析、告警等 酷炫吊炸天的能力。
那怎么玩呢?
确认你的应用添加了对应的模块依赖;
=================
假如你使用的 maven,那看你的 pom.xml 中下面的依赖:
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
那如果你使用的 gradle,对应的 build.gradle 文件中有下面的依赖:
dependencies {
compile("org.springframework.boot:spring-boot-starter-actuator")
}
Actuator 通过 endpoint 来暴露 HTTP 请求 来监控和管理应用;
==========================================
应用启动后,
http://localhost:8080/actuator 会展示出所有通过 HTTP 暴露的 endpoint。
因为太长了,我这里列出的只是一部分 endpoint,你自己一定要动手试下。
就比如,/health 这个 endpoint,提供了关于应用健康的基础信息。
/metrics endpoint 展示了几个非常有用的度量信息,比如 JVM 内存使用情况、系统 CPU 使用情况、打开的文件等等。
/loggers endpoint 展示了应用的日志和可以让你在运行时改变日志等级。
还有好多,下去都自己玩玩,动手就会了。
友情提示,如果你的
http://localhost:8080/actuator 展示的 开放端点很少,那是因为 actuator 的 endpoint 可以显式地被打开和关闭,明白了吧!
默认情况下很多 endpoint 是被关闭掉了,你只需要在 application.properties 配置文件中增加如下配置即可。
management.endpoint.health.show-details=always //显示详细的健康信息
management.endpoints.web.exposure.include=* //粗暴的全部打开,反正自己玩,任性
这里我们就不多说这些 endpoint 每个的作用了, 如果你感兴趣,可以给我留言,我会根据大家需求看下是否在出一期教程详细介绍一下。
当然除了上面默认显示的应用健康信息,我们也可以自定义一个健康指标;
我们可以选择实现 HealthIndicator 接口来实现我们的目标;
@SpringBootApplication
public class Springroad01Application {
// 加上这个 Bean 就可以了
@Bean
HealthIndicator healthIndicator() {
return () -> Health.up().withDetail("app", "i am so good").
withDetail("error","开什么玩笑,老夫怎么可能有错!").build();
}
@Bean
RouterFunction<ServerResponse> routes (UsersDao usersDao) {
return RouterFunctions.route(GET("/users"),serverRequest -> ok().body(usersDao.findAll(),Users.class));
}
public static void main(String[] args) {
SpringApplication.run(Springroad01Application.class, args);
}
}
一旦你加上我们上面自定义的健康指标,重启应用,就会看到 health 这个 endpoint 将展示出我们增加的这些信息:
安全控制
====
对于应用安全的支持,那就绕不开 Spring Security 了。
它可以轻松完成应用的鉴权、授权功能,同时也提供了对响应式编程的支持。
接下来就展示下如果让你的应用快速带上安全套。
加入 Spring-Security 依赖包;
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-security</artifactId>
</dependency>
内存中配置一个用户,设置用户名和密码;
@Bean
// 内存中配置用户名、密码为 admin/admin,用户角色为 USER
MapReactiveUserDetailsService users() {
return new MapReactiveUserDetailsService(User.withUsername("admin").password(PasswordEncoderFactories.createDelegatingPasswordEncoder().encode("admin")).roles("USER").build());
}
评论