web-dev-qa-db-fra.com

Android Rxjava s'abonner à un changement de variable

J'apprends le modèle Observer, je veux que mon observable garde une certaine variable quand il change sa valeur et fait quelques opérations, j'ai fait quelque chose comme:

public class Test extends MyChildActivity {

   private int VARIABLE_TO_OBSERVE = 0;

   Observable<Integer> mObservable = Observable.just(VARIABLE_TO_OBSERVE);  

   protected void onCreate() {/*onCreate method*/
       super();
       setContentView();
       method();
       changeVariable();
   }

   public void changeVariable() {
       VARIABLE_TO_OBSERVE = 1;
   }

   public void method() {
       mObservable.map(value -> {
            if (value == 1) doMethod2();
            return String.valueOf(value);
       }).subScribe(string -> System.out.println(string));
   }

   public void doMethod2() {/*Do additional operations*/}

}

Mais doMethod2 () n'est pas appelé

26
Karate_Dog

Rien n'est magique dans la vie: si vous mettez à jour une valeur, votre Observable ne sera pas averti. Vous devez le faire vous-même. Par exemple, en utilisant un PublishSubject.

public class Test extends MyChildActivity {

    private int VARIABLE_TO_OBSERVE = 0;

    Subject<Integer> mObservable = PublishSubject.create();  

   protected void onCreate() {/*onCreate method*/
        super();
        setContentView();
        method();
        changeVariable();
    }

    public void changeVariable() {
        VARIABLE_TO_OBSERVE = 1;
        // notify the Observable that the value just change
        mObservable.onNext(VARIABLE_TO_OBSERVE);
    }

   public void method() {
       mObservable.map(value -> {
           if (value == 1) doMethod2();
           return String.valueOf(value);
       }).subScribe(string -> System.out.println(string));
   }

   public void doMethod2() {/*Do additional operations*/}

 }
33
dwursteisen

Si vous êtes intéressé ici, une version Kotlin de la classe Variable, qui permet aux abonnés d'être mis à jour après chaque changement de variable.

class Variable<T>(private val defaultValue: T) {
var value: T = defaultValue
    set(value) {
        field = value
        observable.onNext(value)
    }
val observable = BehaviorSubject.createDefault(value)
}

Usage:

val greeting = Variable("Hello!")
greeting.observable.subscribe { Log.i("RxKotlin", it) }
greeting.value = "Ciao!"
greeting.value = "Hola!"

Cela imprimera:

"Hello!"
"Ciao!"
"Hola!"
36
Dario Pellegrini

@dwursteisen Rien n'est magique, non, mais je pense que nous pouvons obtenir un peu plus de magie que ça ... ????

Que diriez-vous d'utiliser un Rx BehaviourSubject de cette manière:

import rx.functions.Action1;
import rx.subjects.BehaviorSubject;    

public class BehaviourSubjectExample {

    public BehaviourSubjectExample() {
        subject.skip(1).subscribe(new Action1<Integer>() {
            @Override
            public void call(Integer integer) {
                System.out.println("The value changed to " + integer );
            }
        });
    }

    public final BehaviorSubject<Integer> subject = BehaviorSubject.create(0);

    public int  getValue()          { return subject.getValue(); }
    public void setValue(int value) { subject.onNext(value);     }
}

Supprimez la .skip(1) si vous voulez que le code d'observation voit la valeur initiale.

Le support variable reste avec le BehaviourSubject et est accessible via le classique Java Getter/Setter. Ceci est un exemple de jouet bien sûr: si votre cas d'utilisation était vraiment aussi simple là-bas '' Je ne suis pas une excuse pour ne pas simplement écrire:

private int value = 0;

public int  getValue() { return value; }
public void setValue(int value) {
    this.value = value;
    System.out.println("The value changed to " + value );
}

... mais l'utilisation de BehaviourSubject vous permet de relier les modifications à d'autres flux de données Rx à l'intérieur de votre classe pour composer des comportements plus avancés.

12
Chris Hatton