[TOC]
# Handler原理分析
## 關鍵流程
1、Looper不斷的從MessageQueue中取出Message,然后交給Message對應的Handler處理。同時Handler還執行發送Message到MessageQueue中的動作。
2、Handler對象持有Looper的目的是為了拿到Looper對應的MessageQueue,并往其中插入消息;Looper從MessageQueue中取出Message后,交給Message的target也就是Handler對象處理。
3、Looper對象存儲在ThreadLocal中,屬于線程局部變量,只能被當前線程訪問。
## 詳細介紹
1、Handler調用sendMessage、post方法發送Message,并插入到MessageQueue中,MessaQueue采用單鏈表結構。
2、Handler類有一個Looper成員變量,Looper屬于線程局部變量,每個線程有且只能有一個Looper;Looper類有一個MessageQueue成員變量;Handler持有Looper主要是為了拿到Looper對應的MessageQueue,并往其中插入消息。
3、子線程需要先調用Looper.prepare方法,來創建一個Looper對象存儲到ThreadLocal中。然后創建Handler時會調用Looper.myLooper方法獲取當前線程的Looper。
4、Looper.loop方法開啟消息循環,Looper會循環從其MessageQueue中提取消息,并調用消息的target(也就是Handler)進行分發處理
5、Handler拿到Message后,先判斷Message的Callback是否為空,不為空直接執行,消息處理結束;為空則判斷Handler的Callback是否為空,不為空則執行,并決定是否進行攔截,攔截則消息處理結束;不攔截則執行Handler的handleMessage方法。
## ThreadLocal介紹
1、ThreadLocal是一個創建線程局部變量的類。
2、一般情況下,我們創建的線程可以被任何一個線程訪問并修改,而使用ThreadLocal創建的變量只能被當前線程訪問,其他線程無法訪問和修改。
3、ThreadLocal的set方法存入的值,實際是放到當前線程的ThreadLocalMap實例中,key是當前ThreadLocal對象,value是存入的值。
4、get方法直接從當前線程的ThreadLocalMap中,根據key獲取對象
# 一個線程中初始化多個Handler,會產生多少個Looper
只會產生一個Looper。因為Looper是和當前線程進行綁定的,屬于線程局部變量,每個線程有且只能有一個Looper
# 如果只有一個Looper,Looper如何區分Handler,Handler發送了消息會不會導致Looper錯亂,最終不知道誰處理
Handler在將消息插入到消息隊列時,會將消息的target屬性設置為該Handler自身,消息的target屬性為空時是無法插入消息隊列的。Looper取出消息后會調用該消息的target屬性(也就是Handler對象)的dispatchMessage方法來分發消息給相應的Handler。
# 多進程和多線程的區別
1、本質區別在于:每個進程擁有自己的一整套變量,而線程則共享數據。
2、進程是程序的一次執行。計算機在同一刻運行多個程序,每個程序稱為一個進程(計算機將 CPU 的時間片分配給每一個進程)
3、線程是 CPU 的基本調度單位。一個程序同時執行多個任務,每個任務稱為一個線程
# 進程間通訊方法,線程間通訊方法
1、進程間通訊方法:
* Broadcast
* Intent/Bundle
* File 共享
* Messenger
* AIDL
* ContentProvider
* Socket
2、線程間通訊方法:
* Handler(AsyncTask、Message、runOnUiThread 等)
* EventBus
* LocalBroadcast
# 關鍵字 synchronized 的作用
# sleep 和 wait 的區別
1、二者都可以暫停當前線程,釋放 CPU 控制權
2、區別在于作用于誰和是否釋放鎖?
3、wait 方法作用于 Object,sleep 方法作用于 Thread
Object.wait 方法在釋放 CPU 的同時,釋放了對象鎖的控制,使得其他線程可以使用同步控制塊或方法;Thread.sleep 方法沒有釋放鎖
- 導讀
- Java知識
- Java基本程序設計結構
- 【基礎知識】Java基礎
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設計結構
- 對象與類
- 【基礎知識】對象與類
- 【專題分析】Java類加載過程
- 【面試清單】對象與類
- 泛型
- 【基礎知識】泛型
- 【面試清單】泛型
- 集合
- 【基礎知識】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎知識】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識
- Activity
- 【基礎知識】Activity
- 【專題分析】運行時權限
- 【專題分析】使用Intent打開三方應用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數據綁定
- 【面試清單】架構組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發機制
- 【源碼分析】按鍵事件分發機制
- 【源碼分析】Android窗口機制
- 【面試清單】界面
- 動畫和過渡
- 【基礎知識】動畫和過渡
- 【面試清單】動畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺任務
- 應用數據和文件
- 基于網絡的內容
- 多線程與多進程
- 【基礎知識】多線程與多進程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進程
- 應用優化
- 【專題分析】布局優化
- 【專題分析】繪制優化
- 【專題分析】內存優化
- 【專題分析】啟動優化
- 【專題分析】電池優化
- 【專題分析】包大小優化
- 【面試清單】應用優化
- Android新特性
- 【專題分析】狀態欄、ActionBar和導航欄
- 【專題分析】應用圖標、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標識符的最佳做法
- 開源庫源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識
- Flutter
- 原生開發與跨平臺開發
- 整體歸納
- 狀態及狀態管理
- 零碎知識點
- 添加Flutter到現有應用
- Git知識
- Git命令
- .gitignore文件
- 設計模式
- 創建型模式
- 結構型模式
- 行為型模式
- RxJava
- 基礎
- Linux知識
- 環境變量
- Linux命令
- ADB命令
- 算法
- 常見數據結構及實現
- 數組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊列
- 算法時間復雜度
- 常見算法思想
- 其他技術
- 正則表達式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識
- 開發歸納
- Android零碎問題
- 其他零碎問題
- 開發思路