匕首2 - 无法注射物体

时间:2015-06-08 18:27:40

标签: android dependency-injection dagger-2

我试图在Android应用中进行非常简单的依赖注入。我使用匕首2作为DI工具。

问题是没有注射:

这是我的代码:

//看到Motor.java敬畏。

public class Motor {

    private int rpm;

    public Motor(){
        this.rpm = 10; //default will be 10
    }

    public int getRpm(){
        return rpm;
    }

    public void accelerate(int value){
        rpm = rpm + value;
    }

    public void brake(){
        rpm = 0;
    }
}

这里是实用电机类的Vehicle.java:

import javax.inject.Inject;


public class Vehicle {

    private Motor motor;

    @Inject
    public Vehicle(Motor motor){
        this.motor = motor;
    }

    public void increaseSpeed(int value){
        motor.accelerate(value);
    }

    public void stop(){
        motor.brake();
    }

    public int getSpeed(){
        return motor.getRpm();
    }
}

然后我创建了一个VehicleModule.java类来定义我的提供者:

import javax.inject.Singleton;

import dagger.Module;
import dagger.Provides;



@Module
public class VehicleModule {

    @Provides
    @Singleton
    Motor provideMotor(){
        return new Motor();
    }

    @Provides @Singleton
    Vehicle provideVehicle(){
        return new Vehicle(new Motor());
    }
}

然后我有一个注释的接口组件,定义如下:

import javax.inject.Singleton;

import Modules.VehicleModule;
import dagger.Component;

@Singleton
@Component(modules = {VehicleModule.class})
public interface VehicleComponent {

    Vehicle provideVehicle();

}

这是我应该注入的Android mainactivity类但不是,任何人都可以帮助:

import javax.inject.Inject;

public class MainActivity extends ActionBarActivity {

    @Inject
    Vehicle vehicle;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);


        Toast.makeText(this, String.valueOf(vehicle.getSpeed()), Toast.LENGTH_SHORT).show();
    }
}

我在车辆上获得空指针异常:

Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'int com.example.uen229.myapplication.Vehicle.getSpeed()' on a null object reference

顺便说一句,我的gradle依赖项看起来像这样:

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:22.2.0'
    compile 'com.google.dagger:dagger:2.0'
}

2 个答案:

答案 0 :(得分:6)

  

这是我应该注入的Android mainactivity类,但不是

当您使用@Inject注释某些内容时,您会期待发生神奇的事情。虽然 会发生神奇的事情,但 。您需要通过实例化Dagger生成的组件实现来自己完成。

你可以用两种方式做到这一点,我将描述两种。

首先,在MainActivity onCreate

private Vehicle vehicle; // Note, no @Inject annotation

public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);

  VehicleComponent vehicleComponent = DaggerVehicleComponent.create();
  this.vehicle = vehicleComponent.provideVehicle();

  Toast.makeText(this, String.valueOf(vehicle.getSpeed()), Toast.LENGTH_SHORT).show();
}

在这种情况下,您创建一个由Dagger实现的VehicleComponent实例,并从中获取Vehicle实例。 vehicle字段未由@Inject注释。这样做的好处是该字段可以是private,这是一件好事。

其次,如果您确实希望Dagger注入您的字段,则需要向inject添加VehicleComponent方法:

@Singleton
@Component(modules = {VehicleModule.class})
public interface VehicleComponent {

  Vehicle provideVehicle();

  void inject(MainActivity mainActivity);
}

MainActivity课程中,您拨打inject(this),这将填充vehicle字段:

@Inject
Vehicle vehicle; // Note, package-local

public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);

  VehicleComponent vehicleComponent = DaggerVehicleComponent.create();
  vehicleComponent.inject(this);

  Toast.makeText(this, String.valueOf(vehicle.getSpeed()), Toast.LENGTH_SHORT).show();
}

这带来了一些额外的配置,但有时是必要的。 不过我喜欢第一种方法。

作为最终评论,让我们看一下你的VehicleModule真的使用Dagger的力量。

您可以自己制作Dagger,而不是使用模块自己创建实例。您已经使用Vehicle@Inject构造函数进行了注释,因此Dagger将知道使用此构造函数。但是,它需要一个Motor的实例,它并不知道。如果您同时向@Inject的构造函数添加Motor注释,并使用Motor注释@Singleton类,则可以完全删除VehicleModule

例如:

@Singleton
public class Motor {

    private int rpm;

    @Inject // Just an annotation to let Dagger know how to retrieve an instance of Motor.
    public Motor(){
        this.rpm = 10; //default will be 10
    }

    public int getRpm(){
        return rpm;
    }

    public void accelerate(int value){
        rpm = rpm + value;
    }

    public void brake(){
        rpm = 0;
    }
}

您的Vehicle班级:

@Singleton
public class Vehicle {

    private Motor motor;

    @Inject
    public Vehicle(Motor motor){
        this.motor = motor;
    }

    public void increaseSpeed(int value){
        motor.accelerate(value);
    }

    public void stop(){
        motor.brake();
    }

    public int getSpeed(){
        return motor.getRpm();
    }
}

您现在可以安全地删除VehicleModule课程,并在VehicleComponent中删除对该课程的引用。

答案 1 :(得分:0)

您缺少以下步骤

您已在应用程序类

中创建此类模块
public class D2EApplication extends Applicaiton {
    public static D2EComponent component(Context context) {
            return ((D2EBaseApplication) context.getApplicationContext()).component;
        }

        public final static class DaggerComponentInitializer {

            public static D2EComponent init(D2EBaseApplication app) {
                return DaggerD2EComponent.builder()
                        .systemServicesModule(new VehicleModule(app))
                        .build();
            }

        }
    }

并将其注入活动

D2EApplication.component(this).inject(this);