6

ActivityInstrumentationTestCase2を継承するAndroid機能テストでDaggerを使用しようとしています。

セットアップコードは次のようになります。

@Override
protected void setUp() {
    // TODO Auto-generated method stub
    try {
        super.setUp();
    } catch (Exception e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    ObjectGraph.create(new TestModule()).inject(this);
    this.activity = super.getActivity();
}`

を呼び出すことによって起動されるOnCreateメソッドはsuper.getActivity()、TestModuleによって提供されるクラスを使用しません。しかし、アクティビティを手動で(テストコンテキストの外で)実行すると、すべての適切なクラスが非テストモジュールによって提供/注入されます。

4

2 に答える 2

3

ActivityInstrumentationTestCase2オブジェクト グラフを遅延作成することで、 Dagger を使用する方法を見つけました。私がしていることは、クラスが初めて注入されるまでオブジェクト グラフを作成するのを待つことです。そのため、呼び出す前にモジュールを追加しgetActivity()(テスト中のアクティビティのアクティビティ ライフサイクルを開始します) overrides = true、テスト モジュールで使用します。 、これはうまくいきます。関連するクラスとスニペットは次のとおりです。

GraphHolderは、その名前が示すように、ObjectGraphオブジェクトを保持します。直接ではなく、このクラスに対してすべての呼び出しを行いますObjectGraph

public class GraphHolder {

    private static GraphHolder sInstance;

    private Object[] mModules;
    private ObjectGraph mGraph;

    private GraphHolder() {
    }

    public static GraphHolder getInstance() {
        if (sInstance == null) {
            sInstance = new GraphHolder();
        }

        return sInstance;
    }

    public void inject(Object object) {
        if (mGraph == null) {
            create();
        }

        mGraph.inject(object);
    }

    public <T> T get(Class<T> type) {
        if (mGraph == null) {
            create();
        }

        return mGraph.get(type);
    }

    public void addModules(Object... modules) {
        if (mGraph != null) {
            mGraph.plus(modules);
        } else {
            if (mModules == null) {
                mModules = modules;
            } else {
                mModules = concatenate(mModules, modules);
            }
        }
    }

    private void create() {
        mGraph = ObjectGraph.create(mModules);
        mModules = null;
    }

    private Object[] concatenate(Object[] a, Object[] b) {
        int aLength = a.length;
        int bLength = b.length;

        Object[] c = new Object[aLength + bLength];
        System.arraycopy(a, 0, c, 0, aLength);
        System.arraycopy(b, 0, c, aLength, bLength);

        return c;
    }
}

Applicationモジュールをクラスに追加します。

public class MyApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();
        GraphHolder.getInstance().addModules(getModules());
    }

    Object[] getModules() {
        return new Object[]{
                // your modules here
        };
    }
}

注入したいクラス内では、単に呼び出すのGraphHolder.getInstance().inject(this)ではなくObjectGraph.inject(this)

overrides = trueテスト モジュールでは、テスト用にオーバーライドして@Moduleアノテーションに追加するオブジェクトを提供します。これにより、競合が発生した場合に、他のモジュールよりもこのモジュールのプロバイダーを優先するようにオブジェクト グラフに指示されます。

次に、私たちのテストでは:

@Inject Foo mFoo;

@Override
public void setUp() {
    super.setUp();
    GraphHolder.getInstance().addModules(new TestFooModule());
    GraphHolder.getInstance().inject(this); // This is when the object graph will be created
}
于 2014-09-01T18:32:09.170 に答える
1

ObjectGraph.create(new TestModule()).inject(this);

このコードは、テストされたアクティビティではなく、TestModule によって作成された依存関係を TestCase に挿入しようとしています。ここであなたがしなければならないことは

ObjectGraph.create(new TestModule()).inject(this.activity);

于 2013-02-24T11:04:10.590 に答える