Glide 圖片庫使用和原理(一)

1 Glide 使用 ---4.11.0

導(dǎo)入庫

implementation 'com.github.bumptech.glide:glide:4.11.0'
    annotationProcessor 'com.github.bumptech.glide:compiler:4.11.0'

1.1 簡單使用

class GlideTest : AppCompatActivity() {
    lateinit var binding: ActivityGlideTestBinding
    val imgUrl = "https://cn.bing.com/sa/simg/hpb/LaDigue_EN-CA1115245085_1920x1080.jpg"
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = DataBindingUtil.setContentView(this, R.layout.activity_glide_test)
        binding.showImg.setOnClickListener {

            Glide.with(this).load(imgUrl).into(binding.contentImg)

        }
    }
}

1.2 其他部分用法

val doc1 = """
    Glide 加載圖片的方式
    // 加載本地圖片
    File file = new File(getExternalCacheDir() + "/image.jpg");
    Glide.with(this).load(file).into(imageView);

    // 加載應(yīng)用資源
    int resource = R.drawable.image;
    Glide.with(this).load(resource).into(imageView);

    // 加載二進(jìn)制流
    byte[] image = getImageBytes();
    Glide.with(this).load(image).into(imageView);

    // 加載Uri對象
    Uri imageUri = getImageUri();
    Glide.with(this).load(imageUri).into(imageView);
"""

val doc2 = """
    Glide 占位圖,和一些屬性實例
    Glide.with(this)
     .load(url)
     .asBitmap()
     .placeholder(R.drawable.loading)
     .error(R.drawable.error)
     .diskCacheStrategy(DiskCacheStrategy.NONE)
     .into(imageView);
     
    Glide.with(this).load(url)
     .asGif()
     .placeholder(R.drawable.loading).error(R.drawable.error).diskCacheStrategy(DiskCacheStrategy.NONE)
     .into(imageView);
     
    Glide.with(this).load(url).placeholder(R.drawable.loading.error(R.drawable.error).diskCacheStrategy(DiskCacheStrategy.NONE)
     .override(100, 100)
     .into(imageView);
"""

2 三部曲with,load,into

2.1

第一步

Glide#width(xxxx) 目標(biāo)獲取 RequestManager

第二步
Glide#load(xxx) 目標(biāo)獲取 RequestBuilder

第三步
Glide#into(xxx) 獲取和處理Target

  1. 運行隊列,等待隊列
  2. 活動緩存
  3. 內(nèi)存緩存
  4. 網(wǎng)絡(luò)模型

2.1.1 with

監(jiān)聽Activity/Fragment生命周期

當(dāng)我們的Fragment或Activity不可見的時候暫停請求,當(dāng)我們的Fragment或Activity可見的時候回復(fù)請求

處理前臺可見的 Activity / Fragment,提高資源利用率;
在有必要時釋放資源以避免在應(yīng)用在后臺時被殺死,提高穩(wěn)定性;

Glide#with(params)
with會有很多重載方法,params參數(shù)有這些種類:(FragmentActivity,androidx.fragment.app.Fragment,android.app.Fragment,View)
通過獲取RequestManagerRetriever 進(jìn)行檢索

RequestManagerRetriever#get方法 多個重載

通過源碼,我們會看到四個重載方法

public RequestManager get(@NonNull Context context)

public RequestManager get(@NonNull FragmentActivity activity)

public RequestManager get(@NonNull Fragment fragment)

public RequestManager get(@NonNull Activity activity)

public RequestManager get(@NonNull View view)

通過這些發(fā)放的分析,我們得出下面的分析

生命周期的作用域(1.Application, 2.Activity, 3.Fragment)
with 參數(shù) 作用域 代碼中線程
Application 子線程使用with
View Fragment/Activity 主線程
Fragment Fragment 主線程
Activity Activity 主線程
FragmentActivity Activity 主線程
ServiceContext/ApplicationContext Application 主線程

一共分為兩種:
第一種是作用域Application,它的生命周期是全局 的,不搞空白Fragment綁定Activity/Fragment
第二種是作用域非Application,它的生命周期是,專門搞空白Fragment綁定 Activity/Fragment

生成默認(rèn)隱藏fragment ,我們發(fā)現(xiàn)有著幾個方法

private RequestManager supportFragmentGet(
      @NonNull Context context,
      @NonNull FragmentManager fm,
      @Nullable Fragment parentHint,
      boolean isParentVisible)

private RequestManager fragmentGet(
      @NonNull Context context,
      @NonNull android.app.FragmentManager fm,
      @Nullable android.app.Fragment parentHint,
      boolean isParentVisible)


supportFragmentGet---->getSupportRequestManagerFragment
獲取SupportRequestManagerFragment (Tag:FRAGMENT_TAG)的fragment

private SupportRequestManagerFragment getSupportRequestManagerFragment(
      @NonNull final FragmentManager fm, @Nullable Fragment parentHint, boolean isParentVisible) {
    SupportRequestManagerFragment current =
        (SupportRequestManagerFragment) fm.findFragmentByTag(FRAGMENT_TAG);
    if (current == null) {
      current = pendingSupportRequestManagerFragments.get(fm);
      if (current == null) {
        current = new SupportRequestManagerFragment();
        current.setParentFragmentHint(parentHint);
        if (isParentVisible) {
          current.getGlideLifecycle().onStart();
        }
        pendingSupportRequestManagerFragments.put(fm, current);
        fm.beginTransaction().add(current, FRAGMENT_TAG).commitAllowingStateLoss();
        handler.obtainMessage(ID_REMOVE_SUPPORT_FRAGMENT_MANAGER, fm).sendToTarget();
      }
    }
    return current;
  }

fragmentGet---->getRequestManagerFragment
獲取RequestManagerFragment (Tag:FRAGMENT_TAG)的fragment

private RequestManagerFragment getRequestManagerFragment(
      @NonNull final android.app.FragmentManager fm,
      @Nullable android.app.Fragment parentHint,
      boolean isParentVisible) {
    RequestManagerFragment current = (RequestManagerFragment) fm.findFragmentByTag(FRAGMENT_TAG);
    if (current == null) {
      current = pendingRequestManagerFragments.get(fm);
      if (current == null) {
        current = new RequestManagerFragment();
        current.setParentFragmentHint(parentHint);
        if (isParentVisible) {
          current.getGlideLifecycle().onStart();
        }
        pendingRequestManagerFragments.put(fm, current);
        fm.beginTransaction().add(current, FRAGMENT_TAG).commitAllowingStateLoss();
        handler.obtainMessage(ID_REMOVE_FRAGMENT_MANAGER, fm).sendToTarget();
      }
    }
    return current;
  }
生命周期的綁定

【記錄保存】 FragmentManager -SupportRequestManagerFragment

RequestManagerRetriever .java

final Map<FragmentManager, SupportRequestManagerFragment> pendingSupportRequestManagerFragments =
new HashMap<>();

supportFragmentGet----getSupportRequestManagerFragment

  1. 從FragmentManager中獲取SupportRequestManagerFragment
SupportRequestManagerFragment current =
        getSupportRequestManagerFragment(fm, parentHint, isParentVisible);

2.從該 Fragment 中獲取 RequestManager

RequestManager requestManager =current.getRequestManager();

3.首次獲取,則實例化 RequestManager

------> getSupportRequestManagerFragment
1.1 嘗試獲取 FRAGMENT_TAG 對應(yīng)的 Fragment
1.2 嘗試從臨時記錄中獲取Fragment
如果為獲取到
1.3 current==null---實例化 Fragment

1.3.1 創(chuàng)建對象SupportRequestManagerFragment
1.3.2 如果父層可見,則調(diào)用 onStart() 生命周期
1.3.3 臨時記錄映射關(guān)系pendingSupportRequestManagerFragments.put(fm, current)
1.3.4 提交 Fragment 事務(wù)
1.3.5 post 一個消息
handler.obtainMessage(ID_REMOVE_SUPPORT_FRAGMENT_MANAGER, fm).sendToTarget();

-》post 一個消息分析: RequestManagerRetriever 本身實現(xiàn)了Handler的callback
handleMessage中

case ID_REMOVE_SUPPORT_FRAGMENT_MANAGER:
        FragmentManager supportFm = (FragmentManager) message.obj;
        key = supportFm;
        removed = pendingSupportRequestManagerFragments.remove(supportFm);
        break;

就是為了避免 SupportRequestManagerFragment 在一個作用域中重復(fù)創(chuàng)建。

因為 commitAllowingStateLoss() 是將事務(wù) post 到消息隊列中的,也就是說,事 務(wù)是異步處理的,而不是同步處理的。假設(shè)沒有臨時保存記錄,一旦在事務(wù)異步等待 執(zhí)行時調(diào)用了 Glide.with(...) ,就會在該作用域中重復(fù)創(chuàng)建 Fragment。

生命周期的監(jiān)聽機制

Glide內(nèi)部會在 Activity/Fragment生命周期監(jiān)聽,網(wǎng)絡(luò)變化監(jiān)聽,自動取消加 載或者重新加載

框架為每個Activity 和 Fragment 作用域創(chuàng)建了 一個無UI的Fragment

分析這個空Fragment---》 SupportRequestManagerFragment.java
看下我們比較關(guān)心的代碼

ActivityFragmentLifecycle lifecycle

public SupportRequestManagerFragment() {
    this(new ActivityFragmentLifecycle());
  }

  
  public SupportRequestManagerFragment(@NonNull ActivityFragmentLifecycle lifecycle) {
    this.lifecycle = lifecycle;
  }

 
  public void setRequestManager(@Nullable RequestManager requestManager) {
    this.requestManager = requestManager;
  }

 @Override
  public void onStart() {
    super.onStart();
    lifecycle.onStart();
  }

  @Override
  public void onStop() {
    super.onStop();
    lifecycle.onStop();
  }

  @Override
  public void onDestroy() {
    super.onDestroy();
    lifecycle.onDestroy();
    unregisterFragmentWithRoot();
  }

回溯一下,RequestManagerRetriever.java 和 RequestManager.java

  1. 實例化RequestManager
Glide glide = Glide.get(context);
      requestManager =
          factory.build(
              glide, current.getGlideLifecycle(), current.getRequestManagerTreeNode(), context);
  1. RequestManager 工廠
public interface RequestManagerFactory {
    @NonNull
    RequestManager build(
        @NonNull Glide glide,
        @NonNull Lifecycle lifecycle,
        @NonNull RequestManagerTreeNode requestManagerTreeNode,
        @NonNull Context context);
  }
  1. 默認(rèn)工廠接口實現(xiàn)
private static final RequestManagerFactory DEFAULT_FACTORY =
      new RequestManagerFactory() {
        @NonNull
        @Override
        public RequestManager build(
            @NonNull Glide glide,
            @NonNull Lifecycle lifecycle,
            @NonNull RequestManagerTreeNode requestManagerTreeNode,
            @NonNull Context context) {
          return new RequestManager(glide, lifecycle, requestManagerTreeNode, context);
        }
      };

RequestManager.java 源碼我們關(guān)心的部分

final Lifecycle lifecycle;

public RequestManager(
      @NonNull Glide glide,
      @NonNull Lifecycle lifecycle,
      @NonNull RequestManagerTreeNode treeNode,
      @NonNull Context context) {
......
    this.lifecycle = lifecycle;
    this.requestTracker = requestTracker;// 生命周期回調(diào)
。。。
if (Util.isOnBackgroundThread()) {
      mainHandler.post(addSelfToLifecycle);
    } else {// 添加監(jiān)聽
      lifecycle.addListener(this);
    }
。。。

public synchronized void onDestroy() {
    targetTracker.onDestroy();
    ......
    lifecycle.removeListener(this);//remove 監(jiān)聽
  }
}

實例化 RequestManager 時需要一個 Lifecycle對象,這個對象是 在無界面 Fragment 中創(chuàng)建的,當(dāng) Fragment 的生命周期變化時,就是通過這個 Lifecycle 對象將事件分發(fā)到RequestManager

生命周期的回調(diào)

RequestsManger 收到的回調(diào)ConnectivityMonitor

public interface LifecycleListener {
    void onStart();
    void onStop();
    void onDestroy();
}

Activity/Fragment 不可見時暫停請求 (onStop() ) 函數(shù) Activity/Fragment 可見時恢復(fù)請求 (onStart() ) 函數(shù) Activity/Fragment 銷毀時銷毀請求 (onDestroy() )函數(shù)

---》 RequestManger 操作 requestTracker

@Override
  public synchronized void onStop() {
    pauseRequests();
    targetTracker.onStop();
  }

@Override
  public synchronized void onStart() {
    resumeRequests();
    targetTracker.onStart();
  }

  public synchronized void resumeRequests() {
    requestTracker.resumeRequests();
  }

public synchronized void pauseRequests() {
    requestTracker.pauseRequests();
  }

@Override
  public synchronized void onDestroy() {
    targetTracker.onDestroy();
    for (Target<?> target : targetTracker.getAll()) {
      clear(target);
    }
    targetTracker.clear();
    requestTracker.clearRequests();
    lifecycle.removeListener(this);
    lifecycle.removeListener(connectivityMonitor);
    mainHandler.removeCallbacks(addSelfToLifecycle);
    glide.unregisterRequestManager(this);
  }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,702評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,143評論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,553評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,620評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,416評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,940評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,024評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,170評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,709評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,597評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,784評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,291評論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,029評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,407評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,663評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,403評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,746評論 2 370

推薦閱讀更多精彩內(nèi)容