springboot源码分析9-random的使用以及原理

简介: 摘要:springboot框架为我们提供了很多的便利,其中有一个非常有意思的功能,那就是可以通过变量的方式来配置一个随机数random,然后使用random随机出各式各样数值。

摘要:springboot框架为我们提供了很多的便利,其中有一个非常有意思的功能,那就是可以通过变量的方式来配置一个随机数random,然后使用random随机出各式各样数值。本位重点讲解一下random的使用以及框架内部的实现机制。

1.1. Springbootrandom的使用

首先我们定义一个配置类,如下所示:

1 @Component

2 public class Config {

3  @Value("${random.value}")

4  private String value;

5  @Value("${random.int}")

6  private int randomInt;

7  @Value("${random.long}")

8  private long randomLong;

9  @Value("${random.uuid}")

10  private String randomuuid;

11  @Value("${random.int(10)}")

12  private int randomInt1;

13  @Value("${random.int[1024,65536]}")

14  private int randomIntRange;

15 

16  @Override

17  public String toString() {

18  return "Config [value=" + value + ", randomInt=" + randomInt + ", randomLong=" + randomLong + ", randomuuid="

19  + randomuuid + ", randomInt1=" + randomInt1 + ", randomIntRange=" + randomIntRange + "]";

20  }

21 }

上述类中我们配置的与random相关的表达式有:

random.value返回值是string类型;random.int返回值是int类型;random.long返回值是long类型;random.uuid返回值是string类型;random.random.int(10)返回值是int类型;random.int[1024,65536]返回值是int类型。

紧接着,我们书写一个springboot启动类并打印Config类。示例代码如下:

1 @SpringBootApplication

2 public class DemoApplication {

3  public static void main(String[] args) {

4  BainuoSpringApplication springApplication = new BainuoSpringApplication(DemoApplication.class);

5  ConfigurableApplicationContext configurableApplicationContext = springApplication.run(args);

6  Config config = configurableApplicationContext.getBean(Config.class);

7  System.out.println(config.toString());

8  }

9 }

运行上述类,程序的输出如下:

Config [value=d4143e016f2f89527af9290cdc5faf8d, randomInt=-1397177543, randomLong=-5400484272385792469, randomuuid=113b3970-ac05-4a31-8a6c-1145ee55b9f9, randomInt1=2, randomIntRange=36897]

 

1.2. Springbootrandom内部实现机制

上述我们讲解了random的各种使用方式,心中不免有个疑惑。Springboot是如何处理这些random的呢?我们又该如何更好的使用呢?正所谓知其然,还要知其所以然。前面的系列文章中,我们详细讲解了环境属性的相关知识点,其中PropertySource类有一个子类RandomValuePropertySource,该类正是处理上文我们使用的random的。我们来快速学习下该类,示例代码如下:

1 public class RandomValuePropertySource extends PropertySource<Random> {

2  public static final String RANDOM_PROPERTY_SOURCE_NAME = "random";

3  private static final String PREFIX = "random.";

4  private static final Log logger = LogFactory.getLog(RandomValuePropertySource.class);

5  public RandomValuePropertySource(String name) {

6  super(name, new Random());

7  }

8  public RandomValuePropertySource() {

9  this(RANDOM_PROPERTY_SOURCE_NAME);

10  }

11  @Override

12 

13  private Object getRandomValue(String type) {

14  if (type.equals("int")) {

15  return getSource().nextInt();

16  }

17  if (type.equals("long")) {

18  return getSource().nextLong();

19  }

20  String range = getRange(type, "int");

21  if (range != null) {

22  return getNextIntInRange(range);

23  }

24  range = getRange(type, "long");

25  if (range != null) {

26  return getNextLongInRange(range);

27  }

28  if (type.equals("uuid")) {

29  return UUID.randomUUID().toString();

30  }

31  return getRandomBytes();

32  }

33 

34  private String getRange(String type, String prefix) {

35  if (type.startsWith(prefix)) {

36  int startIndex = prefix.length() + 1;

37  if (type.length() > startIndex) {

38  return type.substring(startIndex, type.length() - 1);

39  }

40  }

41  return null;

42  }

43 

44  private int getNextIntInRange(String range) {

45  String[] tokens = StringUtils.commaDelimitedListToStringArray(range);

46  int start = Integer.parseInt(tokens[0]);

47  if (tokens.length == 1) {

48  return getSource().nextInt(start);

49  }

50  return start + getSource().nextInt(Integer.parseInt(tokens[1]) - start);

51  }

52 

53  private long getNextLongInRange(String range) {

54  String[] tokens = StringUtils.commaDelimitedListToStringArray(range);

55  if (tokens.length == 1) {

56  return Math.abs(getSource().nextLong() % Long.parseLong(tokens[0]));

57  }

58  long lowerBound = Long.parseLong(tokens[0]);

59  long upperBound = Long.parseLong(tokens[1]) - lowerBound;

60  return lowerBound + Math.abs(getSource().nextLong() % upperBound);

61  }

62 

63  private Object getRandomBytes() {

64  byte[] bytes = new byte[32];

65  getSource().nextBytes(bytes);

66  return DigestUtils.md5DigestAsHex(bytes);

67  }

68 

69  public static void addToEnvironment(ConfigurableEnvironment environment) {

70  environment.getPropertySources().addAfter(

71  StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME,

72  new RandomValuePropertySource(RANDOM_PROPERTY_SOURCE_NAME));

73  logger.trace("RandomValuePropertySource add to Environment");

74  }

75 

76 }

下面重点讲解一下getProperty方法,该方法的处理逻辑非常的简单,首先判断需要获取的变量是否是以random开头的,如果是则开始调用getRandomValue方法进行处理,并截取random.这个字符串。示例代码如下:

1  public Object getProperty(String name) {

2  if (!name.startsWith(PREFIX)) {

3  return null;

4  }

5  return getRandomValue(name.substring(PREFIX.length()));

6  }

比如上文的${random.value},在这里处理之后,传递给getRandomValue方法的参数值是value;同理${random.int}在这里处理之后,传递给getRandomValue方法的参数值是int。好了,继续看getRandomValue方法吧,该方法内部实现中,会调用类似getSource方法,该方法返回值正式Random实例对象。示例代码如下:

1 public RandomValuePropertySource(String name) {

2  super(name, new Random());

3  }

了解了这些之后,getRandomValue方法的处理逻辑就非常简单了,总结梳理如下:

1. 如果是int则直接调用调用new Random().nextInt()。

2. 如果是long则直接调用调用new Random().nextLong()。

3. 如果是int或者long范围的则首先通过getRange获取到区间,然后还是调用new Random()中的区间随机数生成方法。

4. 如果是uuid则直接调用调用UUID.randomUUID().toString()。

5. 如果没有识别出来,则直接调用getRandomBytes生成一个字符串。

RandomValuePropertySource类的处理非常的简单而且经典,只要用一定的java基础均可以看明白。大家下去一步步debug即可看到起内部处理逻辑。但是这个类是怎么注入到当前springboot运行环境的呢?相信很多人有这样的疑问?这个涉及到了Springboot中的监听器使用,我们后续文章再来展开说明。


欢迎关注我的微信公众号,第一时间获得博客更新提醒,以及更多成体系的Java相关原创技术干货。 
扫一扫下方二维码或者长按识别二维码,即可关注。
 


相关文章
|
1月前
|
XML Java 开发者
Spring Boot开箱即用可插拔实现过程演练与原理剖析
【11月更文挑战第20天】Spring Boot是一个基于Spring框架的项目,其设计目的是简化Spring应用的初始搭建以及开发过程。Spring Boot通过提供约定优于配置的理念,减少了大量的XML配置和手动设置,使得开发者能够更专注于业务逻辑的实现。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,为开发者提供一个全面的理解。
36 0
|
4天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
|
10天前
|
NoSQL Java Redis
Spring Boot 自动配置机制:从原理到自定义
Spring Boot 的自动配置机制通过 `spring.factories` 文件和 `@EnableAutoConfiguration` 注解,根据类路径中的依赖和条件注解自动配置所需的 Bean,大大简化了开发过程。本文深入探讨了自动配置的原理、条件化配置、自定义自动配置以及实际应用案例,帮助开发者更好地理解和利用这一强大特性。
55 14
|
1月前
|
Java Spring
SpringBoot自动装配的原理
在Spring Boot项目中,启动引导类通常使用`@SpringBootApplication`注解。该注解集成了`@SpringBootConfiguration`、`@ComponentScan`和`@EnableAutoConfiguration`三个注解,分别用于标记配置类、开启组件扫描和启用自动配置。
59 17
|
1月前
|
Java 容器
springboot自动配置原理
启动类@SpringbootApplication注解下,有三个关键注解 (1)@springbootConfiguration:表示启动类是一个自动配置类 (2)@CompontScan:扫描启动类所在包外的组件到容器中 (3)@EnableConfigutarion:最关键的一个注解,他拥有两个子注解,其中@AutoConfigurationpackageu会将启动类所在包下的所有组件到容器中,@Import会导入一个自动配置文件选择器,他会去加载META_INF目录下的spring.factories文件,这个文件中存放很大自动配置类的全类名,这些类会根据元注解的装配条件生效,生效
|
5月前
|
Java 应用服务中间件 开发者
Java面试题:解释Spring Boot的优势及其自动配置原理
Java面试题:解释Spring Boot的优势及其自动配置原理
130 0
|
2月前
|
Java Spring 容器
springboot @RequiredArgsConstructor @Lazy解决循环依赖的原理
【10月更文挑战第15天】在Spring Boot应用中,循环依赖是一个常见问题,当两个或多个Bean相互依赖时,会导致Spring容器陷入死循环。本文通过比较@RequiredArgsConstructor和@Lazy注解,探讨它们解决循环依赖的原理和优缺点。@RequiredArgsConstructor通过构造函数注入依赖,使代码更简洁;@Lazy则通过延迟Bean的初始化,打破创建顺序依赖。两者各有优势,需根据具体场景选择合适的方法。
121 4
|
3月前
|
Java 应用服务中间件 API
Vertx高并发理论原理以及对比SpringBoot
Vertx 是一个基于 Netty 的响应式工具包,不同于传统框架如 Spring,它的侵入性较小,甚至可在 Spring Boot 中使用。响应式编程(Reactive Programming)基于事件模式,通过事件流触发任务执行,其核心在于事件流 Stream。相比多线程异步,响应式编程能以更少线程完成更多任务,减少内存消耗与上下文切换开销,提高 CPU 利用率。Vertx 适用于高并发系统,如 IM 系统、高性能中间件及需要较少服务器支持大规模 WEB 应用的场景。随着 JDK 21 引入协程,未来 Tomcat 也将优化支持更高并发,降低响应式框架的必要性。
Vertx高并发理论原理以及对比SpringBoot
|
3月前
|
Java 开发者 数据格式
【Java笔记+踩坑】SpringBoot基础4——原理篇
bean的8种加载方式,自动配置原理、自定义starter开发、SpringBoot程序启动流程解析
【Java笔记+踩坑】SpringBoot基础4——原理篇
|
5月前
|
SQL Java 数据库连接
springboot~mybatis-pagehelper原理与使用
【7月更文挑战第15天】MyBatis-PageHelper是用于MyBatis的分页插件,基于MyBatis的拦截器机制实现。它通过在SQL执行前动态修改SQL语句添加LIMIT子句以支持分页。使用时需在`pom.xml`添加依赖并配置方言等参数。示例代码: PageHelper.startPage(2, 10); List&lt;User&gt; users = userMapper.getAllUsers(); PageInfo&lt;User&gt; pageInfo = new PageInfo&lt;&gt;(users); 这使得分页查询变得简单且能获取总记录数等信息。
133 2