Dagger 2 - 无法注入对象

Dagger 2 - unable to inject object

我正在尝试在 Android 应用程序中进行非常简单的依赖注入。我正在使用 dagger 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,它使用电机 class:

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 class 来定义我的提供商:

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 class 应该被注入但是没有,任何人都可以帮忙:

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'
}

您缺少以下步骤

您已经在应用程序中创建了这样的模块 class

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

        }
    }

并将其注入 activity

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

here is my Android mainactivity class that should be injected but its not

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

您可以通过多种方式完成此操作,我将介绍两种。


首先,在您的MainActivityonCreate中:

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

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


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

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

  Vehicle provideVehicle();

  void inject(MainActivity mainActivity);
}

在您的 MainActivity class 中,您调用 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,而不是使用模块自己创建实例。你已经用 @Inject 注释了 Vehicle 构造函数,所以 Dagger 会知道使用这个构造函数。但是,它需要一个它不知道的 Motor 的实例。如果你在Motor的构造函数中也添加一个@Inject注解,并用@Singleton注解Motorclass,你可以去掉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;
    }
}

你的Vehicleclass:

@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 class,并在您的 VehicleComponent.

中删除对它的引用