java - 枚举感知的 ServiceLoader 实现?
问题描述
我希望能够将枚举类型指示为接口实现,然后通过ServiceLoader
API 将所有枚举加载为接口的单独实例/实现。此用例的一个示例是允许我的 API 的下游用户指定自定义值,但提供具有标准/通用实现的枚举。我的接口只需要一个 String name()
,所以任何枚举都已经实现了它。
例如,CopyOption
Java NIO API 中的接口,带有提供的StandardCopyOption
枚举。假设我想将所有CopyOption
s,甚至是类路径上的新 s 连同标准一起加载到单个迭代器中ServiceLoader
(或者我愿意接受其他建议!)
我终于通过批量复制和修改它以在实例化失败时ServiceLoader
尝试使用它来工作(其中的部分是它当前的工作方式,而其中的部分是我添加/更改的部分):getEnumConstants
try
catch
try {
S p = service.cast(c.newInstance());
providers.put(cn, p);
return p;
} catch (Throwable x) {
Object[] arr = c.getEnumConstants();
if (arr == null || arr.length == 0) {
fail(service, "Provider " + cn + " could not be instantiated", x);
}
List<S> list = new LinkedList<>();
for (Object o : arr) {
Enum<?> e = (Enum<?>) o;
S p = service.cast(e);
providers.put(cn + e.ordinal(), p);
list.add(p);
}
subiter = list.iterator();
return subiter.next();
}
我还添加了一些代码,如果subiter
存在并且有下一个,则在继续下一个类名之前对其进行迭代。
我的问题是:有没有更好的方法?
如果最终用途不明确,现在可以通过上述修改实现:
interface ImageType {
String name();
}
@AutoService(ImageType.class)
enum StandardImageType implements ImageType {
IMAGE,
VECTOR,
RASTER,
HANDWRITING,
ICON,
LOGO,
SEAL,
RULE,
BARCODE
}
解决方案
随着 Java 模块的引入,通过默认构造函数进行实例化的替代方法已添加到服务提供者中。但它仅在提供程序位于命名模块中时才有效。
提供者类可以声明服务类型public static T provider()
所在的方法。T
然后,提供者实现类甚至不需要实现或扩展T
自身。
由于数组和泛型类型List<ImageType>
都不能用作服务类型,我们需要另一种类型来潜在地封装多个实际实例,例如
package somemodule;
import java.util.function.Supplier;
public interface ImageType {
String name();
interface ImageTypes extends Supplier<ImageType[]> {}
}
和
package somemodule;
public enum StandardImageType implements ImageType {
IMAGE,
VECTOR,
RASTER,
HANDWRITING,
ICON,
LOGO,
SEAL,
RULE,
BARCODE;
public static ImageTypes provider() {
return StandardImageType::values;
}
}
和一个模块声明,如
module SomeModule {
uses somemodule.ImageType.ImageTypes;
provides somemodule.ImageType.ImageTypes with somemodule.StandardImageType;
}
允许写,例如
List<ImageType> all = ServiceLoader.load(ImageType.ImageTypes.class)
.stream().flatMap(p -> Arrays.stream(p.get().get()))
.collect(Collectors.toList());
模块内的某处(或任何其他带有uses somemodule.ImageType.ImageTypes;
声明的模块)。
推荐阅读
- c# - 在 Daypilot Scheduler 的单元格上禁用鼠标悬停
- android - 如何关闭活动
- docker - Consul 服务器节点与客户端节点
- javascript - Flipclock 初始值为 939,511,314?
- python - 使用 Git-Bash 在 Windows 中激活虚拟环境
- python - 使用python从文本文件中交叉行
- c# - Unity 在没有 IsTrigger 的情况下使用 OnCollisionEnter2D 检测碰撞
- python - 仅打印最早的月份
- mongodb - 带有 typescript 类的 Meteor 和 Mongo 出现错误“类型 'typeof Mongo' 上不存在属性 'Collection'。” 编译时
- r - 使用变量作为 data.table 中的标题进行计算