gpt4 book ai didi

java - Dagger :注入(inject)一个类的不同实例

转载 作者:搜寻专家 更新时间:2023-11-01 01:26:35 25 4
gpt4 key购买 nike

我可以使用Dagger吗?将不同的值注入(inject)到对象图深处的同一类的多个实例中?我想避免通过图中的包含对象传递值(这样我可以更改包含对象的实现而不影响它们的容器)。

这是一个人为的例子。对象图是一个 Top,它包含一个 Left 和 Right,每个包含一个 Show。所以有两个 Show 实例。

class Top {
Left left;
Right right;
void encodeTwice(String data) {
left.encode(data);
right.encode(data.getBytes());
}
}
class Left {
Leaf leaf;
void encode(String data) {
leaf.write(URLEncoder.encode(data));
}
}
class Right {
Leaf leaf;
void encode(byte[] data) {
leaf.write(DatatypeConverter.printBase64Binary(data));
}
}
interface Leaf {
void write(String data);
}
class Show implements Leaf {
String label;
@Override public void write(String data) {
System.out.println(label + ": " + data);
}
}
// There might be other classes that implement Leaf.

我可以使用 Dagger 为 Top.left.leaf.label 和 Top.right.leaf.label 注入(inject)不同的值吗?

这是一次尝试。这并不令人满意,因为 Left 和 Right 依赖于 Leaf 的实现。我想在不涉及左或右的情况下注入(inject) Show.label。

ObjectGraph.create(new TopModule()).get(Top.class).encodeTwice("Hello!");

@Module(injects = Top.class)
public class TopModule {
@Provides @Named("Left.leaf.label") String provideLeftLabel() {
return "URL encoded";
}
@Provides @Named("Right.leaf.label") String provideRightLabel() {
return "Base 64";
}
}
class Top {
@Inject Left left;
@Inject Right right;
void encodeTwice(String data) {
left.encode(data);
right.encode(data.getBytes());
}
}
class Left {
Leaf leaf;
@Inject Left(@Named("Left.leaf.label") String label) {
leaf = new Show(label);
}
void encode(String data) {
leaf.write(URLEncoder.encode(data));
}
}
class Right {
Leaf leaf;
@Inject Right(@Named("Right.leaf.label") String label) {
leaf = new Show(label);
}
void encode(byte[] data) {
leaf.write(DatatypeConverter.printBase64Binary(data));
}
}
interface Leaf {
void write(String data);
}
class Show implements Leaf {
String label;
Show(String label) {
this.label = label;
}
@Override public void write(String data) {
System.out.println(label + ": " + data);
}
}

最佳答案

所以这很难用这样的抽象类型来推理,因为根据 LeftRightTop 实际上我可能会改变方法。

我可以想到三种解决这个问题的方法:Leaf 工厂,通过限定符区分 LeafLeaf 的区分>s by child graphs.

工厂

Leaf 实现隐藏在需要其所需依赖项的工厂后面。

interface LeafFactory {
Leaf create(String name);
}

class TopModule {
// ...

@Provide @Singleton LeafFactory provideLeafFactory() {
return new LeafFactory() {
@Override public Leaf create(String name) {
return new Show(name);
}
};
}
}

class Left {
private final Leaf leaf;

@Inject Left(@Named("..") String name, LeafFactory leafFactory) {
leaf = leafFactory.create(name);
}

// ...
}

限定符

现在您正在通过限定符注解注入(inject)标签。除了使用 Leaf 实例,您没有理由不能做同样的事情。

class TopModule {
@Provides @Named("..") Leaf provideLeftLeaf() {
return new Show("URL encoded");
}

@Provides @Named("..") Leaf provideRightLeaf() {
return new Show("Base64 encoded");
}
}

class Left {
@Inject Left(@Named("..") Leaf leftLeaf) { .. }
}

注意:如果您还需要该抽象,您仍然可以将标签名称作为参数注入(inject) provideFooLeaf 方法。

子图

通过使用“叶范围”扩展图,您可以创建应用程序的单独部分(例如,“右”、“左”),它们只有一个 Leaf 实例。

@Module(addsTo = TopModule.class)
class LeafModule {
private final String label;

LeafModule(String label) {
this.label = label;
}

@Provide @Singleton Leaf provideLeaf() {
return new Show(label);
}
}

现在我们可以将 .plus() this 加到根对象图上以获得作用域。

ObjectGraph ogRoot = ObjectGraph.create(new TopModule());
// ...
ObjectGraph ogLeft = ogRoot.plus(new LeafModule("URL encoded"));
Leaf left = ogLeft.get(Leaf.class);
ObjectGraph ogRight = ogRoot.plus(new LeafModule("Base64 encoded"));
Leaf right = ogRight.get(Leaf.class);

正如我所说,这很难概括为适合您概述的抽象示例应用程序。但我希望你能看到子图如何让你创建一个特殊版本的图,它是“根”图的超集。通过多次执行此操作,您可以更改用于注入(inject)类型的依赖项。

关于java - Dagger :注入(inject)一个类的不同实例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21245284/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com