- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已经实现了一个角色枚举,其中包含带有等级的角色定义。对于 getAssignableRoles()
部分,我的同事说这是滥用 Enum 或过度工程,并指出用这种形式无法读取它。
public enum Role {
USER("ROLE_user", 1),
CUSTOMER("ROLE_customer", 2),
PRODUCT_OWNER("ROLE_product_manager", 3) {
@Override
public List<Role> getAssignableRoles() {
return getLowerRankedRoles().stream().filter(e -> !e.equals(CUSTOMER_ADMIN)).collect(Collectors.toList());
}
},
CUSTOMER_ADMIN("ROLE_customer_admin", 3) {
@Override
public List<Role> getAssignableRoles() {
return getLowerRankedRoles().stream().filter(e -> !e.equals(PRODUCT_OWNER)).collect(Collectors.toList());
}
},
USER_MANAGER("ROLE_user_manager", 5),
ADMIN("ROLE_admin", 99);
private final String value;
private final int rank;
public String getValue() {
return value;
}
public int getRank() {
return rank;
}
Role(String value, int rank) {
this.value = value;
this.rank = rank;
}
public static Role findByAbbr(String abbr) {
return Arrays.stream(values()).filter(value -> value.value.equals(abbr)).findFirst().orElse(UNKNOWN);
}
public String getExactValue() {
return value.replaceFirst("^ROLE_", "");
}
// Each role has a distinct grant rank but for some roles even if they have same grant level their role assigning changes in the context.
public List<Role> getAssignableRoles() {
return getLowerRankedRoles();
}
protected List<Role> getLowerRankedRoles() {
return Arrays.stream(values()).filter(value -> value.rank <= rank).collect(Collectors.toList());
}
public static Predicate<String> isInRealm() {
return (String role) -> (Arrays.stream(values()).anyMatch(value -> value.value.equals(role)));
}
}
我想做的是,在客户端代码中能够像这样调用它
Role.findByAbbr(role).getAssignableRoles()
像这样使用枚举是一个好的实践或设计吗?我们可以从 Enum 中获得大部分动态调度功能吗?
最佳答案
这个问题实际上比乍一看更有趣。我在这里看到两个不同的点:
谈到第 (1) 点,如果一个值不可变,则称为常量,而 Java enum
正是这样:常量值集 ( see the official Java tutorial )。
第 (2) 点指的是 Java enum
的 values()
预定义方法的功能。事实上,使用其他策略实现 Role.findByAbbr(abbr)
会很困惑。
让我进一步详细说明这一点。如果您不希望客户端代码能够创建新角色,常见的策略是利用包友好的可见性范围。在包 com.example.role
中声明一个抽象类 Role
。
package com.example.role;
public abstract class Role {
private int rank;
private String value;
Role(String value, int rank) {
this.value = value;
this.rank = rank;
}
// Other methods...
}
请注意,构造函数上没有显式可见性修饰符(因此它是包友好的)。只有同一包内的类才能调用类Role
的构造函数,因此只能在同一包内继承***。
package com.example.role;
public class UserRole extends Role {
private static final UserRole instance = new UserRole();
private UserRole() {
super("ROLE_user", 1);
}
public UserRole getInstance() {
return instance;
}
}
我们正在尝试使用普通 Java 类来模拟您的 enum
,因此我们还使用单例模式来为每个角色提供一个实例。
一切都很简单,但是当涉及到 values()
预定义方法时,情况就发生了变化。
package com.example.role;
public final class Roles {
private static Collection<Role> values = Collections.unmodifiableList(Arrays.asList(
UserRole.getInstance(),
CustomerRole.getInstance(),
ProductOwnerRole.getInstance(),
CustomerAdminRole.getInstance(),
UserManagerRole.getInstance(),
AdminRole.getInstance()));
private Roles() {
}
public static Collection<Role> values() {
return values;
}
// Other methods...
}
每次添加新的角色
时,您都必须记住将其添加到值
列表中。太丑了。
客户端代码的界面仍然很漂亮,如下所示:
Collection<Role> assignables = Roles.getAssignableRolesFor(role);
总之,您对 enum
属性的利用看起来并没有那么糟糕。您的角色是常量,预定义的 values()
方法为您提供了开箱即用的强大功能。将所有排名和名称放在一个地方而不是分散在多个文件中也非常好。
*** 用户可能会在其客户端代码中创建一个同名的包并创建一个新的角色
,但这是一个不好的做法,我什至不会考虑。
关于java - 使用 Enum 作为功能齐全的对象(像本例一样使用动态调度)是否意味着滥用它或不是一个好的设计?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59586654/
我不明白为什么这甚至可以编译。我尝试过不同的格式,它们似乎都有效。 为什么有一个 enum of enum of enum of.. 是合法的? interface I { enum E im
我有大型数据库(数百万行),我正在尝试为 2 个字段的数据类型做出最佳选择。我制作的大部分内容都是 varchar 或 INT。但是,我想知道 Enum 是否是最好的方法的 2 个字段。 字段 1第一
我是 C++ 新手,我想知道是否有人可以帮助我理解原因 enum difficulty { NOVICE, EASY, NORMAL, HARD, UNBEATABLE }; difficulty m
从 native 代码强制转换(在 C++/CLI 中)的正确方法是什么 enum到托管代码enum其中包含相同的 enum值(value)观?与使用 C# 的强制转换方式有什么区别,例如 (int)
我有以下枚举 [Flags] public enum WeekDays { Monday = 1, Tuesday = 2, Wednesday = 4, Thursd
我刚刚学习 Java 中的枚举。当我运行下面的代码时,我收到一个错误,我也在下面重现该错误。基本上,我的问题是:当我在枚举中定义一个方法,并且在该方法中我想检查枚举的值以便我可以根据该值执行某些操作时
我想要一些语法糖来打开 Enum .当然,一个if else块按预期工作: @enum Fruit apple=1 orange=2 kiwi=3 function talk1(fruit::Frui
我试图在 Enum.each 的函数内为变量设置一个值,但在循环结束时,变量为空,我不知道为什么会出现这种行为。 代码: base = "master" candidates = ["stream",
This question already has answers here: Is there a way to use existing structs as enum variants? (2个
我想让 Java 中的枚举器具有其他枚举作为属性。 public enum Direction { Up(Down), Down(Up), Left(Right), R
我有一个包含两种不同可能“类型”的枚举和一个可以返回其中任何一种的函数,封装在 Option 中: enum Possibilities { First(i32), Second(St
我刚刚注意到 pyhton 中 Enum+defaultdict 的一个非常奇怪的行为。我定义了一个这样的枚举,它收集了一些默认词典: from enum import Enum from colle
我想在运行时从配置文件生成一些类型。为简单起见,假设我已经将数据加载为 Python 字典: color_values = dict(RED = 1, YELLOW = 2, GREEN = 3) 我
我想创建一个方法,在传入参数的任何枚举类上实现 valueOf(并包装一些专门的错误/缺失名称代码)。所以基本上,我有几个枚举,例如: enum Enum1{ A, B, C } enum Enum2
我有一个user数据库表: CREATE TABLE IF NOT EXISTS `user` ( `user_id` INT UNSIGNED NOT NULL AUTO_INCREMENT,
这是我的 JPA @Repository,在这里我们可以得到 list与 findByFullNameContaining(String query) - 通过在查询中提供 fullName 的子字符
(也在这里打开了一个问题:https://github.com/square/moshi/issues/768 但被要求打开一个stackoverflow问题)我正在编写一个通用适配器来转换带有枚举值
请帮助找到以下情况的最佳方法: 有一个表,大约有 20 列。 每列都有自己的短名称、全名称和类型(数字或字符串)。 每个列类型都可以有自己的运算符 - 例如,字符串 - 包含、等于;数字 - 更多、更
如果我在 python 中按功能创建了 enum.Enum(本例中为 3.7),如何从中检索类的名称? import enum def print_the_enum_class_name(some_e
我正在编写一个与第 3 方程序交互的程序。这个第 3 方程序允许用户制作可以运行在第 3 方程序中进行的步骤记录的按钮。 但! 这些按钮还可以运行用户定义的批处理文件。因此,我使用此功能通过创建文件并
我是一名优秀的程序员,十分优秀!