Skip to content

UnPeekLiveData v6.1 设计思路

KunMinX edited this page Jun 13, 2022 · 3 revisions

以下源码设计,及对设计思路理解和解析,属于参与过源码设计讨论有效贡献者及本人共同成果,我们对此享有所有权和最终解释权。

任何个人或组织在引用以下内容时,须注明原作者和链接出处。未经授权不得用于洗稿、广告包装、卖课等商业用途。

Copyright 2019-present KunMinX

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

   http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

 

package com.kunminx.architecture.ui.callback;

import android.util.Log;

import androidx.annotation.NonNull;
import androidx.lifecycle.LifecycleOwner;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.Observer;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * TODO V6.1 版源码源于小伙伴 LiWeiGe 在 issue 中启发,
 * https://github.com/KunMinX/UnPeek-LiveData/issues/16
 * 
 * 相比 V5 版改进在于,引入 Observer 代理类设计,
 * 这使旋屏重建时,无需通过反射方式跟踪和复用基类 Map 中 Observer,
 * 转而通过 removeObserver 方式来自动移除和在页面重建后重建新 Observer,
 * 
 * 因而复杂度由原先分散于基类数据结构,到集中在 proxy 对象这一处,
 * 进一步方便源码逻辑阅读和后续修改。
 * 
 * 
 * TODO 唯一可信源设计
 * 我们在 V6 中继续沿用从 V3 版 "唯一可信源" 理念设计,
 * 确保 "事件" 发送权牢牢握在可信逻辑中枢手里,从而确保所有订阅者收到消息皆可靠且致,
 * 
 * 如这么说无体会,详见《吃透 LiveData 本质,享用可靠消息鉴权机制》解析:
 * https://xiaozhuanlan.com/topic/6017825943
 * 
 * TODO 以及支持消息从内存清空
 * 我们在 V6 中继续沿用从 V3 版 "消息清空" 设计,
 * 支持通过 clear 方法手动将消息从内存中清空,
 * 以免无用消息随着 SharedViewModel 长时间驻留而导致内存溢出发生。
 * 
 * 
 * Create by KunMinX at 2021/6/17
 */
@Deprecated
public class ProtectedUnPeekLiveDataV6_1<T> extends LiveData<T> {

  private final static String TAG = "V6Test";

  protected boolean isAllowNullValue;

  private final ConcurrentHashMap<Observer<? super T>, ObserverProxy> observerMap = new ConcurrentHashMap();

  /**
   * TODO 当 liveData 用作 event 时,可使用该方法观察 "生命周期敏感" 非粘性消息
   * 
   * state 可变且私有,event 只读且公有,
   * state 倒灌应景,event 倒灌不符预期,
   * 
   * 如这么说无体会,详见《吃透 LiveData 本质,享用可靠消息鉴权机制》解析:
   * https://xiaozhuanlan.com/topic/6017825943
   *
   * @param owner
   * @param observer
   */
  @Override
  public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
    Observer<? super T> observer1 = getObserverProxy(observer);
    if (observer1 != null) {
      super.observe(owner, observer1);
    }
  }

  /**
   * TODO 当 liveData 用作 event 时,可使用该方法观察 "生命周期不敏感" 非粘性消息
   * 
   * state 可变且私有,event 只读且公有,
   * state 倒灌应景,event 倒灌不符预期,
   * 
   * 如这么说无体会,详见《吃透 LiveData 本质,享用可靠消息鉴权机制》解析:
   * https://xiaozhuanlan.com/topic/6017825943
   *
   * @param observer
   */
  @Override
  public void observeForever(@NonNull Observer<? super T> observer) {
    Observer<? super T> observer1 = getObserverProxy(observer);
    if (observer1 != null) {
      super.observeForever(observer1);
    }
  }

  private Observer<? super T> getObserverProxy(Observer<? super T> observer) {
    if (observerMap.containsKey(observer)) {
      Log.d(TAG, "observe repeatedly, observer has been attached to owner");
      return null;
    } else {
      ObserverProxy proxy = new ObserverProxy(observer);
      observerMap.put(observer, proxy);
      return proxy;
    }
  }

  private class ObserverProxy implements Observer<T> {

    public final Observer<? super T> target;

    public boolean allow;

    public ObserverProxy(Observer<? super T> target) {
      this.target = target;
    }

    @Override
    public void onChanged(T t) {
      if (allow) {
        allow = false;
        if (t != null || isAllowNullValue) {
          target.onChanged(t);
        }
      }
    }
  }

  /**
   * TODO 当 liveData 用作 state 时,可使用该方法来观察 "生命周期敏感" 粘性消息
   * 
   * state 可变且私有,event 只读且公有,
   * state 倒灌应景,event 倒灌不符预期,
   * 
   * 如这么说无体会,详见《吃透 LiveData 本质,享用可靠消息鉴权机制》解析:
   * https://xiaozhuanlan.com/topic/6017825943
   *
   * @param owner
   * @param observer
   */
  public void observeSticky(LifecycleOwner owner, Observer<T> observer) {
    super.observe(owner, observer);
  }

  /**
   * TODO 当 liveData 用作 state 时,可使用该方法来观察 "生命周期不敏感" 粘性消息
   * 
   * state 可变且私有,event 只读且公有,
   * state 倒灌应景,event 倒灌不符预期,
   * 
   * 如这么说无体会,详见《吃透 LiveData 本质,享用可靠消息鉴权机制》解析:
   * https://xiaozhuanlan.com/topic/6017825943
   *
   * @param observer
   */
  public void observeStickyForever(Observer<T> observer) {
    super.observeForever(observer);
  }

  @Override
  protected void setValue(T value) {
    if (value != null || isAllowNullValue) {
      for (Map.Entry<Observer<? super T>, ObserverProxy> entry : observerMap.entrySet()) {
        entry.getValue().allow = true;
      }
      super.setValue(value);
    }
  }

  @Override
  public void removeObserver(@NonNull Observer<? super T> observer) {
    Observer<? super T> proxy;
    Observer<? super T> target;

    //移除防倒灌 ObserverProxy,否则就是移除粘性 Observer
    if (observer instanceof ProtectedUnPeekLiveDataV6_1.ObserverProxy) {
      proxy = observer;
      target = ((ObserverProxy) observer).target;
    } else {
      proxy = observerMap.get(observer);
      target = (proxy != null) ? observer : null;
    }
    if (proxy != null && target != null) {
      observerMap.remove(target);
      super.removeObserver(proxy);
    }
  }

  /**
   * 手动将消息从内存中清空,
   * 以免无用消息随着 SharedViewModel 的长时间驻留而导致内存溢出发生。
   */
  public void clear() {
    super.setValue(null);
  }

}