Кинжал 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, который должен быть введен, но его нет, может ли кто-нибудь помочь:

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

person j2emanue    schedule 08.06.2015    source источник


Ответы (2)


вот мой класс основной активности Android, который должен быть введен, но не

Вы ожидаете, что волшебные вещи произойдут, когда вы аннотируете что-то с помощью @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();
}

В этом случае вы создаете экземпляр VehicleComponent, реализованный Dagger, и извлекаете из него экземпляр 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. Вы уже аннотировали конструктор 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.

person nhaarman    schedule 08.06.2015
comment
он говорит, что не может найти DaggerVehicleComponent в среде IDE. Любые идеи ? я удалил аннотацию Inject и следую вашему первому методу, поэтому я могу создавать частные методы. его пакет локальный. Пакет Everythings теперь локальный. но не находит DaggerVehicleComponent. Я предполагаю, что это что-то, что автоматически генерируется. - person j2emanue; 08.06.2015
comment
Ознакомьтесь с моим ответом здесь, чтобы узнать, как заставить Android Studio найти сгенерированные классы. - person nhaarman; 08.06.2015
comment
с помощью этого плагина применения: «com.neenbedankt.android-apt» и в файле build.gradle это: classpath «com.neenbedankt.gradle.plugins:android-apt:1.4» заставило его автоматически генерировать классы для меня. Ваше объяснение очень хорошее. Спасибо, сэр. - person j2emanue; 08.06.2015

вы пропускаете следующие шаги

вы создали такой модуль в классе приложения

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);
person rahulrv    schedule 08.06.2015