Spring Boot 2中的功能切换

无论您是否喜欢,软件开发都是一项协作活动。 整合工作一直被妖魔化,并被视为必不可少的邪恶。 有几种方法可以解决有效集成的挑战。 功能切换开关属于该组。 在本文中,您将在实践中看到如何在Spring Boot应用程序中使用功能切换(也称为功能标志)。

1.什么是功能切换?

简而言之, 功能切换是允许根据其当前值在应用程序中执行替代路径的变量。 通过保持不同的执行方案,您可以在不更改代码的情况下修改应用程序的行为。

根据您的需求,可以在启动应用程序之前设置切换开关的值,也可以在运行时对其进行调整。 在后一种情况下,值的更改可以保留或仅影响应用程序的当前执行。

通常,您会读到有关功能标志的信息, 以作为功能源代码分支的替代方法 ,但是,实际上,两种技术可以一起使用。 例如,您可以使用功能分支在应用程序中开发新的用户故事,同时可以使用功能切换来控制对不同环境(例如,具有不同要求的客户端)上功能的访问。

尽管有许多用途,功能切换也有其缺点。 最大的问题是复杂性 。 如果没有适当的策略,他们可能会Swift失控,成为维护的噩梦。 幸运的是,如果您遵循几种良好的做法并围绕features来组织应用程序 ,则使用Feature标志应该更加简单。

2.使用功能切换选择豆

在Spring Boot应用程序中使用功能切换的最常见情况是基于功能切换的当前值激活某些接口的不同实现。 我们来看一个示例来说明所描述的情况。

2.1依赖抽象

假设您有一个Web端点,该端点返回从数据库存储库中获取的产品列表。 您的目标是创建一个功能切换,该功能切换允许将存储库实现切换为使用Web服务作为数据源的实现。

如果要允许要素切换的类直接在其他类中使用,则您要做的第一件事是使用接口抽象依赖关系。

下面的代码片段提供了一个示例Product REST端点,该端点依赖于ProductRepository接口。

@RestController
@RequestMapping("/products")
class ProductController {private final ProductRepository productRepository;ProductController(ProductRepository productRepository) {this.productRepository = productRepository;}@GetMappingCollection<Product> getAll() {return productRepository.findAll();}}

目前,我们只有一个接口实现。 不久,我们将添加另一个,您将通过功能切换激活它。

@Repository
class DbProductRepository implements ProductRepository {//...
}

2.2 application.properties中的功能切换

由于application.properties文件用于配置Spring Boot应用程序,因此是放置功能切换标志的好地方。

feature.toggles.productsFromWebService=true

在提交代码之前,将标志设置为false。 这样,默认情况下,您的队友将禁用新功能。 如果有人要激活该功能,则他们可以在本地开发环境中将标志值更改为true。

2.3有条件的Bean创建

下一步是创建要通过功能切换激活的接口的替代实现。 为了根据创建的属性的值实例化bean,可以使用Spring Boot注释@ConditionalOnProperty 。 设置切换属性的名称和应激活它的值。 该值应与放在application.properties文件中的值相同。

@Repository
@ConditionalOnProperty(name = "feature.toggles.productsFromWebService",havingValue = "true"
)
class WebServiceProductRepository implements ProductRepository {//...
}

在启动应用程序之前,必须禁用数据库存储库,否则,您将获得有关接口的多个活动实现的异常。 返回第一个实现并应用以下更改:

@Repository
@ConditionalOnProperty(name = "feature.toggles.productsFromWebService",havingValue = "false",matchIfMissing = true
)
class DbProductRepository implements ProductRepository {

我们使用与以前相同的功能切换名称,只是其值已更改。 设置matchIfMissing属性是可选的。 这样,如果您从application.properties文件中删除功能切换,即使缺少该值,也将创建该bean。

3.如何通过功能切换禁用控制器

您可以应用相同的策略有条件地激活整个Spring Web控制器。 您不需要创建其他接口,因为您只想通过功能切换控制一个实现。

@RestController
@RequestMapping("/coupons")
@ConditionalOnProperty(name = "feature.toggles.coupons", havingValue = "true")
class CouponController {//...
}

application.properties应该包含以下行。

feature.toggles.coupons=true

当您不将值设置为true时,Spring不会实例化控制器。 客户端将仅收到404 HTTP状态代码。

不幸的是, @ ConditionalOnProperty批注不能在单个@RequestMapping方法上使用。 解决方法是,可以将所需的映射移动到单独的控制器Bean。 或者,可以简单地插入功能切换的值并在映射方法的主体中创建if语句。 但是,您应谨慎使用此解决方案。 如果您有兴趣,为什么在下一段中找到答案。

private final boolean couponsToggled;CouponController(@Value("${feature.toggles.coupons}") boolean couponsToggled) {this.couponsToggled = couponsToggled;
}@GetMapping
List<String> listCouponNames() {if (!couponsToggled) {throw new NotSupportedException();}//...
}

4.多功能切换管理

正如您可以阅读Martin Fowler的bliki上的功能切换一样 , 功能标志倾向于在整个代码库中扩展,并且很快就会变得难以管理 。 即使您的应用程序中只有几个功能切换,最好还是从使用标记的决策点抽象出标记的存储。

4.1避免特征标记耦合

上一段中的最后一个代码示例使用直接从application.properties文件注入的标志值,因此不会抽象存储。 如果要在应用程序的不同部分中使用相同的标志,则必须重复注入。

相反,您可以做的是将所有功能切换值放在一个类中,这将作为单个true来源 。 使用单独的类可为您提供更大的灵活性。 例如,您可以用数据库替换标志的存储,或者实现一种允许在运行时切换标志的机制。

4.2在Spring Boot中提取功能切换决策

一旦具有用于功能切换的单独的bean,就可以使用@ConfigurationProperties批注轻松地从application.properties文件中注入所有标志。 在这里,您可以看到一个示例实现:

@Component
@Component
@ConfigurationProperties("feature")
public class FeatureDecisions {private Map<String, Boolean> toggles = new HashMap<>();public Map<String, Boolean> getToggles() {return toggles;}public boolean couponEnabled() {return toggles.getOrDefault("coupons", false);}}

上面的类将获取所有以feature.toggles开头的属性,并将它们放入切换图。 如您所见,有一个名为couponEnabled()的方法,可用于从决策背后的逻辑中提取功能决策点。

此外,您还需要一个额外的依赖项才能启用对@ConfigurationProperties的处理。

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-configuration-processor</artifactId>
</dependency>

5.用于功能切换的执行器端点

由于您已经将所有功能切换都放在一个位置,因此现在要做的就是使用自定义的Actuator端点公开列表。 以下示例将向您展示如何进行。

@Component
@Endpoint(id = "feature-toggles")
class FeatureToggleInfoEndpoint {private final FeatureDecisions featureDecisions;FeatureToggleInfoEndpoint(FeatureDecisions featureDecisions) {this.featureDecisions = featureDecisions;}@ReadOperationpublic Map<String, Boolean> featureToggles() {return featureDecisions.getToggles();}}

如果您使用默认的Spring Boot 2 Actuator设置,则不会通过HTTP公开端点 。 为了在浏览器中对其进行测试,您必须通过将其标识符添加到application.properties文件中的Web include过滤器来启用Actuator端点。

management.endpoints.web.exposure.include=health,info,feature-toggles

运行应用程序后,请转到http:// localhost:8080 / actuator / feature-toggles查看端点返回的结果:

根据您的需求,您还可以在创建的端点上使用@WriteOperation实现在运行时切换功能切换的可能性。 本示例仅涵盖输出部分。

结论

在本文中,您可以了解Spring Boot应用程序中功能切换的实际示例。 我们从一个非常基本的示例开始,该示例涵盖了框架的所有需求。 之后,我们编写一些自定义代码来完成更多的自定义功能切换要求。 我们完成了有用的Actuator端点,以显示应用程序中所有功能标志的状态。

您可以在Github存储库中找到工作示例应用程序 。 如果您喜欢该帖子并认为它有用,请与您的关注者分享。 我也期待您在本文下面的问题和评论。

翻译自: https://www.javacodegeeks.com/2018/03/feature-toggle-in-spring-boot-2.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/348186.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java学习 第三章 数组(三)排序算法

** Java学习 第三章 数组&#xff08;三&#xff09;排序算法 ** 主要内容&#xff1a;排序算法、排序算法横向比较、Arrays工具类的使用、数组常见异常 1.数组中涉及到的常见算法&#xff1a;排序算法 1.1 排序算法分类&#xff1a;内部排序和外部排序 1.2 十大内部排序算…

【强化学习】一些网站整理

莫烦教程 https://mofanpy.com/tutorials/machine-learning/reinforcement-learning/ 博客园&#xff1a;刘建平Pinard https://www.cnblogs.com/pinard/category/1254674.html Deep-Q-Network 学习笔记 https://www.cnblogs.com/cjnmy36723/p/7017549.html 强化学习到深度强…

xmx java_为什么我的Java进程比Xmx消耗更多的内存?

xmx java你们有些人去过那里。 您已经在启动脚本中添加了-Xmx选项&#xff0c;并放松了下来&#xff0c;因为您知道Java进程将不会消耗比经过微调的选项所允许的更多的内存。 然后&#xff0c;您感到非常讨厌。 要么自己检查开发/测试框中的过程表&#xff0c;要么事情真的变坏…

卫星通信系统概述

卫星通信系统指通过在轨人造卫星作为中继站对无线电信号进行转发&#xff0c;实现地面及空间等用户之间信息传输的系统。卫星通信系统组成包括空间段及地面段&#xff0c;系统组成如图所示。其中空间段主要指在轨卫星、对在轨卫星进行操控的地面站&#xff0c;这些地面站主要实…

ElasticSearch初学者教程

1.简介 在此示例中&#xff0c;我们将演示如何使用Elasticsearch &#xff0c; Elasticsearch是一个基于Apache Lucene的分布式自由文本搜索和分析数据库引擎&#xff0c;具有一个基于maven的简单Java客户端。 在撰写本文时&#xff0c;我们将使用最新版本的Elasticsearch&…

博弈论与纳什均衡

三十分钟理解博弈论“纳什均衡” – Nash Equilibrium https://blog.csdn.net/xbinworld/article/details/50932559 纳什均衡(Nash equilibrium)及经典案例 https://blog.csdn.net/u010420283/article/details/83927742 论文&#xff1a; [1] 刘帅军. 卫星通信系统中动态资源…

Opnet入门

一、opnet快速入门 1.系统界面&文件菜单说明 2.常用文件名后缀及描述 3.Opnet建模层次 用户只有一种节点域模型 三、 OPNET Modeler网络仿真机制 1.事件的属性 每次点击next会出现以下界面&#xff1a; 2.事件的执行 调度型&#xff1a;按照正常程序调度事件 强制性&…

JSON的JUnit Hamcrest Matcher

这篇文章展示了如何编写JUnit测试来检查对象是否与JSON字符串匹配。 如果您要实现REST服务并想测试您的服务是否产生了预期的JSON响应&#xff0c;那么这一点很重要。 JSONassert是比较JSON对象的有用库。 首先&#xff0c;您必须将Java对象转换为JSON字符串&#xff08;例如&…

Python列表推导式

列表推导式 是Python构建列表&#xff08;list&#xff09;的一种快捷方式,可以使用简洁的代码就创建出一个列表&#xff0c;即循环创建列表. for可以用来创建列表&#xff0c;列表推导式就相当于是for循环的简化版 1. 最简单的情况 values [10, 21, 5, 7, 12] squares [] …

一些python函数及其用法

1.np.ravel&#xff08;&#xff09;方法 ravel是将数组维度拉成一维数组&#xff0c;也就是将矩阵向量化 x np.array{ [ [ 1 , 2 , 3 ] , [ 4 , 5 , 6 ] ] } print(np.ravel(x))输出 [ 1 2 3 4 5 6 ]2.b a[np.newaxis,:] import numpy as np a np.arange(0, 10) print(a…

使用Spring WebFlux进行操作

Spring Boot 2.0最近在GA上线了&#xff0c;所以我决定在相当长一段时间内写我的第一篇有关Spring的文章。 自发布以来&#xff0c;我已经看到越来越多的提到Spring WebFlux以​​及有关如何使用它的教程。 但是&#xff0c;在阅读完它们并尝试使它们自己工作之后&#xff0c;我…

【强化学习】Policy Gradient原理

1.Policy Gradient和DQN系列强化算法最大的区别在于&#xff1a; DQN系列基于Value&#xff0c;也就是说执行完所有的动作并保存所得到的价值&#xff0c;根据这些价值计算出最优价值函数&#xff0c;并以此选择动作&#xff0c;最终获得一个特定的策略。 Policy Gradient基于策…

ajax的url怎么将后缀补上_蜂蜜杏仁怎么做?杏仁和蜂蜜腌制方法

蜂蜜杏仁怎么做?杏仁和蜂蜜腌制方法蜂蜜杏仁是一种非常好吃的小零食&#xff0c;很多小可爱都喜欢吃蜂蜜杏仁。不过有些时候忘记补货&#xff0c;就会断粮&#xff0c;于是大家都想要在家里自制蜂蜜杏仁。不过你知道蜂蜜杏仁应该怎么做吗?蜂蜜杏仁的做法其实并不难&#xff0…

【强化学习】Policy Gradients代码注释版本

import gym # import tensorflow as tf import numpy as np# Hyper Parameters GAMMA 0.95 # discount factor 折扣因子 LEARNING_RATE 0.01 # 学习率class Policy_Gradient():# 咱们来搞一下大头&#xff01;def __init__(self, env): # 初始化# 先初始化一些参量sel…

c语言构建栈_选择技术栈构建通用平台

c语言构建栈Java社区中有许多关于Spring vs Java EE的话题。 一群人会争辩说您应该使用一个而不是其他。等等。当我看到这一点时&#xff0c;我不禁要想为什么我们不能同时使用它们呢&#xff1f; 实际上&#xff0c;我认为有效地使用它们将为建立基础架构创建一个伟大的技术堆…

【强化学习】Actor Critic原理

PG算法是一种只基于policy的一种方法&#xff0c;存在的问题就是该算法需要完整的状态序列&#xff0c;且单独对策略函数进行迭代更新&#xff0c;不太容易收敛。 Actor-critic方法呢是一种将 策略(Policy Based)和价值(Value Based)相结合的方法。下面继续来理一下AC方法的思路…

Java 9:对Process API的增强

Java 9对Process API进行了各种改进&#xff0c;用于控制和管理操作系统进程。 获取有关流程的信息 有一个新的ProcessHandle类&#xff0c;提供了进程的pid&#xff0c;父级和后代&#xff0c;以及有关开始时间和累积CPU时间的信息。 jshell> Process p new ProcessBui…

【强化学习】AC注释版本

## 强化学习 Actor-critic # 和PG比起来主要的变化&#xff1a; # 评估点由状态价值变成了TD_error,网络形式变了 # learn函数长得不一样 # action有一个优化函数&#xff0c;优化的是价值函数&#xff0c;希望最大化期望的reward&#xff0c;Critic网络也有一个reward&#xf…

python中sorted的用法append_Python中高阶函数sorted()用法

在Python中&#xff0c;有内置的排序方法&#xff1a;sorted(iterable, key, reverse)。Sorted()函数也是一个高阶函数&#xff0c;它还可以接收一个key函数来实现自定义的排序。key指定的函数将作用于list的每一个元素上&#xff0c;并根据key函数返回的结果进行排序。print(s…

【强化学习】A3C原理

先解释一下什么叫异步、什么叫并发&#xff1a; **异步&#xff1a;**和同步相对&#xff0c;同步是顺序执行&#xff0c;而异步是彼此独立&#xff0c;在等待某个事件的过程中继续做自己的事&#xff0c;不要等待这一事件完成后再工作。线程是实现异步的一个方式&#xff0c;异…