简

人生短暂,学海无边,而大道至简。


  • 首页

  • 归档

  • 分类

  • 标签

springcloud介绍

发表于 2018-08-14 | 分类于 springcloud

使用Spring Cloud Netflix中的Eureka实现了服务注册中心以及服务注册与发现;而服务间通过Ribbon或Feign实现服务的消费以及均衡负载;
通过Spring Cloud Config实现了应用多环境的外部化配置以及版本管理。使用Hystrix的融断机制来避免在微服务架构中个别服务出现异常时引起的故障蔓延。

为了保证对外服务的安全性,需要实现对服务访问的权限控制,如果通过开放服务权限控制的方式去实现权限控制会入侵业务逻辑,破坏了服务集群无状态的特点,还无法直接复用既有接口,所以,通过服务网关的形式来解决以上问题。

服务网关是微服务架构中一个不可缺的部分。通过服务网关统一向外系统提供REST API的过程中,除了具备服务路由、均衡负载功能之外,它还具备了权限控制等功能。
Spring Cloud Netflix中的Zuul就是为微服务架构提供了前门保护的作用,同时将权限控制这些较重的非业务逻辑内容迁移到服务路由层面,使得服务集群主体能够具备更高的可复用性和可测试性。

构建服务网关

使用zuul之前,创建好服务端和消费端分别为:zuul-eureka-consumer、zuul-eureka-previder。
创建网关模块:zuul-gateway,引入依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<?xml version="1.0" encoding="UTF-8"?>
<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>com.hu</groupId>
<artifactId>zuul-gateway</artifactId>
<version>1.0-SNAPSHOT</version>
<parent>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-parent</artifactId>
<version>Dalston.SR1</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-zuul</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-eureka</artifactId>
</dependency>
</dependencies>
</project>

创建配置文件application.yml,并加入服务名、端口号、eureka注册中心的地址。

1
2
3
4
5
6
7
8
9
10
11
12
server:
port: 7903
spring:
application:
name: zuul-gateway
eureka:
client:
serviceUrl:
defaultZone: http://user:123456@localhost:8761/eureka
logging:
level:
root: INFO

启动类:

1
2
3
4
5
6
7
8
//开启Zuul的功能
@EnableZuulProxy
@SpringCloudApplication
public class ZuulApplication {
public static void main(String[] args){
SpringApplication.run(ZuulApplication.class, args);
}
}

这样,基于Spring Cloud Zuul服务网关就已经构建完毕。

由于Spring Cloud Zuul在整合了Eureka之后,具备默认的服务路由功能,当zuul-gateway应用启动并注册到eureka之后,服务网关会发现已经启动的两个服务zuul-eureka-consumer、zuul-eureka-previder,这时候Zuul就会创建两个路由规则。每个路由规则都包含:外部请求的匹配规则和路由的服务ID。

这里就会创建两个规则:
zuul-eureka-consumer服务的请求规则为:/consumer-user/**
zuul-eureka-previder服务的请求规则为:/provider-user/**

通过访问7903端口的服务网关来验证上述路由的正确性:
http://localhost:7903/provider-user/user,该请求将最终被路由到zuul-eureka-previder的/user接口上。
访问成功返回数据:江小白

传统路由配置

传统路由配置方式就是在不依赖于服务发现的情况下,通过在配置文件中具体指定每个路由表达式与服务实例的映射关系来实现API网关对外部请求的路由。

单实例配置:通过一组zuul.routes..path与zuul.routes..url参数对的方式配置

1
2
zuul.routes.user-service.path=/user-service/**
zuul.routes.user-service.url=http://localhost:8080/

实现了对符合/user-service/**规则的请求路径转发到http://localhost:8080/地址的路由规则。
当有一个请求http://localhost:7903/user-service/hello被发送到API网关上,由于/user-service/hello能够被上述配置的path规则匹配,所以API网关会转发请求到http://localhost:8080/hello地址。

多实例配置:通过一组zuul.routes..path与zuul.routes..serviceId参数对的方式配置

1
2
3
4
zuul.routes.user-service.path=/user-service/**
zuul.routes.user-service.serviceId=user-service
ribbon.eureka.enabled=false
user-service.ribbon.listOfServers=http://localhost:8080/,http://localhost:8081/

实现了对符合/user-service/**规则的请求路径转发到http://localhost:8080/和http://localhost:8081/两个实例地址的路由规则。
serviceId是由用户手工命名的服务名称,配合.ribbon.listOfServers参数实现服务与实例的维护。
由于存在多个实例,API网关在进行路由转发时需要实现负载均衡策略,还需要Spring Cloud Ribbon的配合,Spring Cloud Zuul中自带了对Ribbon的依赖,只需要做一些配置即可。

ribbon.eureka.enabled:由于zuul.routes..serviceId指定的是服务名称,默认情况下Ribbon会根据服务发现机制来获取配置服务名对应的实例清单。
但是,该示例并没有整合类似Eureka之类的服务治理框架,所以需要将该参数设置为false,不然配置的serviceId是获取不到对应实例清单的。
user-service.ribbon.listOfServers:该参数内容与zuul.routes..serviceId的配置相对应,开头的user-service对应了serviceId的值,这两个参数的配置相当于在该应用内部手工维护了服务与实例的对应关系。

不论哪种方式,都需要为每一对映射关系指定一个名称,也就是上面配置中的,每一个就对应了一条路由规则。每条路由规则都需要通过path属性来定义一个用来匹配客户端请求的路径表达式,并通过url或serviceId属性来指定请求表达式映射具体实例地址或服务名。

服务路由配置

Spring Cloud Zuul通过与Spring Cloud Eureka的整合,实现了对服务实例的自动化维护,使用服务路由配置的时候,不需要向传统路由配置方式那样为serviceId去指定具体的服务实例地址,只需要通过一组zuul.routes..path与zuul.routes..serviceId参数对的方式配置即可。

例如:

1
2
zuul.routes.user-service.path=/user-service/**
zuul.routes.user-service.serviceId=user-service

实现了对符合/user-service/规则的请求路径转发到名为user-service的服务实例上去的路由规则。
还有一种更简洁的配置方式:zuul.routes.=,其中用来指定路由的具体服务名,用来配置匹配的请求表达式。
例如:zuul.routes.user-service=/user-service/

等价于上面两条的组合。

那么当采用path与serviceId以服务路由方式实现时候,没有配置任何实例地址的情况下,外部请求经过API网关的时候,它是如何被解析并转发到服务具体实例的呢?
在Spring Cloud Netflix中,Zuul巧妙的整合了Eureka来实现面向服务的路由。实际上,可以直接将API网关也看做是Eureka服务治理下的一个普通微服务应用。它除了会将自己注册到Eureka服务注册中心上之外,也会从注册中心获取所有服务以及它们的实例清单。
所以,在Eureka的帮助下,API网关服务本身就已经维护了系统中所有serviceId与实例地址的映射关系。当有外部请求到达API网关的时候,根据请求的URL路径找到最佳匹配的path规则,API网关就可以知道要将该请求路由到哪个具体的serviceId上去。由于在API网关中已经知道serviceId对应服务实例的地址清单,那么只需要通过Ribbon的负载均衡策略,直接在这些清单中选择一个具体的实例进行转发就能完成路由工作了。

过滤器

微服务应用提供的接口就可以通过统一的API网关入口被客户端访问。但是当请求时对它们的访问权限进行限制,系统不会把所有接口全部开放。当前的服务路由并没有限制权限这样的功能,实现对客户端请求的安全校验和权限控制,最简单的方法就是为每个微服务应用都实现过滤器或拦截器。但是这样并不好,增加日后的系统维护难度。所以直接把这些校验剥离出去,构建出一个独立的服务。在完成了剥离之后,还可能会在微服务应用中通过调用校验服务来实现校验,这种方式校验在本质上并没有完全脱离微服务应用,冗余的拦截器或过滤器依然会存在。

由于网关服务的加入,外部客户端访问我们的系统已经有了统一入口,通过在网关中完成校验和过滤,微服务应用端就可以去除各种复杂的过滤器和拦截器了,这使得微服务应用的接口开发和测试复杂度也得到了相应的降低。

Spring Cloud Zuul的另外一个核心功能:过滤器。允许开发者在API网关上通过定义过滤器来实现对请求的拦截与过滤。只需要继承ZuulFilter抽象类并实现它定义的四个抽象函数就可以完成对请求的拦截和过滤了。

过滤器的实现

定义Zuul过滤器,请求被路由之前检查HttpServletRequest中是否有某个参数,若有就进行路由,若没有就拒绝访问,自定义过滤器:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
/**
* 继承ZuulFilter抽象类并重写了下面的四个方法来实现自定义的过滤器
* 定请求被路由之前检查HttpServletRequest中是否有某个参数,若有就进行路由,若没有就拒绝访问。
*/
public class ZuulFilter extends com.netflix.zuul.ZuulFilter {

private static Logger log = LoggerFactory.getLogger(ZuulFilter.class);

/**
* 过滤器的类型,它决定过滤器在请求的哪个生命周期中执行。这里定义为pre,代表会在请求被路由之前执行。
* @return
*/
@Override
public String filterType() {
return "pre";
}

/**
* 过滤器的执行顺序。当请求在一个阶段中存在多个过滤器时,需要根据该方法返回的值来依次执行。
* @return
*/
@Override
public int filterOrder() {
return 0;
}

/**
* 判断该过滤器是否需要被执行。
* 这里直接返回了true,该过滤器对所有请求都会生效。
* @return
*/
@Override
public boolean shouldFilter() {
return true;
}

/**
* 过滤器具体逻辑
* @return
*/
@Override
public Object run() {
//获取请求信息
RequestContext ctx = RequestContext.getCurrentContext();
HttpServletRequest request = ctx.getRequest();

log.info(request.getMethod(), request.getRequestURL().toString());

Object accessToken = request.getParameter("token");
if(accessToken == null) {
log.warn("不通过");
//令zuul过滤该请求,不对其进行路由
ctx.setSendZuulResponse(false);
//设置了其返回的错误码
//还可以通过ctx.setResponseBody(body)对返回body内容进行编辑等。
ctx.setResponseStatusCode(401);
return null;
}
//TODO 还有一些校验信息略过
log.info("通过");
return null;
}

}

自定义过滤器之后,不会直接生效,创建具体的Bean才能启动该过滤器,比如在主类中增加如下内容:

1
2
3
4
@Bean
public ZuulFilter zuulFilter(){
return new ZuulFilter();
}

浏览:http://localhost:7903/consumer-user/user 日志显示

1
2
 INFO 11872 --- [nio-7903-exec-2] com.hu.filter.ZuulFilter                 : GET
WARN 11872 --- [nio-7903-exec-2] com.hu.filter.ZuulFilter : 不通过

浏览:http://localhost:7903/consumer-user/user?token=11
显示出结果,并且日志中显示出通过的信息。

Zuul常见问题

有时候当我们将Spring Cloud Zuul作为API网关接入网站类应用时,会遇到会话无法保持或者重定向后的HOST错误。
1、会话无法保持
通过跟踪一个HTTP请求经过Zuul到具体服务,再到返回结果的全过程。在传递的过程中,HTTP请求头信息中的Cookie和Authorization都没有被正确地传递给具体服务,所以最终导致会话状态没有得到保持。
从Zuul进行路由转发的过滤器(如:RibbonRoutingFilter)中进行一步步查看:
从过滤器的核心逻辑run方法看,其中调用了内部方法buildCommandContext来构建上下文,而buildCommandContext中调用了helper对象的buildZuulRequestHeaders方法来处理请求头信息,helper对象是ProxyRequestHelper类的实例;
而进入到ProxyRequestHelper类中看到,构建头信息的方法buildZuulRequestHeaders通过isIncludedHeader函数来判断当前请求的各个头信息是否在忽略的头信息清单中,如果是的话就不组织到此次转发的请求中去;
如果是忽略的头信息,则也要进行初始化,在PreDecorationFilter源码中可以看到,通过调用ProxyRequestHelper的addIgnoredHeaders方法来添加需要忽略的信息到请求上下文中,供后续ROUTE阶段的过滤器使用。
if/else块分别用来处理全局设置的敏感头信息和指定路由设置的敏感头信息。而全局的敏感头信息定义于ZuulProperties中:

1
2
3
4
5
@Data
@ConfigurationProperties("zuul")
public class ZuulProperties {
private Set<String> sensitiveHeaders = new LinkedHashSet<>(Arrays.asList("Cookie", "Set-Cookie", "Authorization"));
}

所以解决该问题的方法只需要通过设置sensitiveHeaders即可:

1
2
3
4
5
全局设置:
zuul.sensitive-headers=
指定路由设置:
zuul.routes.<routeName>.sensitive-headers=
zuul.routes.<routeName>.custom-sensitive-headers=true

2、重定向问题
这是Zuul没有正确的处理HTTP请求头信息中的Host导致。在Brixton版本中,Spring Cloud Zuul的PreDecorationFilter过滤器实现时完全没有考虑这一问题,但是在Camden版本之后,Zuul增强了该功能,只需要通过配置属性zuul.add-host-header=true就能让有问题的重定向操作得到正确的处理。

Zuu统一异常处理

1、try-catch处理
异常后通过设置上下文参数信息,回传:
error.status_code:错误编码
error.exception:Exception异常对象
error.message:错误信息

1
2
3
4
5
catch (ZuulException ex) {
context.set(ERROR_STATUS_CODE, ex.nStatusCode);
context.set("error.message", ex.errorCause);
context.set("error.exception", ex);
}

2、ErrorFilter处理
使用error类型的过滤器
请求生命周期的pre、route、post三个阶段中有异常抛出的时候都会进入error阶段的处理,可以通过创建一个error类型的过滤器来捕获这些异常信息,并根据这些异常信息在请求上下文中注入需要返回给客户端的错误描述,这里我们可以直接沿用在try-catch处理异常信息时用的那些error参数,这样就可以让这些信息被SendErrorFilter捕获并组织成消息响应返回给客户端。

参考 http://blog.didispace.com/spring-cloud-learning/

Spring Cloud Config

发表于 2018-08-14 | 分类于 springcloud

对于Spring Boot应用,我们可以将配置内容写入application.yml,设置多个profile,也可以用多个application-{profile}.properties文件配置,并在启动时指定spring.profiles.active={profile}来加载不同环境下的配置。
在Spring Cloud微服务架构中,这种方式未必适用,微服务架构对配置管理有着更高的要求,如:

  • 集中管理:成百上千(可能没这么多)个微服务需要集中管理配置,否则维护困难、容易出错;
  • 运行期动态调整:某些参数需要在应用运行时动态调整(如连接池大小、熔断阈值等),并且调整时不停止服务;
  • 自动更新配置:微服务能够在配置发生变化是自动更新配置。

Spring Cloud Config是Spring Cloud团队创建的一个全新项目,用来为分布式系统中的基础设施和微服务应用提供集中化的外部配置支持,它分为服务端与客户端两个部分。

其中服务端也称为分布式配置中心,它是一个独立的微服务应用,用来连接配置仓库并为客户端提供获取配置信息、加密/解密信息等访问接口;而客户端则是微服务架构中的各个微服务应用或基础设施,它们通过指定的配置中心来管理应用资源与业务相关的配置内容,并在启动的时候从配置中心获取和加载配置信息。

Spring Cloud Config实现的配置中心默认采用Git来存储配置信息,所以使用Spring Cloud Config构建的配置服务器,天然就支持对微服务应用配置信息的版本管理,并且可以通过Git客户端工具来方便的管理和访问配置内容。当然它也提供了对其他存储方式的支持,比如:SVN仓库、本地化文件系统。

1、准备配置仓库

https://github.com/huingsn/config-repo-demo.git
假设我们读取配置中心的应用名为config-client

那么我们可以在git仓库中该项目的默认配置文件config-client.yml:

1
2
info:
profile: default
2、创建配置中心

核心依赖:

1
2
3
4
5
6
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-config-server</artifactId>
</dependency>
</dependencies>

配置信息:

1
2
3
4
5
6
7
8
9
10
11
12
13
spring:
application:
name: config-server
cloud:
config:
server:
git:
uri: https://github.com/huingsn/config-repo-demo.git
#如果我们的Git仓库需要权限访问,那么可以通过配置下面的两个属性来实现;
#spring.cloud.config.server.git.username:访问Git仓库的用户名
#spring.cloud.config.server.git.password:访问Git仓库的用户密码
server:
port: 1200

启动类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
//开启Spring Cloud Config的服务端功能。
@EnableConfigServer
@SpringBootApplication
public class Application {

public static void main(String[] args) {
new SpringApplicationBuilder(Application.class).web(true).run(args);
}
}
```
到此,一个简单的Spring Cloud Config实现,并使用Git管理配置内容的分布式配置中心就已经完成了。

启动,通过浏览器工具查看配置信息如下:
<img src="" />

Json中返回了应用名:config-client,环境名:dev,分支名:master,以及default环境和dev环境的配置内容。

访问配置信息的URL与配置文件的映射关系如下:
/{application}/{profile}[/{label}]
/{application}-{profile}.yml
/{label}/{application}-{profile}.yml
上面的url会映射{application}-{profile}.properties对应的配置文件,其中{label}对应Git上不同的分支,默认为master。
<img src="" />

##### 3、创建客户端

创建一个微服务应用,命名为config-client,引入依赖:
```xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-config</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>

创建bootstrap.yml配置,来指定获取配置文件的config-server-git位置:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
spring:
application:
name: config-client
cloud:
config:
uri: http://localhost:1200/ #指定配置中心
profile: default
label: master
# 获取配置文件规则:/{application}/{profile}[/{label}]
# spring.application.name:对应配置文件规则中的{application}部分
# spring.cloud.config.profile:对应配置文件规则中的{profile}部分
# spring.cloud.config.label:对应配置文件规则中的{label}部分
# spring.cloud.config.uri:配置中心config-server的地址
server:
port: 1201

启动类:

1
2
3
4
5
6
@SpringBootApplication
public class ApplicationClient {
public static void main(String[] args) {
SpringApplication.run(ApplicationClient.class, args);
}
}

启动后访问结果:

一个简单的demo完成。
整体流程:

4、加密解密

Spring Cloud Config提供了对属性进行加密解密的功能,以保护配置文件中的信息安全。比如下面的例子:

1
2
spring.datasource.username=didi
spring.datasource.password={cipher}dba6505baa81d7sdfafweafaf234324ji4h32u4h32uh4u32437cfbf143695f5b

在Spring Cloud Config中通过在属性值前使用{cipher}前缀来标注该内容是一个加密值,当微服务客户端来加载配置时,配置中心会自动的为带有{cipher}前缀的值进行解密。


加密准备:
使用Spring Cloud Config的加密解密功能时,为了启用该功能,需要在配置中心的运行环境中安装不限长度的JCE版本(Unlimited Strength Java Cryptography Extension)。
虽然JCE功能在JRE中就有,但是默认使用的是有长度限制的版本。我们可以从Oracle的官方网站中下载到它,它是一个压缩包,解压后可以看到下面三个文件:
README.txt
local_policy.jar
US_export_policy.jar
将local_policy.jar和US_export_policy.jar两个文件复制到$JAVA_HOME/jre/lib/security目录下,覆盖原来的默认内容。

完成了JCE的安装后,启动配置中心。在控制台中,将会输出了一些配置中心特有的端点,主要包括:

  • /encrypt/status:查看加密功能状态的端点
  • /key:查看密钥的端点
  • /encrypt:对请求的body内容进行加密的端点
  • /decrypt:对请求的body内容进行解密的端点
    可以通过GET请求访问/encrypt/status端点,我们将得到如下内容:
    1
    2
    3
    4
    {
    "description": "No key was installed for encryption service",
    "status": "NO_KEY"
    }
    返回说明当前配置中心的加密功能还不能使用,因为没有为加密服务配置对应的密钥。

配置密钥:
通过encrypt.key属性在配置文件中直接指定密钥信息(对称性密钥),比如:

1
2
encrypt:  
key: didispace #指定密钥信息

再访问http://localhost:1200/encrypt/status得到:

1
2
3
{
"status": "OK"
}

此时,我们配置中心的加密解密功能就已经可以使用了。

还有可以通过/encrypt和/decrypt端点来进行加密和解密的功能。
这两个端点都是POST请求,加密和解密信息需要通过请求体来发送。比如,以curl命令为例,我们可以通过下面的方式调用加密与解密端点:

1
2
3
4
$ curl localhost:7001/encrypt -d didispace
3c70a809bfa24ab88bcb5e1df51cb9e4dd4b8fec88301eb7a18177f1769c849ae9c9f29400c920480be2c99406ae28c7
$ curl localhost:7001/decrypt -d 3c70a809bfa24ab88bcb5e1df51cb9e4dd4b8fec88301eb7a18177f1769c849ae9c9f29400c920480be2c99406ae28c7
didispace

这里,我们通过配置encrypt.key参数来指定密钥的实现方式采用了对称性加密。这种方式实现比较简单,只需要配置一个参数即可。另外,我们也可以使用环境变量ENCRYPT_KEY来进行配置,让密钥信息外部化存储。

非对称加密
Spring Cloud Config的配置中心不仅可以使用对称性加密,也可以使用非对称性加密(比如:RSA密钥对)。虽然非对称性加密的密钥生成与配置相对复杂一些,但是它具有更高的安全性。

首先,我们需要通过keytool工具来生成密钥对。
keytool是JDK中的一个密钥和证书管理工具。它使用户能够管理自己的公钥/私钥对及相关证书,用于(通过数字签名)自我认证(用户向别的用户/服务认证自己)或数据完整性以及认证服务。
在JDK 1.4以后的版本中都包含了这一工具,它的位置在:%JAVA_HOME%\bin\keytool.exe。

生成密钥的具体命令如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
C:\Users\hu>keytool -genkeypair -alias config-server -keyalg RSA -keystore config-server.keystore
输入密钥库口令:
密钥库口令太短 - 至少必须为 6 个字符
输入密钥库口令:
再次输入新口令:
您的名字与姓氏是什么?
[Unknown]: 111111
您的组织单位名称是什么?
[Unknown]: 111111
您的组织名称是什么?
[Unknown]: 111111
您所在的城市或区域名称是什么?
[Unknown]: 111111
您所在的省/市/自治区名称是什么?
[Unknown]: 111111
该单位的双字母国家/地区代码是什么?
[Unknown]: 111111
CN=111111, OU=111111, O=111111, L=111111, ST=111111, C=111111是否正确?
[否]: 是

输入 <config-server> 的密钥口令
(如果和密钥库口令相同, 按回车):
再次输入新口令:

输入 <config-server> 的密钥口令
(如果和密钥库口令相同, 按回车):
再次输入新口令:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
如果我们不想逐步的输入那些提示信息,可以使用-dname来直接指定,而密钥库口令与密钥口令可使用-storepass和-keypass来直接指定。所以,我们可以通过下面的命令直接创建出与上述命令一样的密钥库:

keytool -genkeypair -alias config-server -keyalg RSA \
-dname "CN=zhaiyongchao, OU=company, O=organization, L=city, ST=province, C=china" \
-keypass 222222 \
-keystore config-server.keystore \
-storepass 111111 \

默认情况下,上述命令创建的密钥只有90天有效期。如果我们想要调整它的有效期,可以通过增加-validity参数来实现,比如我们可以通过下面的命令,让密钥的有效期延长到一年:
keytool -genkeypair -alias config-server -keyalg RSA \
-dname "CN=zhaiyongchao, OU=company, O=organization, L=city, ST=province, C=china" \
-keypass 222222 \
-keystore config-server.keystore \
-storepass 111111 \
-validity 365 \

上述的三种命令生成方式,最终都会在命令的当前执行目录下生成一个config-server.keystore文件。下面,我们需要将它保存在配置中心的文件系统中的某个位置,比如放在当前的用户目录下,然后在配置中心中加入相关的配置信息:

encrypt.key-store.location=file://${user.home}/config-server.keystore
encrypt.key-store.alias=config-server
encrypt.key-store.password=111111
encrypt.key-store.secret=222222
如果我们将config-server.keystore放在配置中心的src/main/resource目录下,也可以直接这样配置:encrypt.key-store.location=config-server.keystore。另外,非对称加密的配置信息也可以通过环境变量的方式进行配置,它们对应的具体变量名如下:

ENCRYPT_KEY_STORE_LOCATION
ENCRYPT_KEY_STORE_ALIAS
ENCRYPT_KEY_STORE_PASSWORD
ENCRYPT_KEY_STORE_SECRET
通过环境变量来配置密钥库相关信息可以获得更好的安全性,所以我们可以将敏感的口令信息存储在配置中心的环境变量中是一种不错的选择。

配置中心高可用
在生产环境,Config Server与服务注册中心一样,我们也需要将其扩展为高可用的集群。
spring cloud config 实现高可用非常简单,不需要为这些服务端做额外的配置,只需要遵守一个配置规则:将所有的Config Server都指向同一个Git仓库
这样所有的配置内容就通过统一的共享文件系统来维护,而客户端在指定Config Server位置时,只要配置Config Server外的均衡负载即可,就像如下图所示的结构:
虽然通过服务端负载均衡已经能够实现,但是作为架构内的配置管理,本身其实也是可以看作架构中的一个微服务。所以,另外一种方式更为简单的方法就是把config-server也注册为服务,这样所有客户端就能以服务的方式进行访问。
通过这种方法,只需要启动多个指向同一Git仓库位置的config-server就能实现高可用了。
在config-server-git的基础上复制一个名为:config-server-git-eureka
依赖中增加:

1
2
3
4
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-eureka</artifactId>
</dependency>

作为服务放到注册中心

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
spring:
application:
name: config-server
cloud:
config:
server:
git:
uri: https://github.com/huingsn/config-repo-demo.git
server:
port: 1200
# 配置服务注册中心
eureka:
client:
serviceUrl:
defaultZone: http://user:123456@localhost:8761/eureka

启动类增加@EnableDiscoveryClient //用来将config-server注册到上面配置的服务注册中心上去。

启动后可以访问注册中心,看到服务已经注册到注册中心。


config-client配置和config-server-git-eureka一样,复制config-client修改名字为config-client-eureka,需要增加

1
2
3
4
   <dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-eureka</artifactId>
</dependency>

修改配置

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
spring:
application:
name: huyun
cloud:
config:
discovery:
enabled: true #参数设置为true,开启通过服务来访问Config Server的功能
service-id: config-server #参数来指定Config Server注册的服务名。
profile: default #如之前通过URI的方式访问时候一样,用来定位Git中的资源。
label: master
fail-fast: true
server:
port: 1201
# 配置服务注册中心
eureka:
client:
serviceUrl:
defaultZone: http://user:123456@localhost:8761/eureka
debug: true

启动项目:
浏览http://localhost:1201/user即可得到key对应的值
demo 见:spring-cloud-config

Spring Cloud Hystrix

发表于 2018-08-14 | 分类于 springcloud

在微服务架构中,每个单元都在不同的进程中运行,依赖远程调用的方式执行,这就有可能因为网络原因或是依赖服务自身问题出现调用故障或延迟,此时调用方的请求不断增加,最后就会出现因等待出现故障的依赖方响应而形成任务积压,线程资源无法释放,最终导致自身服务的瘫痪,甚至导致整个系统的瘫痪。
为了解决这样的问题,因此产生了断路器等一系列的服务保护机制。

Spring Cloud Hystrix中实现了线程隔离、断路器等一系列的服务保护功能。它也是基于Netflix的开源框架 Hystrix实现的,该框架目标在于通过控制那些访问远程系统、服务和第三方库的节点,从而对延迟和故障提供更强大的容错能力。Hystrix具备了服务降级、服务熔断、线程隔离、请求缓存、请求合并以及服务监控等强大功能。

Hystrix服务降级

创建一个服务消费者hystrix-eureka-consumer-ribbon,pom.xml的dependencies节点中引入spring-cloud-starter-hystrix依赖:

1
2
3
4
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>

主类中使用@EnableCircuitBreaker或@EnableHystrix注解开启Hystrix的使用:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@EnableCircuitBreaker //或@EnableHystrix
@EnableEurekaClient
@SpringBootApplication
public class ConsumerApplication {
/**
* 通过注解 @LoadBalanced 使用Ribbon负载均衡
* Spring Cloud Ribbon是基于Netflix Ribbon实现的一套客户端负载均衡的工具。
* 这样就可以在Controller中,去掉通过LoadBalancerClient选取实例和拼接URL的步骤,直接通过RestTemplate发起请求。
* 请求的host位置并没有使用一个具体的IP地址和端口的形式,而是采用了服务名的方式组成,不需要具体的IP和端口。因为:
* Spring Cloud Ribbon有一个拦截器,它能够在这里进行实际调用的时候,自动的去选取服务实例,并将实际要请求的IP地址和端口替换这里的服务名,从而完成服务接口的调用。
* @return
*/
@Bean
@LoadBalanced
public RestTemplate restTemplate(){
return new RestTemplate();
}
public static void main(String[] args) {
SpringApplication.run(ConsumerApplication.class, args);
}
}

还可以使用Spring Cloud应用中的@SpringCloudApplication注解来修饰应用主类,该注解的具体定义如下,包含了上述的注解。

1
2
3
4
5
6
7
8
9
10
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootApplication
@EnableDiscoveryClient
@EnableCircuitBreaker
public @interface SpringCloudApplication {

}

在服务消费者服务中,新增ConsumerService类,在方法增加@HystrixCommand注解来指定服务降级方法,在controller中直接调用服务方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
    @Service
class ConsumerService {

@Autowired
RestTemplate restTemplate;

/**
* 注解:@HystrixCommand用来指定服务降级方法
*/
@HystrixCommand(fallbackMethod = "userFallback")
public String[] users() {
String url = "http://provider-hello/users";
return restTemplate.getForObject(url, String[].class);
}

/*@HystrixCommand(fallbackMethod = "fallback")
public String users(String id) {
String url = "http://provider-hello/users/" + id;
return restTemplate.getForObject(url, String.class);
}*/

//如果服务降级了,userFallback
public String[] userFallback() {
String[] arr ={"err"};
return arr;
}
/*public String fallback() {
return "fallback";
}*/
}
```
控制器直接调用:
```java
@RestController
public class ConsumerController {
@Autowired
private ConsumerService consumerService;
@Autowired
private DiscoveryClient discoveryClient;

@RequestMapping(value = "/users", method = RequestMethod.GET)
public String[] users() throws InterruptedException{
return consumerService.users();
}

/*@RequestMapping(value = "/users/{id}", method = RequestMethod.GET)
public String users(@PathVariable String id) {
return consumerService.users(id);
}*/
}

浏览正确输出值,为了验证服务降级,对代码做以下修改:

1
2
3
4
5
6
public String[] users() throws InterruptedException{
//为了出发服务降级,返回结果为arr
Thread.sleep(6000L);
System.out.println(Arrays.toString(discoveryClient.getServices().toArray()));
return consumerService.users();
}

由于延迟了,就返回fallback

依赖隔离

Hystrix通过实现线程池的隔离,它会为每一个Hystrix命令创建一个独立的线程池,这样就算某个在Hystrix命令包装下的依赖服务出现延迟过高的情况,也只是对该依赖服务的调用产生影响,而不会拖慢其他的服务。通过对依赖服务的线程池隔离实现,可以带来如下优势:

  • 应用自身得到完全的保护,不会受不可控的依赖服务影响。即便给依赖服务分配的线程池被填满,也不会影响应用自身的额其余部分。
  • 可以有效的降低接入新服务的风险。如果新服务接入后运行不稳定或存在问题,完全不会影响到应用其他的请求。
  • 当依赖的服务从失效恢复正常后,它的线程池会被清理并且能够马上恢复健康的服务,相比之下容器级别的清理恢复速度要慢得多。
  • 当依赖的服务出现配置错误的时候,线程池会快速的反应出此问题(通过失败次数、延迟、超时、拒绝等指标的增加情况)。同时,我们可以在不影响应用功能的情况下通过实时的动态属性刷新(后续会通过Spring Cloud Config与Spring Cloud Bus的联合使用来介绍)来处理它。
  • 当依赖的服务因实现机制调整等原因造成其性能出现很大变化的时候,此时线程池的监控指标信息会反映出这样的变化。同时,我们也可以通过实时动态刷新自身应用对依赖服务的阈值进行调整以适应依赖方的改变。
    每个专有线程池都提供了内置的并发实现,可以利用它为同步的依赖服务构建异步的访问。

总之,通过对依赖服务实现线程池隔离,让我们的应用更加健壮,不会因为个别依赖服务出现问题而引起非相关服务的异常。还可以在不停止服务的情况下,配合动态配置刷新实现性能配置上的调整。

线程池隔离的方案如此多的好处,使用者可能会担心为每一个依赖服务都分配一个线程池是否会过多地增加系统的负载和开销。但是Netflix在设计Hystrix的时候,认为隔离所带来的好处大于线程池上的开销。

如何使用

使用了@HystrixCommand来将某个函数包装成了Hystrix命令,这里除了定义服务降级之外,Hystrix框架还自动为这个方法实现了(框架内部)调用的隔离。依赖隔离、服务降级是一体化实现的。

Hystrix中除了使用线程池之外,还可以使用信号量来控制单个依赖服务的并发度,信号量的开销要比线程池的小,但是不能设置超时和实现异步访问,只有在依赖服务是足够可靠的情况下才使用信号量。
在HystrixCommand和HystrixObservableCommand中2处支持信号量的使用:

  • 命令执行:如果隔离策略参数execution.isolation.strategy设置为SEMAPHORE,Hystrix会使用信号量替代线程池来控制依赖服务的并发控制。
  • 降级逻辑:当Hystrix尝试降级逻辑时候,它会在调用线程中使用信号量。
    信号量的默认值为10,也可以通过动态刷新配置的方式来控制并发线程的数量。
断路器

分布式架构中,当某个服务单元发生故障(类似用电器发生短路)之后,通过断路器的故障监控(类似熔断保险丝),直接切断原来的主逻辑调用。
在Hystrix中的断路器除了切断主逻辑的功能之外,还有更复杂的逻辑。

在服务消费端的服务降级逻辑因为hystrix命令调用依赖服务超时,触发了降级逻辑,但是即使这样,受限于Hystrix超时时间的问题,调用依然很有可能产生堆积。

这个时候断路器就会发挥作用,断路器的三个重要参数:快照时间窗、请求总数下限、错误百分比下限。这个参数的作用分别是:

  • 快照时间窗:断路器确定是否打开需要统计一些请求和错误数据,而统计的时间范围就是快照时间窗,默认为最近的10秒。
  • 请求总数下限:在快照时间窗内,必须满足请求总数下限才有资格根据熔断。默认为20,意味着在10秒内,如果该hystrix命令的调用此时不足20次,即时所有的请求都超时或其他原因失败,断路器都不会打开。
  • 错误百分比下限:当请求总数在快照时间窗内超过了下限,比如发生了30次调用,如果在这30次调用中,有16次发生了超时异常,也就是超过50%的错误百分比,在默认设定50%下限情况下,这时候就会将断路器打开。

在断路器未打开之前,服务会根据之前的降级在延迟时间时返回fallback,当熔断器在10秒内发现请求总数超过20,并且错误百分比超过50%,这个时候熔断器打开。
打开之后,再有请求调用的时候,将不会调用主逻辑,而是直接调用降级逻辑,这个时候就不会等待5秒之后才返回fallback。通过断路器,实现了自动地发现错误并将降级逻辑切换为主逻辑,减少响应延迟的效果。

在断路器打开之后,处理逻辑并没有结束,我们的降级逻辑已经被成了主逻辑,对于原来的主逻辑要恢复问题,hystrix实现了自动恢复功能。当断路器打开,对主逻辑进行熔断之后,hystrix会启动一个休眠时间窗,在这个时间窗内,降级逻辑是临时的成为主逻辑,当休眠时间窗到期,断路器将进入半开状态,释放一次请求到原来的主逻辑上,如果此次请求正常返回,那么断路器将继续闭合,主逻辑恢复,如果这次请求依然有问题,断路器继续进入打开状态,休眠时间窗重新计时。

综述:hystrix的断路器实现了对依赖资源故障的端口、对降级策略的自动切换以及对主逻辑的自动恢复机制,使得微服务在依赖外部服务或资源的时候得到了保护,同时对于一些具备降级逻辑的业务需求可以实现自动化的切换与恢复,相比于设置开关由监控和运维来进行切换的传统实现方式显得更为智能和高效。

Hystrix监控面板

hystrix-eureka-consumer-ribbon中接口实现使用了@HystrixCommand修饰,HystrixCommand和HystrixObservableCommand实例在执行过程中记录这些请求情况的指标信息会在内存中汇总,并保留一段时间,以供内部或外部进行查询使用,users这个接口的调用情况会被Hystrix记录下来,以用来给断路器和Hystrix Dashboard使用。

配置监控

见项目hystrix-dashboard
引入依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
<?xml version="1.0" encoding="UTF-8"?>
<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>com.hu</groupId>
<artifactId>Hystrix-Dashboard</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging>

<parent>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-parent</artifactId>
<version>Dalston.SR1</version>
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
</properties>

<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-hystrix-dashboard</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>

<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>

</project>

配置:

1
2
3
4
5
6
7
8
server:
port: 7904
spring:
application:
name: hystrix-dashboard
logging:
level:
root: INFO

启动类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
* EnableHystrixDashboard注解,开启Hystrix Dashboard功能
* SpringCloudApplication是 spring cloud注解,包含了如下注解:
* -@Target({ElementType.TYPE})
* -@Retention(RetentionPolicy.RUNTIME)
* -@Documented
* -@Inherited
* -@SpringBootApplication
* -@EnableDiscoveryClient Eureka客户端,和EnableEurekaClient一个意思
* -@EnableCircuitBreaker 熔断器
*/
@EnableHystrixDashboard
@SpringCloudApplication
public class HystrixDashboardApplication {
public static void main(String[] args) {
SpringApplication.run(HystrixDashboardApplication.class, args);
}
}

访问:http://localhost:7904/hystrix

Hystrix Dashboard的监控首页,该页面中并没有具体的监控信息。 页面上两个参数:
Delay:该参数用来控制服务器上轮询监控信息的延迟时间,默认为2000毫秒,我们可以通过配置该属性来降低客户端的网络和CPU消耗。
Title:该参数对应了上图头部标题Hystrix Stream之后的内容,默认会使用具体监控实例的URL,我们可以通过配置该信息来展示更合适的标题。

从页面的文字内容中我们可以知道,Hystrix Dashboard共支持三种不同的监控方式,依次为:

  • 默认的集群监控:通过URLhttp://turbine-hostname:port/turbine.stream开启,实现对默认集群的监控。
  • 指定的集群监控:通过URLhttp://turbine-hostname:port/turbine.stream?cluster=[clusterName]开启,实现对clusterName集群的监控。
  • 单体应用的监控:通过URLhttp://hystrix-app:port/hystrix.stream开启,实现对具体某个服务实例的监控。

前两者都对集群的监控,需要整合Turbine才能实现,在这里对单个服务实例的监控。

Hystrix Dashboard监控单实例节点需要通过访问实例的/hystrix.stream接口来实现,那需要为服务实例添加这个端点,而添加该功能的步骤:

1、在服务实例hystrix-eureka-consumer-ribbon的pom.xml中新增spring-boot-starter-actuator监控模块以开启监控相关的端点并确保已经引入断路器的依赖spring-cloud-starter-hystrix

1
2
3
4
5
6
7
8
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

启动类中开启了断路器功能。

2、确保在服务实例的主类中已经使用@EnableCircuitBreaker或@EnableHystrix注解,开启了断路器功能。
之后再Hystrix Dashboard的首页输入http://localhost:7902/hystrix.stream,已启动对hystrix-eureka-consumer-ribbon的监控。

监控页面介绍:
在监控信息的左上部分找到两个重要的图形信息:一个实心圆和一条曲线。
实心圆,共有两种含义: 它通过颜色的变化代表了实例的健康程度,它的健康度从绿色、黄色、橙色、红色递减。 除了颜色的变化之外,大小也会根据实例的请求流量发生变化,流量越大该实心圆就越大。 所以通过该实心圆的展示,我们就可以在大量的实例中快速的发现故障实例和高压力实例。
曲线:用来记录2分钟内流量的相对变化,我们可以通过它来观察到流量的上升和下降趋势。

Hystrix集群监控数据聚合

使用Turbine来对服务的Hystrix数据进行聚合展示,有HTTP收集聚合和通过消息代理收集聚合两种方式。

1、通过HTTP收集聚合

创建一个模块,名字为:turbine,引入依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
<?xml version="1.0" encoding="UTF-8"?>
<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>

<artifactId>turbine</artifactId>

<parent>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-parent</artifactId>
<version>Dalston.SR1</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-turbine</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
</project>

配置:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
server:
port: 7922
spring:
application:
name: turbine
eureka:
client:
serviceUrl:
defaultZone: http://user:123456@localhost:8761/eureka
logging:
level:
root: INFO
management:
port: 9999
turbine:
app-config: eureka-consumer-ribbon-hystrix #指定了需要收集监控信息的服务名
cluster-name-expression: default #指定了集群名称为default
#服务数量非常多的时候,可以启动多个Turbine服务来构建不同的聚合集群,
#而该参数可以用来区分这些不同的聚合集群,同时该参数值可以在Hystrix仪表盘中用来定位不同的聚合集群,
#只需要在Hystrix Stream的URL中通过cluster参数来指定。
combine-host-port: true #让同一主机上的服务通过主机名与端口号的组合来进行区分,默认情况下会以host来区分不同的服务

启动类:

1
2
3
4
5
6
7
8
9
10
11
@Configuration
@EnableAutoConfiguration
@EnableTurbine //注解开启Turbine
@EnableDiscoveryClient
public class TurbineApplication {

public static void main(String[] args) {
SpringApplication.run(TurbineApplication.class, args);
}

}

创建完毕,启动项目:
访问Hystrix Dashboard,输入对turbine监控的url:http://localhost:7922/turbine.stream
将看到针对服务hystrix-eureka-consumer-ribbon的聚合监控数据。

2、通过消息代理收集聚合

Spring Cloud在封装Turbine的时候,还实现了基于消息代理的收集实现,可以将所有需要收集的监控信息都输出到消息代理中,然后Turbine服务再从消息代理中异步的获取这些监控信息,最后将这些监控信息聚合并输出到Hystrix Dashboard中。
创建模块turbine-amqp,添加配置和依赖,见项目turbine-amqp。
创建完成后需要对服务消费者hystrix-eureka-consumer-ribbon做修改,使其监控信息能够输出到RabbitMQ上。
在消费者服务上增加对mq的依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<?xml version="1.0" encoding="UTF-8"?>
<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>
<artifactId>turbine-amqp</artifactId>
<parent>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-parent</artifactId>
<version>Dalston.SR1</version>
<relativePath/>
</parent>

<dependencies>
<!--spring-cloud-starter-turbine-amqp-->
<!--包含了spring-cloud-starter-turbine-stream和pring-cloud-starter-stream-rabbit。-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-turbine-amqp</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
</project>
1
2
3
4
5
6
7
8
9
10
11
@Configuration
@EnableAutoConfiguration
@EnableTurbineStream //启用Turbine Stream的配置
@EnableDiscoveryClient
public class TurbineApplication {

public static void main(String[] args) {
SpringApplication.run(TurbineApplication.class, args);
}

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
server:
port: 7927
spring:
application:
name: turbine-mq
eureka:
client:
serviceUrl:
defaultZone: http://user:123456@localhost:8761/eureka
logging:
level:
root: INFO
management:
port: 9999

完成配置后,通过Hystrix Dashboard开启对turbine-amqp的监控,可以获得和之前的效果,只是这里我们的监控信息收集时是通过了消息代理异步实现的。

spring整合Oauth2

发表于 2018-07-20 | 分类于 spring boot

OAuth2.0是OAuth协议的延续版本,但不向后兼容OAuth 1.0即完全废止了OAuth1.0。 OAuth 2.0关注客户端开发者的简易性。要么通过组织在资源拥有者和HTTP服务商之间的被批准的交互动作代表用户,要么允许第三方应用代表用户获得访问的权限。

举个例子:

在某论坛上,希望用QQ登录。点了QQ登录,那论坛就引导跳转到QQ的授权页面了,同时需要对论坛进行认证。
在这个页面里,我要登录一下QQ了。前面已经认证过论坛了,现在登录就是认证我了。
登录成功后,QQ显示出授权的内容,让我确认,我点了确认。
确认之后,事实上论坛当初转到QQ授权页面的时候,除了提供自己的身份,还有一个redirectUrl。当我点了授权后,QQ把合同给了这个Url,等于就是给了论坛了。
论坛拿到了认证信息,再把自己的登录信息给QQ再验证一下,QQ就把我的资料给了论坛了。

标准的认证过程

OAuth认证和授权的过程如下

1、用户访问第三方网站网站,想对用户存放在服务商的某些资源进行操作。
2、第三方网站向服务商请求一个临时令牌。
3、服务商验证第三方网站的身份后,授予一个临时令牌。
4、第三方网站获得临时令牌后,将用户导向至服务商的授权页面请求用户授权,然后这个过程中将临时令牌和第三方网站的返回地址发送给服务商。
5、用户在服务商的授权页面上输入自己的用户名和密码,授权第三方网站访问所相应的资源。
6、授权成功后,服务商将用户导向第三方网站的返回地址。
7、第三方网站根据临时令牌从服务商那里获取访问令牌。
8、服务商根据令牌和用户的授权情况授予第三方网站访问令牌。
9、第三方网站使用获取到的访问令牌访问存放在服务商的对应的用户资源。

一些概念

1)Third-party application:第三方应用程序(client)。
2)HTTP service:HTTP服务提供商。
3)Resource Owner:资源所有者-"用户"(user)。
4)User Agent:用户代理-浏览器。
5)Authorization server:认证服务器,即服务提供商专门用来处理认证的服务器。
6)Resource server:资源服务器,即服务提供商存放用户生成的资源的服务器。它与认证服务器,可以是同一台服务器,也可以是不同的服务器。

运行流程

(A)用户打开客户端,客户端要求用户给予授权。
(B)用户同意给予客户端授权。
(C)客户端使用上一步获得的授权(一般是Code),向认证服务器申请令牌TOKEN。
(D)认证服务器对客户端进行认证以后,确认无误,同意发放令牌。
(E)客户端使用令牌,向资源服务器申请获取资源(用户信息等)。
(F)资源服务器确认令牌无误,同意向客户端开放资源。

客户端获取授权的五种模式

客户端必须得到用户的授权(authorization grant),才能获得令牌(access token)。

OAuth 2.0定义了五种授权方式:

关于session和token

    session是空间换时间,而token是时间换空间。session占用空间,但是可以管理过期时间,token管理部了过期时间,但是不占用空间.
    sessionId失效问题和token内包含。
    session基于cookie,app请求并没有cookie 。
    token更加安全(每次请求都需要带上)。

授权码模式(authorization code)
功能最完整、流程最严密的授权模式。
简化模式(implicit)
密码模式(resource owner password credentials)
客户端模式(client credentials)
扩展模式(Extension)

Java日期时间API完全解析

发表于 2018-06-22 | 分类于 java

时区

GMT(Greenwich Mean Time):格林尼治时间,格林尼治标准时间的正午是指当太阳横穿格林尼治子午线时(也就是在格林尼治上空最高点时)的时间。

UTC(Universal Time Coordinated):统一协调时间,其以原子时秒长为基础,在时刻上尽量接近于格林尼治标准时间,标准 UTC 时间格式 yyyy-MM-dd’T’HH:mm:ss.SSSXXX。

格林尼治时间已经不再被作为标准时间使用,UTC 是最主要的世界时间标准。

Java提供了获取当前时间的方法

  • System.currentTimeMillis() 返回当前时间,以毫秒为单位。
    表示是当前时刻至 1970-01-01 00:00:00.000 的毫秒差值。返回的long值可以用来初始化java.util.Date, java.sql.Date, java.sql.Timestamp和java.util.GregorianCalendar对象。

  • System.nanoTime() 返回一个时间值(系统计时器的当前值),精确到纳秒。
    它是由 JVM 提供的一个时间,主要用来精确衡量两个时间段之间的时间

时间粒度:System.currentTimeMillis()方法的时间粒度是大于1毫秒的。如果你反复执行这个方法,短时间内得到的结果是相同的,或又突然在某一次结果增加了几十毫秒,这是正常的。

旧的时间API

  • Java的日期/时间类的定义并不一致,在java.util和java.sql的包中都有日期类,此外用于格式化和解析的类在java.text包中定义。
  • java.util.Date同时包含日期和时间,而java.sql.Date仅包含日期,将其纳入java.sql包并不合理。另外这两个类都有相同的名字,这本身就是一个非常糟糕的设计。
  • 对于时间、时间戳、格式化以及解析,并没有一些明确定义的类。对于格式化和解析的需求,我们有java.text.DateFormat抽象类,但通常情况下,SimpleDateFormat类被用于此类需求。
  • 所有的日期类都是可变的,因此他们都不是线程安全的,这是Java日期类最大的问题之一。
  • 日期类并不提供国际化,没有时区支持,因此Java引入了java.util.Calendar和java.util.TimeZone类,但他们同样存在上述所有的问题。
java.util.Date

java.util.Date类用于封装日期及时间信息,一般仅用它显示某个日期,不对他作任何操作处理,作处理推荐用Calendar类,计算方便。

构造方法

Date() :分配 Date对象并初始化此对象,以表示分配它的时间(精确到毫秒)。

Date(long date):分配 Date对象并初始化此对象,以表示自从标准基准时间(称为“历元(epoch)”,即 1970 年 1 月 1 日 00:00:00 GMT)以来的指定毫秒数。

Calendar与GregorianCalendar

java.util.Calendar类用于封装日历信息,其主作用在于其方法可以对时间分量进行运算。Calendar类是一个抽象类,它为特定瞬间与一组诸如 YEAR、MONTH、DAY_OF_MONTH、HOUR 等 日历字段之间的转换提供了一些方法,并为操作日历字段(例如获得下星期的日期)提供了一些方法。

Calendar类是抽象类,构造方法是protected的,API中提供了getInstance方法用来创建对象。

Java只提供java.util.GregorianCalendar这一种java.util.Calendar的实现类。

SimpleDateFormat

SimpleDateFormat(String pattern),pattern -为描述日期和时间格式的模式,注:此构造方法可能不支持所有语言环境。要覆盖所有语言环境,请使用 DateFormat 类中的工厂方法。

1
2
public final String format(Date date)将一个 Date 格式化为日期/时间字符串
public Date parse(String source)throws ParseException从给定字符串的开始解析文本,以生成一个日期。
DateFormat

java.text.DateFormat类(抽象类)是SimpleDateFormat类的父类

java.sql.Date

java.sql.Date继承java.util.Date,为了把前者转为后者,不支持Date参数的构造器,传入long类型的时间。

java.sql.Date是SQL中的单纯的日期类型,没有时分秒。如果同时需要日期和时间,应该使用Timestamp。它也是 java.util.Date 的子类,Timestamp 则包含时间戳的完整信息。

java.sql.Timestamp是java.util.Date的派生类(继承),所以在java.util.Date上能做的事,也可以在java.sql.Timestamp上做。

现在的Date类中大部分方法已经弃用,现在一般使用旧的API,Date只负责存储一个时间,并对Calendar和DateFormat提供操作接口。Calendar获取Date中特定的信息,对日期时间进行操作,SimpleDateFormat对日期时间进行格式化输入输出。

总的来说,Date、Calendar 和 DateFormat 已经能够处理一般的时间日期问题了。但是它们依然很繁琐,不好用并且这些日期类都是可变且线程不安全的。

Java 8 时间日期API

特性

1
2
3
4
5
不变性:新的日期/时间API中,所有的类都是不可变的,这对多线程很有好处。
关注点分离:借鉴了Joda库的一些优点,新的API将人可读的日期时间和机器时间(unix timestamp)明确分离,它为日期(Date)、时间(Time)、日期时间(DateTime)、时间戳(unix timestamp)以及时区定义了不同的类。
清晰:在所有的类中,方法都被明确定义用以完成相同的行为。例如要拿到当前实例我们可以使用now()方法,在所有的类中都定义了format()和parse()方法,而不是像以前那样专门有一个独立的类。为了更好的处理问题,所有的类都使用了工厂模式和策略模式,一旦你使用了其中某个类的方法,与其他类协同工作并不困难。
实用操作:所有新的日期/时间API类都实现了一系列方法用以完成通用的任务,如:加、减、格式化、解析、从日期/时间中提取单独部分,等等。
可扩展性:新的日期/时间API是工作在ISO-8601日历系统上的,但我们也可以将其应用在非IOS的日历上。

Java8日期时间的默认格式如下:yyyy-MM-dd-HH-mm-ss.zzz

主要的 核心类:

1
2
3
4
5
6
7
8
9
10
11
12
LocalDate:日期类,不带时间
LocalTime:时间类,不带日期
LocalDateTime:日期和时间类
ZonedDateTime:时区日期时间类
OffsetDateTime:按UTC时间偏移来得到日期时间
Clock:获取某个时区下当前的瞬时时间,日期或者时间
Instant:Unix时间,代表时间戳,比如 2018-01-14T02:20:13.592Z
Duration:两个时间之间,表示一个绝对的精确跨度,使用毫秒为单位
Period:两个日期之间
ZoneId:时区
DateTimeFormatter:格式化输出
TemporalAdjusters:获得指定日期时间等,如当月的第一天、今年的最后一天等等
LocalDate、LocalTime、LocalDateTime

LocalDate是不变的日期时间对象代表一个日期,往往被视为年月日。其他日期字段,如一年中的一天,一周和一周的一天,也可以访问。

LocalTime是不变的日期时间对象代表一个时间,往往被视为小时分钟秒。时间为代表的纳秒级精度。

LocalDateTime是不变的日期时间对象代表一个日期时间,往往被视为年、月、日、时、分、秒。其他日期和时间字段,如一年中的一天,一周和一周的一天,也可以访问。时间为代表的纳秒级精度。

DateTimeFormatter

格式器用于解析日期字符串和格式化日期输出,创建格式器最简单的方法是通过 DateTimeFormatter 的静态工厂方法以及常量。创建格式器一般有如下三种方式:

1
2
3
常用 ISO 格式常量,如 ISO_LOCAL_DATE
字母模式,如 ofPattern("yyyy/MM/dd")
本地化样式,如 ofLocalizedDate(FormatStyle.MEDIUM)

使用DateTimeFormatter完成格式化

1
2
3
4
5
6
7
8
9
LocalDateTime localDateTime = LocalDateTime.now();
//创建一个格式化程序使用指定的模式。
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
String formatDateTime = localDateTime.format(formatter);
System.out.println(formatDateTime);

//DateTimeFormatter提供了一些默认的格式化器,DateTimeFormatter.ISO_LOCAL_DATE_TIME 格式 yyyy-MM-ddTHH:mm:ss.SSS
String dateTime2 = localDateTime.format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);
System.out.println(dateTime2);

使用DateTimeFormatter完成格式化

1
2
LocalDate localDate = LocalDate.parse("2018/11/11",DateTimeFormatter.ofPattern("yyyy/MM/dd"));
System.out.println(localDate); //2018-11-11

和旧的 java.util.DateFormat 相比较,所有的 DateTimeFormatter 实例都是线程安全的。

Instant

Instant 表示时间线上的一点(与 Date 类似),它只是简单地表示自 1970 年 1 月 1 日 0 时 0 分 0 秒(UTC)开始的秒数。

Instant 由两部分组成,一是从原点开始到指定时间点的秒数 s, 二是距离该秒数 s 的纳秒数。它以 Unix 时间戳的形式存储日期时间。

Instant类的工厂方法创建一个Instant实例

1
2
3
4
5
6
7
Instant now = Instant.now();
//第一个参数是秒,第二个是纳秒参数,纳秒的存储范围是0至999,999,999 2s之后的在加上100万纳秒(1s)
Instant instant = Instant.ofEpochSecond(2,1000000000);
System.out.println(instant); //1970-01-01T00:00:03Z
//java.util.Date与Instant可相互转换
Instant timestamp = new Date().toInstant();
Date.from(Instant.now());
Clock

Clock用于查找当前时刻,可以用来获取某个时区下当前的日期和时间,也可以用来代替旧的System.currentTimeMillis()方法和TimeZone.getDefault()方法。

Duration

一个Duration实例是不可变的,当创建出对象后就不能改变它的值了。你只能通过Duration的计算方法,来创建出一个新的Durtaion对象.

Period

Period 是以年月日来衡量一个时间段,用于计算两个日期间隔,所以 between() 方法只能接收 LocalDate 类型的参数。

ZonedDateTime和ZonedId

ZonedDateTime类是Java 8中日期时间功能里,用于表示带时区的日期与时间信息的类。ZonedDateTime 类的值是不可变的,所以其计算方法会返回一个新的ZonedDateTime 实例。

Java 使用 ZoneId 来标识不同的时区,从基准 UTC 开始的一个固定偏移。

TemporalAdjusters

有的时候,你需要进行一些更加复杂的操作,比如,将日期调整到下个周日、下个工作日,或者是本月的最后一天。

1
2
3
4
5
6
7
8
9
10
11
LocalDate localDate = LocalDate.now();  
// 1. 本月第一天
LocalDate firstDayOfMonth = localDate.with(TemporalAdjusters.firstDayOfMonth());
// 2. 本月最后一天
LocalDate lastDayOfMonth = localDate.with(TemporalAdjusters.lastDayOfMonth());
// 3. 本年第一天
LocalDate firstDayOfYear = localDate.with(TemporalAdjusters.firstDayOfYear());
// 4. 下个月第一天
LocalDate firstDayOfNextMonth = localDate.with(TemporalAdjusters.firstDayOfNextMonth());
// 5. 本年度最后一天
LocalDate lastDayOfYear = localDate.with(TemporalAdjusters.lastDayOfYear());

用重载版本的with方法,向其传递一个提供了更多定制化选择的TemporalAdjuster对象,更加灵活地处理日期。TemporalAdjusters类通过静态方法提供了大量的常用的TemporalAdjuster的实现供我们使用。

这些日期类之间转换

java.util.Date 与 LocalDate、LocalTime、LocalDateTime 转换

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
//将Date转换为LocalDate,LocalTime,LocalDateTime可以借助于ZonedDateTime和Instant,实现如下:

Date date = new Date();

// Date -> LocalDateTime
LocalDateTime localDateTime = date.toInstant().atZone(ZoneId.systemDefault()).toLocalDateTime();
System.out.println("localDateTime by Instant: " + localDateTime);

// Date -> LocalDate
LocalDate localDate = date.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();

// Date -> LocalTime
LocalTime localTime = date.toInstant().atZone(ZoneId.systemDefault()).toLocalTime();

//2. Date -> LocalDateTime
localDateTime = LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault());
System.out.println("localDateTime by ofInstant: " + localDateTime);

由于JDK8实现了向下兼容,所以Date里在JDK8版本引入了2个方法,from和toInstant,所以我们可以借助这两个方法来实现LocalDateTime到Date的转换。
将LocalDateTime转为Date如下:

1
2
3
4
5
6
7
8
LocalDateTime localDateTime = LocalDateTime.now();

// LocalDateTime -> Date
Date date = Date.from(localDateTime.atZone(ZoneId.systemDefault()).toInstant());

// LocalDate -> Date,时间默认都是00
LocalDate localDate = LocalDate.now();
date = Date.from(localDate.atStartOfDay(ZoneId.systemDefault()).toInstant());

日期与字符串的转换

1
2
3
4
5
6
7
8
9
10
11
12
//通过LocalDate,LocalTime,LocalDateTime的parse方法和DateTimeFormatter来实现:

//字符串->日期
LocalDate localDate = LocalDate.parse("2018-09-09", DateTimeFormatter.ofPattern("yyyy-MM-dd"));
LocalDateTime localDateTime = LocalDateTime.parse("2018-09-10 12:12:12", DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));

//日期->字符串
String localDate = LocalDate.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd"));
String localDateTime = LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
// 也可以通过DateTimeFormatter的format方法
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
localDateTime = dateTimeFormatter.format(LocalDateTime.now());

时间戳与LocalDateTime转换

1
2
3
4
5
6
7
8
9
10
11
12
13
//时间戳->LocalDateTime
public static LocalDateTime convertToDate(long timestamp) {
// ofEpochSecond 以秒为单位, ofEpochMilli 以毫秒为单位
// Instant.ofEpochSecond(timestamp);
Instant instant = Instant.ofEpochMilli(timestamp);
return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());
}

//LocalDateTime->时间戳
public static long convertToTimestamp() {
LocalDateTime localDateTime = LocalDateTime.now();
return localDateTime.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();
}

新的 API 区分各种日期时间概念并且各个概念使用相似的方法定义模式,这种相似性非常有利于 API 的学习。总结一下一般的方法或者方法前缀:

1
2
3
4
5
6
7
8
9
10
11
of:静态工厂方法,用于创建实例
now:静态工厂方法,用当前时间创建实例
parse:静态工厂方法,从字符串解析得到对象实例
get:获取时间日期对象的部分状态。
is:检查某些东西的是否是 true,例如比较时间前后
with:返回一个部分状态改变了的时间日期对象拷贝
plus:返回一个时间增加了的、时间日期对象拷贝
minus:返回一个时间减少了的、时间日期对象拷贝
to:转换到另一个类型
at:把这个对象与另一个对象组合起来,例如 date.atTime(time)
format:提供格式化时间日期对象的能力

参考文章:https://www.cnblogs.com/kiwenzhang/p/10960741.html

Joda-Time

Joda-Time提供了一组Java类包用于处理包括ISO8601标准在内的date和time。可以利用它把JDK Date和Calendar类完全替换掉,而且仍然能够提供很好的集成。

Joda 与 JDK 是百分之百可互操作的,因此您无需替换所有 Java 代码,只需要替换执行日期/时间计算的那部分代码。

如果时间需要按照今天,昨天这样返回,例子如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public class DateFormatUtil {
public static String format(Date date) {
DateTime now = new DateTime();
DateTime today_start = new DateTime(now.getYear(), now.getMonthOfYear(), now.getDayOfMonth(), 0, 0, 0);
DateTime today_end = today_start.plusDays(1);
DateTime yesterday_start = today_start.minusDays(1);

if(date.after(today_start.toDate()) && date.before(today_end.toDate())) {
return String.format("今天 %s", new DateTime(date).toString("HH:mm:ss"));
} else if(date.after(yesterday_start.toDate()) && date.before(today_start.toDate())) {
return String.format("昨天 %s", new DateTime(date).toString("HH:mm:ss"));
}

return new DateTime(date).toString("yyyy-MM-dd HH:mm:ss");
}
public static String formats(String dateString) {
if(StringUtils.isEmpty(dateString)){
return "";
}
try {
SimpleDateFormat dateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date date = dateFormat.parse(dateString);
return format(date);
}catch (ParseException e){
e.printStackTrace();
}
return dateString;
}
/*public static void main(String[] args) throws ParseException {
System.out.println(formats("2019-06-01 10:12:05"));
System.out.println(formats("2019-07-12 10:12:05"));
System.out.println(formats("2019-07-11 10:12:05"));
System.out.println(formats("2019-07-10 10:12:05"));
}*/
}

需要引入包:

1
2
3
4
<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
</dependency>

MongoDB基本操作

发表于 2018-06-14 | 分类于 MongoDB

1、查看数据库信息

在客户端指定数据库进行连接:(默认连接本机test数据库)

[root@localhost conf]# mongo 127.0.0.1/admin

查看数据库版本

1
2
> db.version()
3.6.11

查看所有数据库

1
2
3
4
> show dbs
admin 0.000GB
config 0.000GB
local 0.000GB

切换数据库

1
2
> use test;
switched to db test

显示当前数据库

1
2
3
4
> db
test
> db.getName()
test

查看当前数据库状态

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
> db.stats();
{
"db" : "test",
"collections" : 0,
"views" : 0,
"objects" : 0,
"avgObjSize" : 0,
"dataSize" : 0,
"storageSize" : 0,
"numExtents" : 0,
"indexes" : 0,
"indexSize" : 0,
"fileSize" : 0,
"fsUsedSize" : 0,
"fsTotalSize" : 0,
"ok" : 1
}

查看当前数据库的连接机器地址

1
2
> db.getMongo();
connection to 127.0.0.1:27017

2、数据管理

创建数据库,当use的时候,系统就会自动创建一个数据库。
如果use之后没有创建任何集合。系统就会删除这个数据库。

1
2
> use demo;
switched to db demo

删除数据库:如果没有选择任何数据库,会删除默认的test数据库

1
2
> db.dropDatabase()
{ "ok" : 1 }

创建集合

1
2
3
4
> db.createCollection('h');
{ "ok" : 1 }
> db.createCollection('u');
{ "ok" : 1 }

显示集合

1
2
3
4
5
> show collections;
h
u
> db.getCollectionNames()
[ "h", "u" ]

当插入一个文档的时候,一个集合就会自动创建。

1
2
3
4
> db.c.insert({name:'hu'});
WriteResult({ "nInserted" : 1 })
> db.c.insert({ming:'yun'});
WriteResult({ "nInserted" : 1 })

查看创建的集合

1
2
> db.getCollectionNames()
[ "c", "h", "u" ]

查看集合内容

1
2
3
> db.c.find()
{ "_id" : ObjectId("5c84a575435a08991662e280"), "name" : "hu" }
{ "_id" : ObjectId("5c84a58d435a08991662e281"), "ming" : "yun" }

重命名集合

1
2
3
4
> db.c.renameCollection('shun');
{ "ok" : 1 }
> db.getCollectionNames()
[ "h", "shun", "u" ]

删除集合

1
2
3
4
> db.h.drop();
true
> db.getCollectionNames()
[ "shun", "u" ]

批量插入数据到集合

1
2
> for(i=0;i<1000;i++){db.h.insert({'id':i,'name':'huyun','age':100,'date':new Date()});}
WriteResult({ "nInserted" : 1 })

查询集合中的查询所有记录

1
> db.h.find()

注:默认每页显示20条记录,当显示不下的的情况下,可以用it迭代命令查询下一页数据。

每页显示50条

1
2
> DBQuery.shellBatchSize=50;
50

查询第一条

1
2
3
4
5
6
7
8
9
10
> db.h.findOne()
{
"_id" : ObjectId("5c84a69f435a08991662e282"),
"id" : 1,
"name" : "huyun",
"age" : 100,
"date" : ISODate("2019-03-10T05:54:39.904Z")
}
> db.h.count()
1000

按照id查找

1
2
3
4
5
> db.h.find({id:100});
{ "_id" : ObjectId("5c84a8a8435a08991662e6ce"), "id" : 100, "name" : "huyun", "age" : 100, "date" : ISODate("2019-03-10T06:03:20.283Z") }

> db.h.distinct("name") # 查询去掉当前集合中某列的重复数据
> db.h.remove({}) # 删除集合中所有记录

查看集合存储信息

1
2
3
4
5
> db.h.stats()          # 查看数据状态
> db.h.dataSize() # 集合中数据的原始大小
> db.h.totalIndexSize() # 集合中索引数据的原始大小
> db.h.totalSize() # 集合中索引+数据压缩存储之后的大小
> db.h.storageSize() # 集合中数据压缩存储的大小

MongoDB介绍和安装

发表于 2018-06-14 | 分类于 MongoDB

MongoDB介绍

MongoDB是跨平台的、一个基于分布式文件存储的数据库。由C++语言编写。用它创建的数据库具备性能高、可用性强、易于扩展等特点。MongoDB将数据存储为一个文档,数据结构由键值(key:value)对组成。MongoDB文档类似于 JSON 对象。字段值可以包含字符串、数字、数组、当然也可以是另一个文档。文档是可以嵌套的。

20200413174638

20200413174652

集合相当于关系型数据库中的表。集合就是一组MongoDB文档。集合存在于数据库中,集合没有固定的结构,这意味着可以对集合插入不同格式和类型的数据,但通常情况下我们插入集合的数据都会有一定的关联性。

注:

集合名不能是空字符串""。
集合名不能含有\0字符(空字符),这个字符表示集合名的结尾。
集合名不能以"system."开头,这是为系统集合保留的前缀。
集合名用户创建的集合名字不能含有保留字符。有些驱动程序的确支持在集合名里面包含,这是因为某些系统生成的集合中包含该字符。除非你要访问这种系统创建的集合,否则千万不要在名字里出现$。

文档相当于关系型数据库中的行。它一组键值对,具有动态的模式,所以文档不需要设置相同的字段,并且相同的字段不需要相同的数据类型,这一点,跟关系型数据库有很大的不同。

注:

文档中的键/值对是有序的。
MongoDB区分类型和大小写。
MongoDB的文档不能有重复的键。
键不能含有\0 (空字符)。
.和$有特别的意义,只有在特定环境下才能使用。

参考:https://www.cnblogs.com/clsn/p/8214194.html#auto_id_10

MongoDB数据存储格式:JSON格式和BSON格式

BSON基于JSON格式,选择JSON进行改造的原因主要是JSON的通用性及JSON的schemaless的特性。

  二进制的JSON,JSON文档的二进制编码存储格式
  BSON有JSON没有的Date和BinData
  MongoDB中document以BSON形式存放

MongoDB适用场景

  网站数据、缓存等大尺寸、低价值的数据
  在高伸缩性的场景,用于对象及JSON数据的存储。

20200413174557

MongoDB安装

MongoDB官网:https://www.mongodb.com/

CentOS6.X版本软件下载地址:https://www.mongodb.org/dl/linux/x86_64-rhel62

步骤:

创建用户:

[root@localhost ~]# groupadd -g 800 mongod
[root@localhost ~]# useradd -u 801 -g mongod mongod

修改密码

[root@localhost ~]# echo 123456 |passwd --stdin mongod

下载安装程序:

[root@localhost ~]# mkdir -p /usr/local/mongodb/
[root@localhost ~]# cd /usr/local/mongodb/
[root@localhost mongodb]# wget http://downloads.mongodb.org/linux/mongodb-linux-x86_64-rhel62-3.6.11.tgz

解压安装

[root@localhost mongodb]# tar -xf mongodb-linux-x86_64-rhel62-3.6.11.tgz

创建目录

[root@localhost mongodb]# mkdir  -p  bin  conf  log  data

复制文件

[root@localhost mongodb]# cp mongodb-linux-x86_64-rhel62-3.6.11/bin/* bin/

添加环境变量

[root@localhost mongodb]# vim /etc/profile
添加export PATH=/usr/local/mongodb/bin:$PATH
[root@localhost mongodb]# source /etc/profile

测试:

1
2
3
4
5
6
7
8
9
10
启动:mongod --dbpath=/usr/local/mongodb/data --logpath=/usr/local/mongodb/log/mongodb.log --port=27017 --logappend --fork
关闭:mongod --shutdown --dbpath=/usr/local/mongodb/data --logpath=/usr/local/mongodb/log/mongodb.log --port=27017 --logappend --fork
--dbpath 数据存放路径
--logpath 日志文件路径
--logappend 日志输出方式
--port 启用端口号
--fork 在后台运行
--auth 是否需要验证权限登录(用户名和密码)
--bind_ip 限制访问的ip
--shutdown 关闭数据库

登录数据库:

[root@localhost bin]# mongo
MongoDB shell version v3.6.11
connecting to: mongodb://127.0.0.1:27017/?gssapiServiceName=mongodb

使用配置文件的方式启动数据库:

[root@localhost bin]# cd ../conf/
[root@localhost conf]# vim mongod.conf
      dbpath=/usr/local/mongodb/data
      logpath=/usr/local/mongodb/log/mongodb.log
      port=27017
      logappend=1
      fork=1

启动

[root@localhost conf]# mongod -f mongod.conf

关闭

[root@localhost conf]# mongod -f mongod.conf –shutdown

使用YAML格式配置文件:

1
2
3
4
5
6
7
8
9
10
11
12
13
[root@localhost conf]# vim mongod.conf
systemLog:
destination: file
path: "/usr/local/mongodb/log/mongodb.log"
logAppend: true
storage:
journal:
enabled: true
dbPath: "/usr/local/mongodb/data"
processManagement:
fork: true
net:
port: 27017

在数据库中关闭数据库的方法:

use admin
db.shutdownServer()

用户管理

MongoDB数据库默认是没有用户名及密码的,即无权限访问限制。为了方便数据库的管理和安全,需创建数据库用户。

用户权限

Read      允许用户读取指定数据库
readWrite           允许用户读写指定数据库
dbAdmin             允许用户在指定数据库中执行管理函数,如索引创建、删除,查看统计或访问system.profile
userAdmin           允许用户向system.users集合写入,可以找指定数据库里创建、删除和管理用户
clusterAdmin        只在admin数据库中可用,赋予用户所有分片和复制集相关函数的管理权限。
readAnyDatabase     只在admin数据库中可用,赋予用户所有数据库的读权限
readWriteAnyDatabase        只在admin数据库中可用,赋予用户所有数据库的读写权限
userAdminAnyDatabase        只在admin数据库中可用,赋予用户所有数据库的userAdmin权限
dbAdminAnyDatabase          只在admin数据库中可用,赋予用户所有数据库的dbAdmin权限。
root                只在admin数据库中可用。超级账号,超级权限

创建用户

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
> use admin
switched to db admin

> db.createUser(
... {
... user:"root",
... pwd:'123456',
... roles:[{role:"root",db:"admin"}]
... });
Successfully added user: {
"user" : "root",
"roles" : [
{
"role" : "root",
"db" : "admin"
}
]
}

创建管理员角色用户的时候,必须到admin下创建。 删除的时候也要到相应的库下操作。

创建完成后查看用户

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
> show tables;
system.users
system.version
> show users;
{
"_id" : "admin.root",
"user" : "root",
"db" : "admin",
"roles" : [
{
"role" : "root",
"db" : "admin"
}
]
}

验证用户是否能用

1
2
> db.auth('root','123456');
1

在配置文件中开启用户验证,修改配置文件

1
2
3
4
5
[root@localhost conf]# cat >>mongod.conf<<-'EOF'
> security:
> authorization: enabled
> EOF
[root@localhost conf]# mongod restart

重启后

mongo -uroot -p123456 admin

创建对某库的只读用户roles: [ { role: “read”, db: “test” } ]

创建某库的读写用户roles: [ { role: “readWrite”, db: “test” } ]

创建对多库不同权限的用户

roles: [ { role: "readWrite", db: "app" },
        { role: "read", db: "test" }
]

删除用户

db.dropUser("hu")

创建数据库的管理员

roles: [ { role: "dbAdmin", db: "test" } ]

创建数据库读写权限的用户并具有clusterAdmin权限:

roles: [ { role: "readWrite", db: "test" },
{ role: "clusterAdmin", db: "admin" }

MongoDB和SQL对比(转自网络)

20200413175520

20200413175548

20200413175600

安装启动服务失败,直接跳过,手动启动

.\mongod –dbpath=”D:\Program Files\MongoDB\Server\4.0\data”

Springboot中使用AOP统一处理Web请求日志异常

发表于 2018-06-14 | 分类于 spring boot

日志

面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
AOP是Spring框架中的一个重要内容,它通过对既有程序定义一个切入点,然后在其前后切入不同的执行内容,比如常见的有:打开数据库连接/关闭数据库连接、打开事务/关闭事务、记录日志等。

基于AOP不会破坏原来程序逻辑,因此它可以很好的对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?xml version="1.0" encoding="UTF-8"?>
<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>com.hu</groupId>
<artifactId>spring-boot-aop-logging</artifactId>
<version>1.0-SNAPSHOT</version>

<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.10.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
</dependencies>
</project>

配置:

1
2
3
4
5
6
#引入AOP依赖包后,一般来说并不需要去做其他配置。也不用主类中增加@EnableAspectJAutoProxy来启用。
#spring.aop.auto属性默认是开启的,也就是说只要引入了AOP依赖后,默认已经增加了@EnableAspectJAutoProxy。
spring:
aop:
auto: true #默认值
#proxy-target-class: true#CGLIB来实现AOP,不然默认使用的是标准Java的实现。

切面类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
@Aspect //定义切面类
@Component
public class WebLogAspect {

private Logger logger = Logger.getLogger(getClass());

//如果需要统计一个方法处理请求的处理时间消耗,需要在doBefore中记录时间,在doAfterReturning中统计时间
//就需要一个对象统一处理时间,在两个方法中调用,就会涉及到同步问题,可以通过ThreadLocal来记录
ThreadLocal<Long> startTime = new ThreadLocal<>();


//定义一个切入点
@Pointcut("execution(public * com.hu.controller..*.*(..))")
public void webLog(){}

@Before("webLog()")
public void doBefore(JoinPoint joinPoint) throws Throwable {
startTime.set(System.currentTimeMillis());
// 接收到请求,记录请求内容
ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = attributes.getRequest();

// 记录下请求内容
logger.info(joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName()
+">>>URL:" + request.getRequestURL().toString()+">>>HTTP_METHOD:" + request.getMethod()+">>>IP:" + request.getRemoteAddr()
+ ">>>ARGS:" + Arrays.toString(joinPoint.getArgs()));

}

@AfterReturning(returning = "ret", pointcut = "webLog()")
public void doAfterReturning(Object ret) throws Throwable {
// 处理完请求,返回内容
logger.info(">>>RESPONSE:" + ret);
logger.info(">>>SPEND TIME:" + (System.currentTimeMillis() - startTime.get()));

}
/*AOP切面的优先级
由于通过AOP实现,程序得到了很好的解耦,如果涉及有多个切面,会有切面执行顺序问题,就需要定义切面的优先级。
@Order(i)注解来标识切面的优先级。i的值越小,优先级越高。
在切入点前的操作,按order的值由小到大执行
在切入点后的操作,按order的值由大到小执行*/
}

统一异常处理

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
@ControllerAdvice
public class ControllerExceptionHandler {
private static final Logger logger = LoggerFactory.getLogger(ControllerExceptionHandler.class);

/**
* 处理不可知的一些异常,包括业务上RuntimeException异常
* @param e
* @return
*/
@ExceptionHandler(Exception.class)
@ResponseBody
public RetViewModel handlerException(Exception e, HttpServletRequest request){
logger.error(e.getMessage(),e);
return new RetViewModel(HttpStatus.INTERNAL_SERVER_ERROR.value(), Constants.INTERNAL_SERVER_ERROR,null);
}

/**
* 远程请求异常
*/
@ExceptionHandler(HttpClientErrorException.class)
@ResponseBody
public RetViewModel httpClientErrorException(HttpClientErrorException e, HttpServletRequest request){
logger.error(Constants.REMOTE_R,e.getMessage());
return new RetViewModel(e.getRawStatusCode(), Constants.REMOTE_R,null);
}

/**
* 不支持当前请求方法
* @param e
* @return
*/
@ExceptionHandler(HttpRequestMethodNotSupportedException.class)
@ResponseBody
public RetViewModel handleHttpRequestMethodNotSupportedException(HttpRequestMethodNotSupportedException e) {
logger.error(Constants.METHOD_NOT_ALLOWED, e);
return new RetViewModel(HttpStatus.METHOD_NOT_ALLOWED.value(),Constants.METHOD_NOT_ALLOWED,null);
}

/**
* 不支持当前媒体类型
* @param e
* @return
*/
@ExceptionHandler(HttpMediaTypeNotSupportedException.class)
@ResponseBody
public RetViewModel handleHttpMediaTypeNotSupportedException(Exception e) {
logger.error(Constants.UNSUPPORTED_MEDIA_TYPE, e);
return new RetViewModel(HttpStatus.UNSUPPORTED_MEDIA_TYPE.value(),Constants.UNSUPPORTED_MEDIA_TYPE,null);
}

/**
* 数据校验
*/
@ExceptionHandler(BindException.class)
@ResponseBody
public RetViewModel methodArgumentNotValidHandler(BindException e) {
logger.error(Constants.BAD_REQUEST, e);
BindingResult bindingResult = e.getBindingResult();
Map<String, Object> map = getErrors(bindingResult);
RetViewModel responseResult = new RetViewModel(HttpStatus.BAD_REQUEST.value(),map.values().toString(),null);
return responseResult;
}
/**
* 参数类型不匹配
*/
@ExceptionHandler({TypeMismatchException.class})
@ResponseBody
public RetViewModel requestTypeMismatch(TypeMismatchException ex){
logger.error(Constants.BAD_REQUEST, ex);
return new RetViewModel(HttpStatus.BAD_REQUEST.value(),"参数类型不匹配,参数" + ex.getValue() + "类型应该为" + ex.getRequiredType(),null);
}

/**
* 缺少参数异常
*/
@ExceptionHandler({MissingServletRequestParameterException.class})
@ResponseBody
public RetViewModel requestMissingServletRequest(MissingServletRequestParameterException ex){
logger.error(Constants.BAD_REQUEST, ex);
return new RetViewModel(HttpStatus.BAD_REQUEST.value(), "缺少必要参数,参数名称为" + ex.getParameterName(),null);
}
/**
* 请求参数错误
* @param e
* @return
*/
@ExceptionHandler({HttpMessageNotReadableException.class})
@ResponseBody
public RetViewModel handleHttpMessageNotReadableException(HttpMessageNotReadableException e) {
logger.error(Constants.BAD_REQUEST, e);
return new RetViewModel(HttpStatus.BAD_REQUEST.value(),Constants.BAD_REQUEST,null);
}

private Map<String, Object> getErrors(BindingResult result) {
Map<String, Object> map = new HashMap<>();
List<FieldError> list = result.getFieldErrors();
for (FieldError error : list) {
map.put(error.getField(), error.getDefaultMessage());
}
return map;
}
}
Exception Type HTTP Status Code
ConversionNotSupportedException 500 (Internal Server Error)
HttpMediaTypeNotSupportedException 406 (Not Acceptable)
HttpMediaTypeNotSupportedException 415 (Unsupported Media Type)
HttpMessageNotReadableException 400 (Bad Request)
HttpMessageNotWritableException 500 (Internal Server Error)
HttpRequestMethodNotSupportedException 405 (Method Not Allowed)
MissingServletRequestParameterException 400 (Bad Request)
NoSuchRequestHandlingMethodException 404 (Not Found)
TypeMismatchException 400 (Bad Request)

Rest和Restful风格

发表于 2018-05-14 | 分类于 rest

网络应用程序,分为前端和后端两个部分。当前的发展趋势,就是前端设备层出不穷(手机、平板、桌面电脑、其他专用设备……)。

REST 用来规范应用如何在 HTTP 层与 API 提供方进行数据交互 。REST 是面向资源的,这个概念非常重要,而资源是通过 URI 进行暴露。

一、传统下的API接口

删除一个数据,以往的做法通常是 delete/{id} 
更新一个数据,可能是Post数据放Body,然后方法是 update/{id}, 或者是artichle/{id}?method=update 
1
2
3
4
5
6
7
8
/api/getUser
/api/createApp
/api/searchResult
/api/deleteAllUsers
/api/getUser (用来获取某个用户的信息,还需要以参数方式传入用户 id 信息)
/api/updateUser (用来更新用户信息)
/api/deleteUser/id (用来删除单个用户)
/api/resetUser (重置用户的信息)

这样的弊端在于:首先加上了动词,肯定是使 URL 更长了;其次对一个资源实体进行不同的操作就是一个不同的 URL,造成 URL 过多难以管理。

二、REST 架构的链接

1. URL 中不应该出现任何表示操作的动词,链接只用于对应资源;
2. URL 中应该单复数区分,推荐的实践是永远只用复数;比如 GET /api/users 表示获取用户的列表;如果获取单个资源,传入 ID,比如 /api/users/123 表示获取单个用户的信息;
3. 按照资源的逻辑层级,对 URL 进行嵌套,比如一个用户属于某个团队,而这个团队也是众多团队之一;那么获取这个用户的接口可能是这样:

GET /api/teams/123/members/234 表示获取 id 为 123 的小组下,id 为234 的成员信息

rest原则:

<1>网络上的所有事物都被抽象为资源
<2> 每个资源都有一个唯一的资源标识符
<3> 同一个资源具有多种表现形式(xml,json等)
<4> 对资源的各种操作不会改变资源标识符
<5> 所有的操作都是无状态的

REST是一个标准,一种规范,遵循REST风格可以使开发的接口通用,便于调用者理解接口的作用。

关于restful

基于REST构建的API就是Restful风格。

RESTful架构风格规定,数据的元操作,即CRUD(create, read, update和delete,即数据的增删查改)操作,分别对应于HTTP方法:GET用来获取资源,POST用来新建资源(也可以用于更新资源),PUT用来更新资源,DELETE用来删除资源,这样就统一了数据操作的接口,仅通过HTTP方法,就可以完成对数据的所有增删查改工作。

GET(SELECT):从服务器取出资源(一项或多项)。
POST(CREATE):在服务器新建一个资源。
PUT(UPDATE):在服务器更新资源(客户端提供完整资源数据)。
PATCH(UPDATE):在服务器更新资源(客户端提供需要修改的资源数据)。
DELETE(DELETE):从服务器删除资源。

restful用法:

在Restful之前的操作:

http://127.0.0.1/user/query/1 GET  根据用户id查询用户数据
http://127.0.0.1/user/save POST 新增用户
http://127.0.0.1/user/update POST 修改用户信息
http://127.0.0.1/user/delete GET/POST 删除用户信息

RESTful用法:

http://127.0.0.1/user/1 GET  根据用户id查询用户数据
http://127.0.0.1/user 新增用户
http://127.0.0.1/user 修改用户信息
http://127.0.0.1/user 删除用户信息

在RESTful架构中,每个网址代表一种资源(resource),所以网址中不能有动词,只能有名词,而且所用的名词往往与数据库的表名对应,一般来说,数据库中的表都是同种记录的”集合”(collection),所以API中的名词也应该使用复数。

使用标准的状态码

GET

安全且幂等 获取表示 变更时获取表示(缓存)

200(OK) - 表示已在响应中发出
204(无内容) - 资源有空表示
301(Moved Permanently) - 资源的URI已被更新
303(See Other) - 其他(如,负载均衡)
304(not modified)- 资源未更改(缓存)
400 (bad request)- 指代坏请求(如,参数错误)
404 (not found)- 资源不存在
406 (not acceptable)- 服务端不支持所需表示
500 (internal server error)- 通用错误响应
503 (Service Unavailable)- 服务端当前无法处理请求

POST

不安全且不幂等 使用服务端管理的(自动产生)的实例号创建资源

创建子资源 部分更新资源

如果没有被修改,则不过更新资源(乐观锁)

200(OK)- 如果现有资源已被更改
201(created)- 如果新资源被创建
202(accepted)- 已接受处理请求但尚未完成(异步处理)
301(Moved Permanently)- 资源的URI被更新
303(See Other)- 其他(如,负载均衡)
400(bad request)- 指代坏请求
404 (not found)- 资源不存在
406 (not acceptable)- 服务端不支持所需表示
409 (conflict)- 通用冲突
412 (Precondition Failed)- 前置条件失败(如执行条件更新时的冲突)
415 (unsupported media type)- 接受到的表示不受支持
500 (internal server error)- 通用错误响应
503 (Service Unavailable)- 服务当前无法处理请求

PUT

不安全但幂等,用客户端管理的实例号创建一个资源,通过替换的方式更新资源,如果未被修改,则更新资源(乐观锁)

200 (OK)- 如果已存在资源被更改
201 (created)- 如果新资源被创建
301(Moved Permanently)- 资源的URI已更改
303 (See Other)- 其他(如,负载均衡)
400 (bad request)- 指代坏请求
404 (not found)- 资源不存在
406 (not acceptable)- 服务端不支持所需表示
409 (conflict)- 通用冲突
412 (Precondition Failed)- 前置条件失败(如执行条件更新时的冲突)
415 (unsupported media type)- 接受到的表示不受支持
500 (internal server error)- 通用错误响应
503 (Service Unavailable)- 服务当前无法处理请求

DELETE

不安全但幂等,删除资源。

200 (OK)- 资源已被删除
301 (Moved Permanently)- 资源的URI已更改
303 (See Other)- 其他,如负载均衡
400 (bad request)- 指代坏请求
404 (not found)- 资源不存在
409 (conflict)- 通用冲突
500 (internal server error)- 通用错误响应
503 (Service Unavailable)- 服务端当前无法处理请求

表示结构

json xml

JVM内存分配和参数

发表于 2018-05-10 | 分类于 java

打印参数

在虚拟机运行过程中,可以根据一些跟踪系统状态的参数,来排查故障或者参数设置,可以打印系统运行时的一些相关参数,分析实际问题。对参数的配置,主要也就是围绕堆、栈、方法区进行配置。

打印jvm日志的方式

eclipse

20200410181219

控制台得到参数

1
2
3
4
5
6
7
8
9
Heap
PSYoungGen total 35840K, used 2458K [0x00000000d8500000, 0x00000000dad00000, 0x0000000100000000)
eden space 30720K, 8% used [0x00000000d8500000,0x00000000d8766888,0x00000000da300000)
from space 5120K, 0% used [0x00000000da800000,0x00000000da800000,0x00000000dad00000)
to space 5120K, 0% used [0x00000000da300000,0x00000000da300000,0x00000000da800000)
ParOldGen total 81920K, used 0K [0x0000000088e00000, 0x000000008de00000, 0x00000000d8500000)
object space 81920K, 0% used [0x0000000088e00000,0x0000000088e00000,0x000000008de00000)
Metaspace used 2748K, capacity 4486K, committed 4864K, reserved 1056768K
class space used 293K, capacity 386K, committed 512K, reserved 1048576K

IntelliJ IDEA:

于是,运行程序后,GC日志就可以打印出来了,和eclipse测试一样。

解读日志信息:

1
2
3
4
5
6
7
8
9
10
11
12
Heap
新生代分配情况
PSYoungGen total 35840K, used 3072K [0x00000000d8500000, 0x00000000dad00000, 0x0000000100000000)
eden space 30720K, 10% used [0x00000000d8500000,0x00000000d88002b8,0x00000000da300000)
from space 5120K, 0% used [0x00000000da800000,0x00000000da800000,0x00000000dad00000)
to space 5120K, 0% used [0x00000000da300000,0x00000000da300000,0x00000000da800000)
老年代分配情况
ParOldGen total 81920K, used 0K [0x0000000088e00000, 0x000000008de00000, 0x00000000d8500000)
object space 81920K, 0% used [0x0000000088e00000,0x0000000088e00000,0x000000008de00000)
元空间分配情况,jdk8彻底去除了永久代
Metaspace used 3125K, capacity 4496K, committed 4864K, reserved 1056768K
class space used 342K, capacity 388K, committed 512K, reserved 1048576K

堆的分配参数

1、-Xmx -Xms:指定最大堆和最小堆
    -Xmx设置程序能获得最大堆大小,-Xms设置程序启动时初始堆大小
2、-XX:+PrintGC 遇到GC就会打印日志。
3、-XX:+UseSerialGC 配置串行回收器
4、-XX:+PrintGCDetails查看GC详情,包括各个回收区的情况。
5、-XX:+PrintCommandLineFlags:可以将隐式或显示传给JVM的参数输出,就是把配置的参数所改变的信息也打印输出。

-XX这种这是是对jvm系统级别的配置,非-XX配置基本是对应用级别的设置。+启用-禁用

实际中,可以直接将初始的堆大小与最大堆大小设置相等,这样的好处是可以减少程序运行时的垃圾回收次数,从而提到性能。

默认程序堆大小,可以通过Java程序获取

1
2
3
4
5
public static void main(String[] args) {
System.out.println("Xmx=" + Runtime.getRuntime().maxMemory() / 1024.0 / 1024 + "M"); // 系统的最大空间
System.out.println("free mem=" + Runtime.getRuntime().freeMemory() / 1024.0 / 1024 + "M"); // 系统的空闲空间
System.out.println("total mem=" + Runtime.getRuntime().totalMemory() / 1024.0 / 1024 + "M"); // 当前可用的总空间
}
Xmx=1694.5M
free mem=113.19983673095703M
total mem=115.0M

Demo测试设置(多个命令一起执行): -XX:+PrintGC -Xms40M -Xmx40M -Xmn20M -XX:+PrintGCDetails -XX:+UseSerialGC -XX:+PrintCommandLineFlags

-XX:+PrintGC 打印GC
-Xms40M 设置堆初始大小40M
-Xmx40M 设置堆最大大小40M
-Xmn20M 设置堆中 new Generation 新生代的大小为20M
-XX:+PrintGCDetails 输出GC的详细日志
-XX:+UseSerialGC 设置GC回收器模式是串型垃圾回收器
-XX:+PrintCommandLineFlags  配置的参数所改变的信息也打印输出

Java代码:

1
2
3
4
5
6
7
8
9
10
/**
* -XX:+PrintGC -Xms40M -Xmx40M -Xmn20M -XX:+PrintGCDetails -XX:+UseSerialGC -XX:+PrintCommandLineFlags
* @param args
*/
public static void main(String[] args) {
byte[] b1 = new byte[4 * 1024 * 1024];
byte[] b2 = new byte[4 * 1024 * 1024];
byte[] b3 = new byte[4 * 1024 * 1024];
byte[] b4 = new byte[8 * 1024 * 1024];
}

分析:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

初始分配:
Java堆,共有 40M
新生代,有20M
Eden区,16M
Servivor From ,2M
Servior To ,2M
老年代,有20M
当b1/b2/b3执行完时
Eden 被分配12M
当a4执行时
因为需要分配8M,但现在Eden中只有4M空闲,无法满足b4申请;
from和to中也只有2M,也不能满足,所以要触发MinorGC;
MinorGC执行,处理b1/b2/b3的空间,他们都是4M,所以servivor也不够转移,直接回被移到老年代;
MinorGC执行后,分配如下:
新生代,total 20M
Eden区 free 16M
Servivor From free 2M
Servior To free 2M
老年代,total 20M free 8M
再次分配b4
新生代,total 20M
Eden区 free 8M
Servivor From free 2M
Servior To free 2M
老年代,total 20M free 8M

运行结果分析: 配置项大小结果

1
2
3
4
5
6
7
8
9
10
11
12
13
14
-XX:InitialHeapSize=41943040 -XX:MaxHeapSize=41943040 -XX:MaxNewSize=20971520 -XX:NewSize=20971520 -XX:+PrintCommandLineFlags -XX:+PrintGC -XX:+PrintGCDetails -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseSerialGC
[GC (Allocation Failure) [DefNew: 13599K->553K(18432K), 0.0109283 secs] 13599K->12841K(38912K), 0.0118473 secs] [Times: user=0.01 sys=0.00, real=0.01 secs]
Heap
新生代
def new generation total 18432K, used 8909K [0x00000000fd800000, 0x00000000fec00000, 0x00000000fec00000)
eden space 16384K, 51% used [0x00000000fd800000, 0x00000000fe0290e0, 0x00000000fe800000)
from space 2048K, 27% used [0x00000000fea00000, 0x00000000fea8a558, 0x00000000fec00000)
to space 2048K, 0% used [0x00000000fe800000, 0x00000000fe800000, 0x00000000fea00000)
老年代
tenured generation total 20480K, used 12288K [0x00000000fec00000, 0x0000000100000000, 0x0000000100000000)
the space 20480K, 60% used [0x00000000fec00000, 0x00000000ff800030, 0x00000000ff800200, 0x0000000100000000)
元空间
Metaspace used 2749K, capacity 4486K, committed 4864K, reserved 1056768K
class space used 293K, capacity 386K, committed 512K, reserved 1048576K

运行结果基本符合推导结果。

6、-Xmn

设置新生代大小

7、-XX:NewRatio 

新生代(eden+2*s)和老年代(不包含永久区)的比值,默认2
例如:4,表示新生代:老年代=1:4,即新生代占整个堆的1/5

8、-XX:SurvivorRatio 设置两个Survivor区和eden的比值,默认8

 例如:8,表示两个Survivor:eden=2:8,即一个Survivor占年轻代的1/10

9、-XX:+HeapDumpOnOutOfMemoryError

OOM(Out Of Memory内存溢出)时导出堆到文件,根据这个文件,我们可以看到系统dump时发生了什么。
内存分析工具:Memory Anakyaer,eclipse插件
-XX:+HeapDumpPath        导出OOM的路径
例如:XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=d:/a.dump
我们可以用VisualVM打开这个dump文件。
注:关于VisualVM的使用,可以参考下面这篇博客:
使用 VisualVM 进行性能分析及调优:http://www.ibm.com/developerworks/cn/java/j-lo-visualvm/
或者使用Java bin目录里自带的Java VisualVM工具也行

10、-XX:OnOutOfMemoryError

在OOM时,执行一个脚本。可以在OOM时,发送邮件,甚至是重启程序。
例:-XX:OnOutOfMemoryError=D:/..../jdk/bin/printstack.bat %p    p代表的是当前进程的pid
上方参数的意思是说,执行printstack.bat脚本,而这个脚本做的事情是:D:/..../jdk/bin/jstack -F %1 > D:/a.txt,即当程序OOM时,在D:/a.txt中将会生成线程的dump。

11、-XX:PermSize -XX:MaxPermSize

设置永久区的初始空间和最大空间。也就是说,jvm启动时,永久区一开始就占用了PermSize大小的空间,如果空间还不够,可以继续扩展,但是不能超过MaxPermSize,否则会OOM。
jdk8开始,已经移除永久代。

12、

-XX:MaxTenuringThreshold 指定新生代经过多少次回收后进入老年代,默认15
-XX:PretenureSizeThreshold 设置对象大小超过指定大小后,直接进入老年代。注意线程TLAB区优先分配空间问题。
TLAB(Thread Local Allocation Buffer)线程本地分配缓存,是为了加速对象 分配而生的。线程私有。
JVM使用这个区来避免多线程冲突问题,提高对象分配的效率,TLAB空间不会太大,当大对象无法再TLAB上分配时,才会直接分配到堆上。
-XX:+UseTLAB 使用TLAB
-XX:+TLABSize 设置大小
-XX:TLABRefillWasteFraction 设置维护进入TLAB空间的单个对象大小,他是一个比值,默认64,如果对象DAU整个空间的1/64则在堆上创建对象。
-XX:+PrintTLAB 查看TLAB信息
-XX:ResizeTLAB 自调整TLABRefillWasteFraction的阈值。

总结:实际中,根基实际情况调整新生代和幸存代的大小;在OOM时,记得把dump文件导出,可以促进排查问题。

栈的分配参数

1、-Xss 指定线程的最大栈空间,决定了函数可以调用的最大深度。

  
每个线程都有独立的栈空间;
局部变量、参数 分配在栈上;
栈空间是每个线程私有的区域。栈里面的主要内容是栈帧,而栈帧存放的是局部变量表,局部变量表的内容是:局部变量、参数。

示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
 public class Demo2 {
private static int count = 0;
public static void recursion() {
long l1 = 1;
count++;
recursion();
}
public static void main(String[] args) {
try {
recursion();
} catch (Throwable e) {
System.out.println(count);
e.printStackTrace();
}
}
}

这段代码是没有出口的递归,肯定会出现OOM.

如果设置栈大小为:-Xss1m,方法调用26037了次

26037
java.lang.StackOverflowError
    at com.hu.demo.Demo2.recursion(Demo2.java:9)

如果设置栈大小为:-Xss2m,方法调用了48156次
48156
java.lang.StackOverflowError
    at com.hu.demo.Demo2.recursion(Demo2.java:8)

所以:-Xss设置决定函数调用的深度。

方法区参数

和堆一样,方法区是一块所有线程共享的内存,用于保存类的信息,可以保存多少信息,可以通过配置:

-XX:MaxPermSize ,默认64M,实际中,如果系统产生大量的类,就需要设置这个合适的参数,一面出现OOM。
-XX:sPermSize

直接内存配置

特别是在NIO编程中,直接内存跳过了java堆,使其java直接访问内存空间。

-XX:MaxDirechMemorySize,如果不设置,默认为最大堆空间-Xmx,直接内存到上限时就会触发垃圾回收。

垃圾回收的JVM配置

运行的垃圾回收器类型

配置

-XX:+UseSerialGC        串行垃圾回收器
-XX:+UseParallelGC        并行垃圾回收器
-XX:+UseConcMarkSweepGC        并发标记扫描垃圾回收器
-XX:ParallelCMSThreads=        并发标记扫描垃圾回收器 =为使用的线程数量
-XX:+UseG1GC        G1垃圾回收器

GC的优化配置

配置

-Xms        初始化堆内存大小
-Xmx        堆内存最大值
-Xmn        新生代大小
-XX:PermSize        初始化永久代大小
-XX:MaxPermSize        永久代最大容量

并行收集器设置

-XX:ParallelGCThreads=n:设置并行收集器收集时使用的CPU数。并行收集线程数。
-XX:MaxGCPauseMillis=n:设置并行收集最大暂停时间
-XX:GCTimeRatio=n:设置垃圾回收时间占程序运行时间的百分比。公式为1/(1+n)

并发收集器设置

-XX:+CMSIncrementalMode:设置为增量模式。适用于单CPU情况。
-XX:ParallelGCThreads=n:设置并发收集器年轻代收集方式为并行收集时,使用的CPU数。并行收集线程数。

JDK8相关参数

1、由于jdk8开始,彻底去除永久区,所以永久区的参数就不用了。

-XX:PermSize
-XX:MaxPermSize

2、metaspace,元数据空间,专门用来存元数据的,jdk8里特有的数据结构用来替代perm

(1)CompressedClassSpaceSize参数作用是设置Klass Metaspace的大小,默认1G

Klass Metaspace就是用来存klass的,klass是的class文件在jvm里的运行时数据结构,没有开启压缩指针,就不会有CompressedClassSpaceSize这块内存,但是jdk1.8里应该是默认开启的,并且,如果这块内存会如果没有满会一直增加。

但是-Xmx超过了32G,压缩指针是默认不开启的,而这个参数也就失去了设置的意义。

(2)MaxMetaspaceSize

   默认基本是无穷大,这个参数很可能会因为没有限制而导致metaspace被无止境使用(一般是内存泄漏)而被OS Kill。这个参数会限制metaspace(包括了Klass Metaspace以及NoKlass Metaspace)被committed的内存大小,会保证committed的内存不会超过这个值,一旦超过就会触发GC,这里要注意和MaxPermSize的区别,MaxMetaspaceSize并不会在jvm启动的时候分配一块这么大的内存出来,而MaxPermSize是会分配一块这么大的内存的。

3.MaxDirectMemorySize

此参数主要影响的是非堆内存的direct byte buffer,jvm默认会设置64M,可根据功能适当加大此项参数,因为非堆内存,故而不会被GC回收掉,容易出现java.lang.OutOfMemoryError: Direct buffer memory错误
如出现以上错误,可通过以下参数打印log,之后用工具进行分析

-XX:-HeapDumpOnOutOfMemoryError
-XX:HeapDumpPath=logs/oom_dump.log

4、G1收集器参数

-XX:+UseG1GC
使用G1收集器
-XX:MaxGCPauseMillis=200
用户设定的最大gc 停顿时间,默认是200ms.
-XX:InitiatingHeapOccupancyPercent=45
默认是45,也就是heap中45%的容量被使用,则会触发concurrent gc

-XX:NewRatio=n
新生代与老生代(new/old generation)的大小比例(Ratio). 默认值为 2.
-XX:SurvivorRatio=n    eden/survivor
空间大小的比例(Ratio). 默认值为 8.
-XX:MaxTenuringThreshold=n
提升年老代的最大临界值(tenuring threshold). 默认值为 15.
-XX:ParallelGCThreads=n
设置垃圾收集器在并行阶段使用的线程数,默认值随JVM运行的平台不同而不同.
-XX:ConcGCThreads=n
并发垃圾收集器使用的线程数量. 默认值随JVM运行的平台不同而不同.
-XX:G1ReservePercent=n
设置堆内存保留为假天花板的总量,以降低提升失败的可能性. 默认值是 10.
-XX:G1HeapRegionSize=n
使用G1时Java堆会被分为大小统一的的区(region)。此参数可以指定每个heap区的大小. 默认值将根据 heap size 算出最优解. 最小值为 1Mb, 最大值为 32Mb.

目前jvm支持Client和Server两种运行模式,

使用参数-client可以指定使用client模式。

使用参数-server启动server模式。

查看当前jvm使用的模式java -version

区别:client比server启动较快,server启动较慢,如果是要对其进行复杂的系统性能信息收集和使用更复杂的算法对程序进行优化,一般都会启动server模式,长期运行的性能远远快于client模式。

上一页1…131415…25下一页
初晨

初晨

永远不要说你知道本质,更别说真相了。

249 日志
46 分类
109 标签
近期文章
  • WebSocket、Socket、TCP、HTTP区别
  • Springboot项目的接口防刷
  • 深入理解Volatile关键字及其实现原理
  • 使用vscode搭建个人笔记环境
  • HBase介绍安装与操作
© 2018 — 2020 Copyright
由 Hexo 强力驱动
|
主题 — NexT.Gemini v5.1.4