随机端口
我们知道为spring boot配置随机端口非常简单,有两种办法:
1. 设置端口为0
server.port=0
eureka.instance.instance-id=${spring.application.name}-${spring.cloud.client.ipAddress}:${random.int}
2. 设置端口为随机值
`server.port=${random.int[1000,1999]}`
问题
我们无论使用上述的哪种办法,都会存在一些问题,有且不限于如下:
1. eureka界面上的Status无法显示真实的端口
2. 点击Status列的链接会跳转到错误的端口
3. 如果使用了LoadBalancerInterceptor
会导致连接异常,原因也是导向了错误的端口
4. …
解决思路
不难发现问题的根本就是每个地方都会再调用一次random.int生成不同的随机数,如果新生成的随机数被赋予端口意义使用,那么会因为这个数和真实端口并不一致导致发生异常。
random.int 是springboot默认提供的,我想实现一个自己的random.int,只赋值一次即可,往后再调用便返回已生成的值。
解决方法
random.int 是springboot默认提供的,
Ctrl+Click
发现实现类是RandomValuePropertySource
.我尝试把
server.port=${random.int[1000,2000]}
改成server.port=${randomServerPort.value[1000,2000]}
(${randomServerPort.value[1000,2000]}是我们要自定义的),运行后得到异常的堆栈信息Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder 'randomServerPort.value[1000,2000]' in value "${randomServerPort.value[1000,2000]}" at org.springframework.util.PropertyPlaceholderHelper.parseStringValue(PropertyPlaceholderHelper.java:174) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.util.PropertyPlaceholderHelper.replacePlaceholders(PropertyPlaceholderHelper.java:126) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.AbstractPropertyResolver.doResolvePlaceholders(AbstractPropertyResolver.java:236) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.AbstractPropertyResolver.resolveRequiredPlaceholders(AbstractPropertyResolver.java:210) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.AbstractPropertyResolver.resolveNestedPlaceholders(AbstractPropertyResolver.java:227) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.PropertySourcesPropertyResolver.getProperty(PropertySourcesPropertyResolver.java:84) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.PropertySourcesPropertyResolver.getProperty(PropertySourcesPropertyResolver.java:66) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.core.env.AbstractEnvironment.getProperty(AbstractEnvironment.java:537) ~[spring-core-4.3.14.RELEASE.jar:4.3.14.RELEASE] at org.springframework.boot.bind.RelaxedPropertyResolver.getProperty(RelaxedPropertyResolver.java:84) ~[spring-boot-1.5.10.RELEASE.jar:1.5.10.RELEASE] at org.springframework.boot.bind.RelaxedPropertyResolver.getProperty(RelaxedPropertyResolver.java:74) ~[spring-boot-1.5.10.RELEASE.jar:1.5.10.RELEASE] at org.springframework.boot.actuate.autoconfigure.EndpointWebMvcAutoConfiguration$ManagementServerPort.getPortProperty(EndpointWebMvcAutoConfiguration.java:409) ~[spring-boot-actuator-1.5.10.RELEASE.jar:1.5.10.RELEASE] at org.springframework.boot.actuate.autoconfigure.EndpointWebMvcAutoConfiguration$ManagementServerPort.get(EndpointWebMvcAutoConfiguration.java:361) ~[spring-boot-actuator-1.5.10.RELEASE.jar:1.5.10.RELEASE] at org.springframework.boot.actuate.autoconfigure.EndpointWebMvcAutoConfiguration$OnManagementMvcCondition.getMatchOutcome(EndpointWebMvcAutoConfiguration.java:345) ~[spring-boot-actuator-1.5.10.RELEASE.jar:1.5.10.RELEASE] at org.springframework.boot.autoconfigure.condition.SpringBootCondition.matches(SpringBootCondition.java:47) ~[spring-boot-autoconfigure-1.5.10.RELEASE.jar:1.5.10.RELEASE] ... 16 more
从堆栈信息中找到解析的类
PropertyPlaceholderHelper
,打上断点
跳转到
PropertySourcesPropertyResolver
,找到了真正的解析器propertySources
我们只要自己写一个
PropertySource
,然后放进去就可以了。照着RandomValuePropertySource
写一个自己的import lombok.extern.slf4j.Slf4j; import org.springframework.core.env.PropertySource; import org.springframework.util.StringUtils; @Slf4j public class RandomServerPortPropertySource extends PropertySource<RandomServerPort> { /** * Name of the random {@link PropertySource}. */ public static final String RANDOM_SERVER_PORT_PROPERTY_SOURCE_NAME = "randomServerPort"; private static final String PREFIX = "randomServerPort."; public RandomServerPortPropertySource(String name) { super(name, new RandomServerPort()); } public RandomServerPortPropertySource() { this(RANDOM_SERVER_PORT_PROPERTY_SOURCE_NAME); } public RandomServerPortPropertySource(String name, RandomServerPort source) { super(name, source); } @Override public Object getProperty(String name) { if (!name.startsWith(PREFIX)) { return null; } if (log.isTraceEnabled()) { log.trace("Generating randomServerPort property for '" + name + "'"); } return getRandomServerPortValue(name.substring(PREFIX.length())); } private Object getRandomServerPortValue(String type) { String range = getRange(type, "value"); if (range != null) { return getNextValueInRange(range); } return null; } private String getRange(String type, String prefix) { if (type.startsWith(prefix)) { int startIndex = prefix.length() + 1; if (type.length() > startIndex) { return type.substring(startIndex, type.length() - 1); } } return null; } private int getNextValueInRange(String range) { String[] tokens = StringUtils.commaDelimitedListToStringArray(range); int start = Integer.parseInt(tokens[0]); if (tokens.length == 1) { return getSource().nextValue(start); } return getSource().nextValue(start, Integer.parseInt(tokens[1])); } }
RandomServerPort:
import org.apache.commons.lang3.RandomUtils;
/**
* 随机生成一个端口,并只生成一次
*/
public class RandomServerPort {
private int serverPort;
private final int start = 0;
private final int end = 65535;
public int nextValue(int start) {
return nextValue(start, end);
}
public int nextValue(int start, int end) {
start = start < this.start? this.start: start;
end = end > this.end? this.end: end;
if (serverPort == 0){
synchronized (this){
if (serverPort == 0){
serverPort = RandomUtils.nextInt(start, end);
}
}
}
return serverPort;
}
}
6. 这种解析器是解析application.properties
的,所以需要很早就加载上去,不难想到用ApplicationListener<ApplicationEnvironmentPreparedEvent>
import org.springframework.boot.context.event.ApplicationEnvironmentPreparedEvent;
import org.springframework.context.ApplicationListener;
public class MyApplicationEnvironmentPreparedEventListener implements ApplicationListener<ApplicationEnvironmentPreparedEvent> {
@Override
public void onApplicationEvent(ApplicationEnvironmentPreparedEvent event) {
event.getEnvironment().getPropertySources().addLast(new RandomServerPortPropertySource());
}
}
7. 添加监听器
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication app = new SpringApplication(Application.class);
app.addListeners(new MyApplicationEnvironmentPreparedEventListener());
app.run(args);
}
}
8. 配置文件修改一下
server.port=${randomServerPort.value[1000,2000]}
eureka.instance.instance-id=${spring.application.name}-${spring.cloud.client.ipAddress}:${randomServerPort.value[1000,2000]}
# 省略...
9. 结果如下
点击跳转