4

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

这是一个人为的例子。对象图是一个 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 注入不同的值吗?

这是一个尝试。这是不能令人满意的,因为 Left 和 Right 依赖于 Leaf 的实现。我想在不涉及 Left 或 Right 的情况下注入 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);
  }
}
4

3 回答 3

10

所以这很难用这样的抽象类型来推理,因为基于 what Left, Right, and Topactually are 我可能会改变方法。

我能想到解决这个问题的三种方法:Leaf工厂、Leaf通过限定符对Leafs 进行微分、通过子图对 s 进行微分。

工厂

将实现隐藏在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);
  }

  // ...
}

预选赛

现在,您正在通过限定符注释注入标签。没有理由你不能做同样的事情,但有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) { .. }
}

注意:如果您还需要该抽象,您仍然可以将标签名称作为参数注入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()它放到根对象图上以获取范围。

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);

就像我说的那样,这很难概括以适合您概述的抽象示例应用程序。但我希望您能看到子图如何允许您创建图的专用版本,它是“根”图的超集。通过多次执行此操作,您可以更改用于注入类型的依赖项。

于 2014-01-23T20:45:05.827 回答
0

Jake 展示了不同选择的绝佳示例。据我了解,标签实际上是由 Left 或 Right 类定义的。您可以将字段移动到这些类并将它们传递给 Leaf。如果你想要一个无标签的 Leaf,你可以使用 LabeledLeaf 扩展接口,然后 Left 和 Right 可以 instanceof LabeledLeaf 来决定它们是否应该使用带标签的 write 方法。

您的示例设置实际上听起来很像Timber。您应该考虑看看我们如何获得Tree能够根据其实现写入不同输出的 s,并且一些能够被标记。最终结果的不同之处在于 Timber 的 API 是静态方法,而不是您注入到类中的组件。

于 2014-01-24T05:46:00.143 回答
0

如果您为注入类的每个实例创建一个单独的 ObjectGraph,Dagger 就可以做到这一点。为了提供完整的图,模块可以实现委托给子图的 @Provides 方法。感谢 Jake Wharton 的建议,促使我采用了这种技术。

这是一个例子:

LeafModule leftLeaf = new ShowModule("URL encoded");
LeafModule rightLeaf = new ShowModule("Base 64");
ObjectGraph top = ObjectGraph.create(new TopModule(leftLeaf, rightLeaf));
top.get(Top.class).encodeTwice("Hello!");

@Module(injects = Top.class)
class TopModule {
  TopModule(LeafModule leftLeaf, LeafModule rightLeaf) {
    leftGraph = ObjectGraph.create(leftLeaf).plus(new MiddleModule());
    rightGraph = ObjectGraph.create(rightLeaf).plus(new MiddleModule());
  }
  private final ObjectGraph leftGraph;
  private final ObjectGraph rightGraph;
  @Provides Left getLeft() {
    return leftGraph.get(Left.class);
  }
  @Provides Right getRight() {
    return rightGraph.get(Right.class);
  }
  @Module(injects = {Left.class, Right.class},
          complete = false /* doesn't inject Leaf */)
  static class MiddleModule {
  }
}
/** A Dagger module that can provide a Leaf. */
interface LeafModule {
}
@Module(injects = Leaf.class)
class ShowModule implements LeafModule {
  ShowModule(String showLabel) {
    label = showLabel;
  }
  private final String label;
  @Provides @Named("Show.label") String getLabel() {
    return label;
  }
  @Provides Leaf provideLeaf(Show implementation) {
    return implementation;
  }
}
class Top {
  @Inject Left left;
  @Inject Right right;
  void encodeTwice(String data) {
    left.encode(data);
    right.encode(data.getBytes());
  }
}
class Left {
  @Inject Leaf leaf;
  void encode(String data) {
    leaf.write(URLEncoder.encode(data));
  }
}
class Right {
  @Inject Leaf leaf;
  void encode(byte[] data) {
    leaf.write(DatatypeConverter.printBase64Binary(data));
  }
}
interface Leaf {
  void write(String data);
}
class Show implements Leaf {
  @Inject @Named("Show.label") String label;
  @Override public void write(String data) {
    System.out.println(label + ": " + data);
  }
}
于 2014-01-26T00:11:10.420 回答