久久精品国产精品国产精品污,男人扒开添女人下部免费视频,一级国产69式性姿势免费视频,夜鲁夜鲁很鲁在线视频 视频,欧美丰满少妇一区二区三区,国产偷国产偷亚洲高清人乐享,中文 在线 日韩 亚洲 欧美,熟妇人妻无乱码中文字幕真矢织江,一区二区三区人妻制服国产

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android消息处理机制

發(fā)布時間:2025/6/17 Android 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android消息处理机制 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

Google參考了Windows的消息處理機(jī)制,在Android系統(tǒng)中實現(xiàn)了一套類似的消息處理機(jī)制。學(xué)習(xí)Android的消息處理機(jī)制,有幾個概念(類)必須了解:

1.?????? Message

消息,理解為線程間通訊的數(shù)據(jù)單元。例如后臺線程在處理數(shù)據(jù)完畢后需要更新UI,則可發(fā)送一條包含更新信息的Message給UI線程。

2.?????? Message Queue

消息隊列,用來存放通過Handler發(fā)布的消息,按照先進(jìn)先出執(zhí)行。

3.?????? Handler

Handler是Message的主要處理者,負(fù)責(zé)將Message添加到消息隊列以及對消息隊列中的Message進(jìn)行處理。

4.?????? Looper

循環(huán)器,扮演Message Queue和Handler之間橋梁的角色,循環(huán)取出Message Queue里面的Message,并交付給相應(yīng)的Handler進(jìn)行處理。

?

5.??? 線程

UI thread 通常就是main thread,而Android啟動程序時會替它建立一個Message Queue。

每一個線程里可含有一個Looper對象以及一個MessageQueue數(shù)據(jù)結(jié)構(gòu)。在你的應(yīng)用程序里,可以定義Handler的子類別來接收Looper所送出的消息。

?

運(yùn)行機(jī)理:

??????每個線程都可以并僅可以擁有一個Looper實例,消息隊列MessageQueue在Looper的構(gòu)造函數(shù)中被創(chuàng)建并且作為成員變量被保存,也就是說MessageQueue相對于線程也是唯一的。Android應(yīng)用在啟動的時候會默認(rèn)會為主線程創(chuàng)建一個Looper實例,并借助相關(guān)的Handler和Looper里面的MessageQueue完成對Activities、Services、Broadcase Receivers等的管理。而在子線程中,Looper需要通過顯式調(diào)用Looper. Prepare()方法進(jìn)行創(chuàng)建。Prepare方法通過ThreadLocal來保證Looper在線程內(nèi)的唯一性,如果Looper在線程內(nèi)已經(jīng)被創(chuàng)建并且嘗試再度創(chuàng)建"Only one Looper may be created per thread"異常將被拋出。

?

??????Handler在創(chuàng)建的時候可以指定Looper,這樣通過Handler的sendMessage()方法發(fā)送出去的消息就會添加到指定Looper里面的MessageQueue里面去。在不指定Looper的情況下,Handler綁定的是創(chuàng)建它的線程的Looper。如果這個線程的Looper不存在,程序?qū)伋?#34;Can't create handler inside thread that has not called Looper.prepare()"。

?

??????整個消息處理的大概流程是:1. 包裝Message對象(指定Handler、回調(diào)函數(shù)和攜帶數(shù)據(jù)等);2. 通過Handler的sendMessage()等類似方法將Message發(fā)送出去;3. 在Handler的處理方法里面將Message添加到Handler綁定的Looper的MessageQueue;4. Looper的loop()方法通過循環(huán)不斷從MessageQueue里面提取Message進(jìn)行處理,并移除處理完畢的Message;5. 通過調(diào)用Message綁定的Handler對象的dispatchMessage()方法完成對消息的處理。

?

??????在dispatchMessage()方法里,如何處理Message則由用戶指定,三個判斷,優(yōu)先級從高到低:1. Message里面的Callback,一個實現(xiàn)了Runnable接口的對象,其中run函數(shù)做處理工作;2. Handler里面mCallback指向的一個實現(xiàn)了Callback接口的對象,由其handleMessage進(jìn)行處理;3. 處理消息Handler對象對應(yīng)的類繼承并實現(xiàn)了其中handleMessage函數(shù),通過這個實現(xiàn)的handleMessage函數(shù)處理消息。???

?

?

Android的消息機(jī)制(一)

?

android 有一種叫消息隊列的說法,這里我們可以這樣理解:假如一個隧道就是一個消息隊列,那么里面的每一部汽車就是一個一個消息,這里我們先忽略掉超車等種種因素,只那么先進(jìn)隧道的車將會先出,這個機(jī)制跟我們android 的消息機(jī)制是一樣的。

一、????角色描述

1.Looper:(相當(dāng)于隧道)一個線程可以產(chǎn)生一個Looper對象,由它來管理此線程里的Message Queue(車隊,消息隧道)

2.Handler:你可以構(gòu)造Handler對象來與Looper溝通,以便push新消息到Message Queue里;或者接收Looper(Message Queue取出)所送來的消息。

3. Message Queue(消息隊列):用來存放線程放入的消息。

4.線程:UI thread通常就是main thread,而Android啟動程序時會替它建立一個Message Queue

每一個線程里可含有一個Looper對象以及一個MessageQueue數(shù)據(jù)結(jié)構(gòu)。在你的應(yīng)用程序里,可以定義Handler的子類別來接收Looper所送出的消息。

?

在你的Android程序里,新誕生一個線程,或執(zhí)行(Thread)時,并不會自動建立其Message Loop

Android里并沒有GlobalMessage Queue數(shù)據(jù)結(jié)構(gòu),例如,不同APK里的對象不能透過Massage Queue來交換訊息(Message)

例如:線程AHandler對象可以傳遞消息給別的線程,讓別的線程BC等能送消息來給線程A(存于AMessage Queue)

線程AMessage Queue里的訊息,只有線程A所屬的對象可以處理。

使用Looper.myLooper可以取得當(dāng)前線程的Looper對象。

使用mHandler = new EevntHandler(Looper.myLooper());可用來構(gòu)造當(dāng)前線程的Handler對象;其中,EevntHandler是自已實現(xiàn)的Handler的子類別。

使用mHandler = new EevntHandler(Looper.getMainLooper());可誕生用來處理main線程的Handler對象;其中,EevntHandler是自已實現(xiàn)的Handler的子類別。

?

這樣描述可能太抽像,下面舉幾個實際的例子來說明:

二、????舉例

1.??同線程內(nèi)不同組件間的消息傳遞

Looper類用來管理特定線程內(nèi)對象之間的消息交換(Message Exchange)。你的應(yīng)用程序可以產(chǎn)生許多個線程。而一個線程可以有許多個組件,這些組件之間常常需要互相交換訊息。如果有這種需要,您可以替線程構(gòu)造一個Looper對象,來擔(dān)任訊息交換的管理工作。Looper對象會建立一個MessageQueue數(shù)據(jù)結(jié)構(gòu)來存放各對象傳來的消息(包括UI事件或System事件等)。如下圖:

?

每一個線程里可含有一個Looper對象以及一個MessageQueue數(shù)據(jù)結(jié)構(gòu)。在你的應(yīng)用程序里,可以定義Handler的子類別來接收Looper所送出的消息。

同線程不同組件之間的消息傳遞:

publicclass?Activity1extends?Activityimplements?OnClickListener{

?????? Buttonbutton?=null;

?????? TextViewtext?=null;

??????@Override

??????protectedvoid?onCreate(Bundle savedInstanceState) {

?????????????super.onCreate(savedInstanceState);

????????????? setContentView(R.layout.activity1);????????

?????????????button?= (Button)findViewById(R.id.btn);

?????????????button.setOnClickListener(this);

?????????????text?= (TextView)findViewById(R.id.content);

?????? }

??????publicvoid?onClick(View v) {

?????????????switch?(v.getId()) {

?????????????case?R.id.btn:

???????????????????? Looper looper = Looper.myLooper();//取得當(dāng)前線程里的looper

???????????????????? MyHandler mHandler =new?MyHandler(looper);//構(gòu)造一個handler使之可與looper通信

????????????????????//buton等組件可以由mHandler將消息傳給looper后,再放入messageQueue中,同時mHandler也可以接受來自looper消息

???????????????????? mHandler.removeMessages(0);

???????????????????? String msgStr ="主線程不同組件通信:消息來自button";

???????????????????? Message m = mHandler.obtainMessage(1, 1, 1, msgStr);//構(gòu)造要傳遞的消息

???????????????????? mHandler.sendMessage(m);//發(fā)送消息:系統(tǒng)會自動調(diào)用handleMessage方法來處理消息

????????????????????break;

?????????????? }????????????

?????? }?????

??????privateclass?MyHandlerextends?Handler{?????????????

?????????????public?MyHandler(Looper looper){

????????????????????super(looper);

????????????? }

?????????????@Override

?????????????publicvoid?handleMessage(Message msg) {//處理消息

????????????????????text.setText(msg.obj.toString());

????????????? }????????????

?????? }

}

?

說明:

此程序啟動時,當(dāng)前線程(即主線程, main thread)已誕生了一個Looper對象,并且有了一個MessageQueue數(shù)據(jù)結(jié)構(gòu)。

??? looper = Looper.myLooper ();?

調(diào)用Looper類別的靜態(tài)myLooper()函數(shù),以取得目前線程里的Looper對象.

mHandler = new MyHandler (looper);

構(gòu)造一個MyHandler對象來與Looper溝通。Activity等對象可以藉由MyHandler對象來將消息傳給Looper,然后放入MessageQueue里;MyHandler對象也扮演Listener的角色,可接收Looper對象所送來的消息。

Message m = mHandler.obtainMessage(1, 1, 1, obj);

先構(gòu)造一個Message對象,并將數(shù)據(jù)存入對象里。

mHandler.sendMessage(m);

就透過mHandler對象而將消息m傳給Looper,然后放入MessageQueue里。

此時,Looper對象看到MessageQueue里有消息m,就將它廣播出去,mHandler對象接到此訊息時,會呼叫其handleMessage()函數(shù)來處理,于是輸出"This my message!"于畫面上,

Android消息處理機(jī)制(二)

角色綜述(回顧):

?? (1)UI thread通常就是main thread,而Android啟動程序時會替它建立一個MessageQueue

(2)當(dāng)然需要一個Looper對象,來管理該MessageQueue

(3)我們可以構(gòu)造Handler對象來push新消息到Message Queue里;或者接收Looper(Message Queue取出)所送來的消息。

(4)線程AHandler對象可以傳遞給別的線程,讓別的線程BC等能送訊息來給線程A(存于AMessage Queue)

(5)線程AMessage Queue里的消息,只有線程A所屬的對象可以處理。

?

子線程傳遞消息給主線程

publicclass?Activity2extends?Activityimplements?OnClickListener{

?????? Buttonbutton?=null;

?????? TextViewtext?=null;

?????? MyHandlermHandler?=null;

?????? Threadthread?;

??????@Override

??????protectedvoid?onCreate(Bundle savedInstanceState) {

?????????????super.onCreate(savedInstanceState);

????????????? setContentView(R.layout.activity1);????????

?????????????button?= (Button)findViewById(R.id.btn);

?????????????button.setOnClickListener(this);

?????????????text?= (TextView)findViewById(R.id.content);

?????? }

??????publicvoid?onClick(View v) {

?????????????switch?(v.getId()) {

?????????????case?R.id.btn:

????????????????????thread?=new?MyThread();

????????????????????thread.start();

????????????????????break;

????????????? }????????????

?????? }?????

??????privateclass?MyHandlerextends?Handler{?????????????

?????????????public?MyHandler(Looper looper){

????????????????????super(looper);

????????????? }

?????????????@Override

?????????????publicvoid?handleMessage(Message msg) {//處理消息

????????????????????text.setText(msg.obj.toString());

????????????? }????????????

?????? }

??????privateclass?MyThreadextends?Thread{

?????????????@Override

?????????????publicvoid?run() {

???????????????????? Looper curLooper = Looper.myLooper();

???????????????????? Looper mainLooper = Looper.getMainLooper();

???????????????????? String msg ;

????????????????????if(curLooper==null){

???????????????????????????mHandler?=new?MyHandler(mainLooper);

??????????????????????????? msg ="curLooper is null";

???????????????????? }else{

???????????????????????????mHandler?=new?MyHandler(curLooper);

??????????????????????????? msg ="This is curLooper";

???????????????????? }

????????????????????mHandler.removeMessages(0);

???????????????????? Message m =mHandler.obtainMessage(1, 1, 1, msg);

????????????????????mHandler.sendMessage(m);

????????????? }????????????

?????? }

}

說明:

Android會自動替主線程建立Message Queue。在這個子線程里并沒有建立Message Queue。所以,myLooper值為null,而mainLooper則指向主線程里的Looper。于是,執(zhí)行到:

mHandler = new MyHandler (mainLooper);

mHandler屬于主線程。

?? mHandler.sendMessage(m);

就將m消息存入到主線程的Message Queue里。mainLooper看到Message Queue里有訊息,就會作出處理,于是由主線程執(zhí)行到mHandlerhandleMessage()來處理消息。

用Android線程間通信的Message機(jī)制

在Android下面也有多線程的概念,在C/C++中,子線程可以是一個函數(shù),一般都是一個帶有循環(huán)的函數(shù),來處理某些數(shù)據(jù),優(yōu)先線程只是一個復(fù)雜的運(yùn)算過程,所以可能不需要while循環(huán),運(yùn)算完成,函數(shù)結(jié)束,線程就銷毀。對于那些需要控制的線程,一般我們都是和互斥鎖相互關(guān)聯(lián),從而來控制線程的進(jìn)度,一般我們創(chuàng)建子線程,一種線程是很常見的,那就是帶有消息循環(huán)的線程。
消息循環(huán)是一個很有用的線程方式,曾經(jīng)自己用C在Linux下面實現(xiàn)一個消息循環(huán)的機(jī)制,往消息隊列里添加數(shù)據(jù),然后異步的等待消息的返回。當(dāng)消息隊列為空的時候就會掛起線程,等待新的消息的加入。這是一個很通用的機(jī)制。
在Android,這里的線程分為有消息循環(huán)的線程和沒有消息循環(huán)的線程,有消息循環(huán)的線程一般都會有一個Looper,這個事android的新概念。我們的主線程(UI線程)就是一個消息循環(huán)的線程。針對這種消息循環(huán)的機(jī)制,我們引入一個新的機(jī)制Handle,我們有消息循環(huán),就要往消息循環(huán)里面發(fā)送相應(yīng)的消息,自定義消息一般都會有自己對應(yīng)的處理,消息的發(fā)送和清除,消息的的處理,把這些都封裝在Handle里面,注意Handle只是針對那些有Looper的線程,不管是UI線程還是子線程,只要你有Looper,我就可以往你的消息隊列里面添加?xùn)|西,并做相應(yīng)的處理。
但是這里還有一點,就是只要是關(guān)于UI相關(guān)的東西,就不能放在子線程中,因為子線程是不能操作UI的,只能進(jìn)行數(shù)據(jù)、系統(tǒng)等其他非UI的操作。
那么什么情況下面我們的子線程才能看做是一個有Looper的線程呢?我們?nèi)绾蔚玫剿麹ooper的句柄呢?
Looper.myLooper();獲得當(dāng)前的Looper
Looper.getMainLooper () 獲得UI線程的Lopper
我們看看Handle的初始化函數(shù),如果沒有參數(shù),那么他就默認(rèn)使用的是當(dāng)前的Looper,如果有Looper參數(shù),就是用對應(yīng)的線程的Looper。
如果一個線程中調(diào)用Looper.prepare(),那么系統(tǒng)就會自動的為該線程建立一個消息隊列,然后調(diào)用 Looper.loop();之后就進(jìn)入了消息循環(huán),這個之后就可以發(fā)消息、取消息、和處理消息。這個如何發(fā)送消息和如何處理消息可以再其他的線程中通過Handle來做,但前提是我們的Hanle知道這個子線程的Looper,但是你如果不是在子線程運(yùn)行 Looper.myLooper(),一般是得不到子線程的looper的。
public void run() {
? ?? ?? ?? ?synchronized (mLock) {
? ?? ?? ?? ?? ? Looper.prepare();
? ?? ?? ?? ?? ?//do something
? ?? ?? ?? ?}
? ?? ?? ?? ?Looper.loop();
? ?? ???}
所以很多人都是這樣做的:我直接在子線程中新建handle,然后在子線程中發(fā)送消息,這樣的話就失去了我們多線程的意義了。
class myThread extends Thread{
? ?? ?? ?? ? private EHandler mHandler ;
? ?? ?? ?? ? public void run() {
? ?? ?? ?? ?? ???Looper myLooper, mainLooper;
? ?? ?? ?? ?? ???myLooper = Looper.myLooper ();
? ?? ?? ?? ?? ? mainLooper = Looper.getMainLooper ();
? ?? ?? ?? ?? ? String obj;
? ?? ?? ?? ?? ? if (myLooper == null ){
? ?? ?? ?? ?? ?? ?? ?? ? mHandler = new EHandler(mainLooper);
? ?? ?? ?? ?? ?? ?? ?? ? obj = "current thread has no looper!" ;
? ?? ?? ?? ?? ? }
? ?? ?? ?? ?? ? else {
? ?? ?? ?? ?? ?? ?? ?mHandler = new EHandler(myLooper);
? ?? ?? ?? ?? ?? ?? ?obj = "This is from current thread." ;
? ?? ?? ?? ?? ? }
? ?? ?? ?? ?? ? mHandler .removeMessages(0);
? ?? ?? ?? ?? ? Message m = mHandler .obtainMessage(1, 1, 1, obj);
? ?? ?? ?? ?? ? mHandler .sendMessage(m);
? ?? ?? ?? ? }
??}
可以讓其他的線程來控制我們的handle,可以把 private EHandler mHandler ;放在外面,這樣我們的發(fā)消息和處理消息都可以在外面來定義,這樣增加程序代碼的美觀,結(jié)構(gòu)更加清晰。
對如任何的Handle,里面必須要重載一個函數(shù)
public void handleMessage(Message msg)
這個函數(shù)就是我們的消息處理,如何處理,這里完全取決于你,然后通過 obtainMessage和 sendMessage等來生成和發(fā)送消息, removeMessages(0)來清除消息隊列。Google真是太智慧了,這種框架的產(chǎn)生,我們寫代碼更加輕松了。
有的時候,我們的子線程想去改變UI了,這個時候千萬不要再子線程中去修改,獲得UI線程的Looper,然后發(fā)送消息即可。
我們看看Goole Music App的源代碼。
在MediaPlaybackActivity.java中,我們可以看一下再OnCreate中的有這樣的兩句:
? ?? ???mAlbumArtWorker = new Worker("album art worker");
? ?? ???mAlbumArtHandler = new AlbumArtHandler(mAlbumArtWorker.getLooper());
很明顯這兩句,是構(gòu)建了一個子線程。并且這個子線程還是Looper的子線程,這里很牛逼的使用了 mAlbumArtWorker.getLooper()這個函數(shù),因為我們知道,我們能夠得到子線程的Looper的途徑只有一個:就是在子線程中調(diào)用 Looper.myLooper (),并且這個函數(shù)還要在我們perpare之后調(diào)用才能得到正確的Looper,但是他這里用了一個這樣的什么東東 getLooper,不知道它是如何實現(xiàn)的?
這里有一個大概的思路,我們在子線程的的prepare之后調(diào)用 myLooper ()這個方法,然后保存在一個成員變量中,這個getLooper就返回這個東西,但是這里會碰到多線程的一個很突出的問題,同步。我們在父線程中調(diào)用 mAlbumArtWorker.getLooper(),但是想要這個返回正確的looper就必須要求我們的子線程運(yùn)行了prepare,但是這個東西實在子線程運(yùn)行的,我們?nèi)绾伪WC呢?
我們看Google是如何實現(xiàn)的?
? ?private class Worker implements Runnable {
? ?? ???private final Object mLock = new Object();
? ?? ???private Looper mLooper;
? ?? ???
? ?? ???/**
? ?? ?? ?* Creates a worker thread with the given name. The thread
? ?? ?? ?* then runs a [email=%7B@link]{@link[/email] android.os.Looper}.
? ?? ?? ?* @param name A name for the new thread
? ?? ?? ?*/
? ?? ???Worker(String name) {
? ?? ?? ?? ?Thread t = new Thread(null, this, name);
? ?? ?? ?? ?t.setPriority(Thread.MIN_PRIORITY);
? ?? ?? ?? ?t.start();
? ?? ?? ?? ?synchronized (mLock) {
? ?? ?? ?? ?? ? while (mLooper == null) {
? ?? ?? ?? ?? ?? ???try {
? ?? ?? ?? ?? ?? ?? ?? ?mLock.wait();
? ?? ?? ?? ?? ?? ???} catch (InterruptedException ex) {
? ?? ?? ?? ?? ?? ???}
? ?? ?? ?? ?? ? }
? ?? ?? ?? ?}
? ?? ???}
? ?? ???
? ?? ???public Looper getLooper() {
? ?? ?? ?? ?return mLooper;
? ?? ???}
? ?? ???
? ?? ???public void run() {
? ?? ?? ?? ?synchronized (mLock) {
? ?? ?? ?? ?? ? Looper.prepare();
? ?? ?? ?? ?? ? mLooper = Looper.myLooper();
? ?? ?? ?? ?? ? mLock.notifyAll();
? ?? ?? ?? ?}
? ?? ?? ?? ?Looper.loop();
? ?? ???}
? ?? ???
? ?? ???public void quit() {
? ?? ?? ?? ?mLooper.quit();
? ?? ???}
? ? }
我們知道,一個線程類的構(gòu)造函數(shù)是在主線程中完成的,所以在我們的 Worker的構(gòu)造函數(shù)中我們創(chuàng)佳一個線程,然后讓這個線程運(yùn)行,這一這個線程的創(chuàng)建是指定一個 Runnabl,這里就是我們的Worker本身,在主線程調(diào)用 t.start();,這后,我們子線程已經(jīng)創(chuàng)建,并且開始執(zhí)行work的run方法。然后下面的代碼很藝術(shù):
synchronized (mLock) {
? ?? ?? ?? ?? ? while (mLooper == null) {
? ?? ?? ?? ?? ?? ???try {
? ?? ?? ?? ?? ?? ?? ?? ?mLock.wait();
? ?? ?? ?? ?? ?? ???} catch (InterruptedException ex) {
? ?? ?? ?? ?? ?? ???}
? ?? ?? ?? ?? ? }
? ?? ?? ?? ?}
我們開始等待我們的子線程給mLooper賦值,如果不賦值我們就繼續(xù)等,然后我們的子線程在運(yùn)行run方法之后,在給 mLooper賦值之后,通知worker夠著函數(shù)中的wait,然后我們的構(gòu)造函數(shù)才能完成,所以我們說:
mAlbumArtWorker = new Worker("album art worker");
這句本身就是阻塞的,它創(chuàng)建了一個子線程,開啟了子線程,并且等待子線程給mLooper賦值,賦值完成之后,這個函數(shù)才返回,這樣才能保證我們的子線程的Looper的獲取絕對是正確的,這個構(gòu)思很有創(chuàng)意。值得借鑒

Android中Handler的使用方法——在子線程中更新界面

本文主要介紹Android的Handler的使用方法。Handler可以發(fā)送Messsage和Runnable對象到與其相關(guān)聯(lián)的線程的消息隊列。每個Handler對象與創(chuàng)建它的線程相關(guān)聯(lián),并且每個Handler對象只能與一個線程相關(guān)聯(lián)。

1.??? Handler一般有兩種用途:1)執(zhí)行計劃任務(wù),你可以再預(yù)定的實現(xiàn)執(zhí)行某些任務(wù),可以模擬定時器。2)線程間通信。在Android的應(yīng)用啟動時,會創(chuàng)建一個主線程,主線程會創(chuàng)建一個消息隊列來處理各種消息。當(dāng)你創(chuàng)建子線程時,你可以再你的子線程中拿到父線程中創(chuàng)建的Handler對象,就可以通過該對象向父線程的消息隊列發(fā)送消息了。由于Android要求在UI線程中更新界面,因此,可以通過該方法在其它線程中更新界面。
◆ 通過Runnable在子線程中更新界面的例子

1.○ 在onCreate中創(chuàng)建Handler?
public class HandlerTestApp extends Activity {?
??????? Handler mHandler;?
??????? TextView mText;?
??????? /** Called when the activity is first created. */?
?????? @Override?
?????? public void onCreate(Bundle savedInstanceState) {?
?????????? super.onCreate(savedInstanceState);?
?????????? setContentView(R.layout.main);?
?????????? mHandler = new Handler();//創(chuàng)建Handler?
?????????? mText = (TextView) findViewById(R.id.text0);//一個TextView?
?????? }?
???? ○ 構(gòu)建Runnable對象,在runnable中更新界面,此處,我們修改了TextView的文字.此處需要說明的是,Runnable對象可以再主線程中創(chuàng)建,也可以再子線程中創(chuàng)建。我們此處是在子線程中創(chuàng)建的。?
???? Runnable mRunnable0 = new Runnable()?
??? {?
??????????????? @Override?
??????????????? public void run() {?
??????????????????????? mText.setText("This is Update from ohter thread, Mouse DOWN");
??????????????? }?
??? };?
???? ○ 創(chuàng)建子線程,在線程的run函數(shù)中,我們向主線程的消息隊列發(fā)送了一個runnable來更新界面。

??? private void updateUIByRunnable(){?
????????? new Thread()??
???????? {??
?????????????? //Message msg = mHandler.obtainMessage();??
????????????? public void run()??
???????????? {?

?????????????????? //mText.setText("This is Update from ohter thread, Mouse DOWN");//這句將拋出異常
???????????????????mHandler.post(mRunnable0);??
???????????? }??
???????? }.start();

???? }

◆ 用Message在子線程中來更新界面

1.??? 用Message更新界面與Runnable更新界面類似,只是需要修改幾個地方。
??? ○ 實現(xiàn)自己的Handler,對消息進(jìn)行處理

??? private class?MyHandler?extends Handler
??? {?

??????? @Override?
??????? public void handleMessage(Message msg) {?
??????????? super.handleMessage(msg);?
??????????? switch(msg.what)?
??????????? {?
??????????? case?UPDATE://在收到消息時,對界面進(jìn)行更新?
??????????????? mText.setText("This update by message");?
??????????????? break;?
??????????? }?
??????? }?
??? }

??? ○ 在新的線程中發(fā)送消息?????
??? private void updateByMessage()?
??? {?
??????? //匿名對象?
???????? new Thread()?
???????? {?
??????????????? public void run()?
??????????????? {?
??????????????????? //mText.setText("This is Update from ohter thread, Mouse DOWN");

??????????????????? //UPDATE是一個自己定義的整數(shù),代表了消息ID?
??????????????????? Message msg = mHandler.obtainMessage(UPDATE);
????????????????????mHandler.sendMessage(msg);
??????????????? }?
???????? }.start();?
??? }

?

---------------------------------------------------------------------------------------------華麗的分割線----------------------------------------------------------------------------------------------

?

?

android的消息處理有三個核心類:Looper,Handler和Message。其實還有一個Message Queue(消息隊列),但是MQ被封裝到Looper里面了,我們不會直接與MQ打交道,因此我沒將其作為核心類。下面一一介紹:

線程的魔法師 Looper

Looper的字面意思是“循環(huán)者”,它被設(shè)計用來使一個普通線程變成Looper線程。所謂Looper線程就是循環(huán)工作的線程。在程序開發(fā)中(尤其是GUI開發(fā)中),我們經(jīng)常會需要一個線程不斷循環(huán),一旦有新任務(wù)則執(zhí)行,執(zhí)行完繼續(xù)等待下一個任務(wù),這就是Looper線程。使用Looper類創(chuàng)建Looper線程很簡單:

[java]?view plaincopyprint?
  • <pre?name="code"?class="java">public?class?LooperThread?extends?Thread?{??
  • ??
  • ?@Override??
  • public?void?run()?{??
  • ????????//?將當(dāng)前線程初始化為Looper線程???
  • ????????Looper.prepare();??
  • ????????//?...其他處理,如實例化handler???
  • ????????//?開始循環(huán)處理消息隊列???
  • ????????Looper.loop();??
  • ????}??
  • }??
  • 通過上面兩行核心代碼,你的線程就升級為Looper線程了!!!是不是很神奇?讓我們放慢鏡頭,看看這兩行代碼各自做了什么。

    1)Looper.prepare()

    通過上圖可以看到,現(xiàn)在你的線程中有一個Looper對象,它的內(nèi)部維護(hù)了一個消息隊列MQ。注意,一個Thread只能有一個Looper對象,為什么呢?咱們來看源碼。

    [java]?view plaincopyprint?
  • public?class?Looper?{??
  • ????//?每個線程中的Looper對象其實是一個ThreadLocal,即線程本地存儲(TLS)對象???
  • ????private?static?final?ThreadLocal?sThreadLocal?=?new?ThreadLocal();??
  • ??
  • ????//?Looper內(nèi)的消息隊列???
  • ????final?MessageQueue?mQueue;??
  • ??
  • ????//?當(dāng)前線程???
  • ????Thread?mThread;??
  • ??
  • ????//?。。。其他屬性???
  • ??
  • ????//?每個Looper對象中有它的消息隊列,和它所屬的線程???
  • ????private?Looper()?{??
  • ????????mQueue?=?new?MessageQueue();??
  • ????????mRun?=?true;??
  • ????????mThread?=?Thread.currentThread();??
  • ????}??
  • ??
  • ????//?我們調(diào)用該方法會在調(diào)用線程的TLS中創(chuàng)建Looper對象???
  • ????public?static?final?void?prepare()?{??
  • ????????if?(sThreadLocal.get()?!=?null)?{??
  • ????????????//?試圖在有Looper的線程中再次創(chuàng)建Looper將拋出異常???
  • ????????????throw?new?RuntimeException("Only?one?Looper?may?be?created?per?thread");??
  • ????????}??
  • ????????sThreadLocal.set(new?Looper());??
  • ????}??
  • ????//?其他方法???
  • }??
  • 通過源碼,prepare()背后的工作方式一目了然,其核心就是將looper對象定義為ThreadLocal。如果你還不清楚什么是ThreadLocal,請參考《理解ThreadLocal》。

    2)Looper.loop()

    調(diào)用loop方法后,Looper線程就開始真正工作了,它不斷從自己的MQ中取出隊頭的消息(也叫任務(wù))執(zhí)行。其源碼分析如下:

    [java]?view plaincopyprint?
  • public?static?final?void?loop()?{??
  • ????????Looper?me?=?myLooper();??//得到當(dāng)前線程Looper???
  • ????????MessageQueue?queue?=?me.mQueue;??//得到當(dāng)前l(fā)ooper的MQ???
  • ????????//?這兩行沒看懂=?=?不過不影響理解???
  • ????????Binder.clearCallingIdentity();??
  • ??
  • ????????final?long?ident?=?Binder.clearCallingIdentity();??
  • ??
  • ????????//?開始循環(huán)???
  • ????????while?(true)?{??
  • ????????????Message?msg?=?queue.next();?//?取出message???
  • ????????????if?(msg?!=?null)?{??
  • ????????????????if?(msg.target?==?null)?{??
  • ????????????????????//?message沒有target為結(jié)束信號,退出循環(huán)???
  • ????????????????????return;??
  • ????????????????}??
  • ????????????????//?日志。。。???
  • ????????????????if?(me.mLogging!=?null)?me.mLogging.println(??
  • ????????????????????????">>>>>?Dispatching?to?"?+?msg.target?+?"?"??
  • ????????????????????????+?msg.callback?+?":?"?+?msg.what??
  • ????????????????????????);??
  • ????????????????//?非常重要!將真正的處理工作交給message的target,即后面要講的handler???
  • ????????????????msg.target.dispatchMessage(msg);??
  • ????????????????//?還是日志。。。???
  • ????????????????if?(me.mLogging!=?null)?me.mLogging.println(??
  • ????????????????????????"<<<<<?Finished?to????"?+?msg.target?+?"?"??
  • ????????????????????????+?msg.callback);??
  • ????????????????????????????????//?下面沒看懂,同樣不影響理解???
  • ????????????????final?long?newIdent?=?Binder.clearCallingIdentity();??
  • ????????????????if?(ident?!=?newIdent)?{??
  • ????????????????????Log.wtf("Looper",?"Thread?identity?changed?from?0x"??
  • ????????????????????????????+?Long.toHexString(ident)?+?"?to?0x"??
  • ????????????????????????????+?Long.toHexString(newIdent)?+?"?while?dispatching?to?"??
  • ????????????????????????????+?msg.target.getClass().getName()?+?"?"+?msg.callback?+?"?what="?+?msg.what);??
  • ????????????????}??
  • ????????????????//?回收message資源???
  • ????????????????msg.recycle();??
  • ????????????}??
  • ????????}??
  • ????}??

  • 除了prepare()和loop()方法,Looper類還提供了一些有用的方法,比如

    Looper.myLooper()得到當(dāng)前線程looper對象:

    [java]?view plaincopyprint?
  • public?static?final?Looper?myLooper()?{??
  • ????????//?在任意線程調(diào)用Looper.myLooper()返回的都是那個線程的looper???
  • ????????return?(Looper)sThreadLocal.get();??
  • }??
  • getThread()得到looper對象所屬線程:

    [java]?view plaincopyprint?
  • public?Thread?getThread()?{??
  • ????????return?mThread;??
  • }??
  • quit()方法結(jié)束looper循環(huán):

    [java]?view plaincopyprint?
  • public?void?quit()?{??
  • ????????//?創(chuàng)建一個空的message,它的target為NULL,表示結(jié)束循環(huán)消息???
  • ????????Message?msg?=?Message.obtain();??
  • ????????//?發(fā)出消息???
  • ????????mQueue.enqueueMessage(msg,?0);??
  • }??
  • 到此為止,你應(yīng)該對Looper有了基本的了解,總結(jié)幾點:

    1.每個線程有且最多只能有一個Looper對象,它是一個ThreadLocal

    2.Looper內(nèi)部有一個消息隊列,loop()方法調(diào)用后線程開始不斷從隊列中取出消息執(zhí)行

    3.Looper使一個線程變成Looper線程。

    那么,我們?nèi)绾瓮鵐Q上添加消息呢?下面有請Handler!(掌聲~~~)

    異步處理大師 Handler

    什么是handler?handler扮演了往MQ上添加消息和處理消息的角色(只處理由自己發(fā)出的消息),即通知MQ它要執(zhí)行一個任務(wù)(sendMessage),并在loop到自己的時候執(zhí)行該任務(wù)(handleMessage),整個過程是異步的。handler創(chuàng)建時會關(guān)聯(lián)一個looper,默認(rèn)的構(gòu)造方法將關(guān)聯(lián)當(dāng)前線程的looper,不過這也是可以set的。默認(rèn)的構(gòu)造方法:

    [java]?view plaincopyprint?
  • public?class?handler?{??
  • ????final?MessageQueue?mQueue;??//?關(guān)聯(lián)的MQ???
  • ??
  • ????final?Looper?mLooper;??//?關(guān)聯(lián)的looper???
  • ??
  • ????final?Callback?mCallback;???
  • ??
  • ????//?其他屬性???
  • ??
  • ????public?Handler()?{??
  • ????????//?沒看懂,直接略過,,,???
  • ????????if?(FIND_POTENTIAL_LEAKS)?{??
  • ????????????final?Class<??extends?Handler>?klass?=?getClass();??
  • ????????????if?((klass.isAnonymousClass()?||?klass.isMemberClass()?||?klass.isLocalClass())?&&??
  • ????????????????????(klass.getModifiers()?&?Modifier.STATIC)?==?0)?{??
  • ????????????????Log.w(TAG,?"The?following?Handler?class?should?be?static?or?leaks?might?occur:?"?+??
  • ????????????????????klass.getCanonicalName());??
  • ????????????}??
  • ????????}??
  • ????????//?默認(rèn)將關(guān)聯(lián)當(dāng)前線程的looper???
  • ????????mLooper?=?Looper.myLooper();??
  • ????????//?looper不能為空,即該默認(rèn)的構(gòu)造方法只能在looper線程中使用???
  • ????????if?(mLooper?==?null)?{??
  • ????????????throw?new?RuntimeException(??
  • ????????????????"Can't?create?handler?inside?thread?that?has?not?called?Looper.prepare()");??
  • ????????}??
  • ????????//?重要!!!直接把關(guān)聯(lián)looper的MQ作為自己的MQ,因此它的消息將發(fā)送到關(guān)聯(lián)looper的MQ上???
  • ????????mQueue?=?mLooper.mQueue;??
  • ????????mCallback?=?null;??
  • ????}??
  • ????//?其他方法???
  • }??
  • 下面我們就可以為之前的LooperThread類加入Handler:

    [java]?view plaincopyprint?
  • public?class?LooperThread?extends?Thread?{??
  • ????private?Handler?handler1;??
  • ??
  • ????private?Handler?handler2;??
  • ??
  • ????@Override??
  • ????public?void?run()?{??
  • ????????//?將當(dāng)前線程初始化為Looper線程???
  • ????????Looper.prepare();??
  • ????????//?實例化兩個handler???
  • ????????handler1?=?new?Handler();??
  • ????????handler2?=?new?Handler();??
  • ????????//?開始循環(huán)處理消息隊列???
  • ????????Looper.loop();??
  • ????}??
  • }??
  • 加入handler后的效果如下圖:

    可以看到,一個線程可以有多個Handler,但是只能有一個Looper!

    Handler發(fā)送消息

    有了handler之后,我們就可以使用?post(Runnable),postAtTime(Runnable, long),postDelayed(Runnable, long),sendEmptyMessage(int),sendMessage(Message),sendMessageAtTime(Message, long)和sendMessageDelayed(Message, long)這些方法向MQ上發(fā)送消息了。光看這些API你可能會覺得handler能發(fā)兩種消息,一種是Runnable對象,一種是message對象,這是直觀的理解,但其實post發(fā)出的Runnable對象最后都被封裝成message對象了,見源碼:

    [java]?view plaincopyprint?
  • <pre?name="code"?class="java"><span?style="color:#000000;">public?final?boolean?post(Runnable?r)??
  • {??
  • ????//?注意getPostMessage(r)將runnable封裝成message???
  • ????return??sendMessageDelayed(getPostMessage(r),?0);??
  • }??
  • ??
  • private?final?Message?getPostMessage(Runnable?r)?{??
  • ??
  • ?????Message?m?=?Message.obtain();??//得到空的message???
  • ?????m.callback?=?r;??//將runnable設(shè)為message的callback???
  • ?????return?m;??
  • }??
  • ??
  • public?boolean?sendMessageAtTime(Message?msg,?long?uptimeMillis)??
  • {??
  • ?????boolean?sent?=?false;??
  • ?????MessageQueue?queue?=?mQueue;??
  • ?????if?(queue?!=?null)?{??
  • ?????????msg.target?=?this;??//?message的target必須設(shè)為該handler!???
  • ?????????sent?=?queue.enqueueMessage(msg,?uptimeMillis);??
  • ?????}??
  • ?????else?{??
  • ?????????RuntimeException?e?=?new?RuntimeException(??
  • ?????????????this?+?"?sendMessageAtTime()?called?with?no?mQueue");??
  • ?????????Log.w("Looper",?e.getMessage(),?e);??
  • ??????}??
  • ??????return?sent;??
  • }</span>??
  • 其他方法就不羅列了,總之通過handler發(fā)出的message有如下特點:

    1.message.target為該handler對象,這確保了looper執(zhí)行到該message時能找到處理它的handler,即loop()方法中的關(guān)鍵代碼

    msg.target.dispatchMessage(msg);

    2.post發(fā)出的message,其callback為Runnable對象

    Handler處理消息

    說完了消息的發(fā)送,再來看下handler如何處理消息。消息的處理是通過核心方法dispatchMessage(Message?msg)與鉤子方法handleMessage(Message?msg)完成的,見源碼

    [java]?view plaincopyprint?
  • //?處理消息,該方法由looper調(diào)用???
  • public?void?dispatchMessage(Message?msg)?{??
  • ????if?(msg.callback?!=?null)?{??
  • ????????//?如果message設(shè)置了callback,即runnable消息,處理callback!???
  • ????????handleCallback(msg);??
  • ????}?else?{??
  • ????????//?如果handler本身設(shè)置了callback,則執(zhí)行callback???
  • ????????if?(mCallback?!=?null)?{??
  • ?????????????/*?這種方法允許讓activity等來實現(xiàn)Handler.Callback接口,避免了自己編寫handler重寫handleMessage方法。見http://alex-yang-xiansoftware-com.iteye.com/blog/850865?*/??
  • ????????????if?(mCallback.handleMessage(msg))?{??
  • ????????????????return;??
  • ????????????}??
  • ????????}??
  • ????????//?如果message沒有callback,則調(diào)用handler的鉤子方法handleMessage???
  • ????????handleMessage(msg);??
  • ????}??
  • }??
  • ??
  • ????//?處理runnable消息???
  • private?final?void?handleCallback(Message?message)?{??
  • ????message.callback.run();??//直接調(diào)用run方法!???
  • }??
  • ??
  • //?由子類實現(xiàn)的鉤子方法???
  • public?void?handleMessage(Message?msg)?{??
  • }??
  • 可以看到,除了handleMessage(Message?msg)和Runnable對象的run方法由開發(fā)者實現(xiàn)外(實現(xiàn)具體邏輯),handler的內(nèi)部工作機(jī)制對開發(fā)者是透明的。這正是handler API設(shè)計的精妙之處!

    Handler的用處

    我在小標(biāo)題中將handler描述為“異步處理大師”,這歸功于Handler擁有下面兩個重要的特點:

    1.handler可以在任意線程發(fā)送消息,這些消息會被添加到關(guān)聯(lián)的MQ上。

    ??????????????

    2.handler是在它關(guān)聯(lián)的looper線程中處理消息的。

    這就解決了android最經(jīng)典的不能在其他非主線程中更新UI的問題。android的主線程也是一個looper線程(looper在android中運(yùn)用很廣),我們在其中創(chuàng)建的handler默認(rèn)將關(guān)聯(lián)主線程MQ。因此,利用handler的一個solution就是在activity中創(chuàng)建handler并將其引用傳遞給worker thread,worker thread執(zhí)行完任務(wù)后使用handler發(fā)送消息通知activity更新UI。(過程如圖)

    當(dāng)然,handler能做的遠(yuǎn)遠(yuǎn)不僅如此,由于它能post Runnable對象,它還能與Looper配合實現(xiàn)經(jīng)典的Pipeline Thread(流水線線程)模式。請參考此文《Android Guts: Intro to Loopers and Handlers》

    封裝任務(wù) Message

    在整個消息處理機(jī)制中,message又叫task,封裝了任務(wù)攜帶的信息和處理該任務(wù)的handler。message的用法比較簡單,這里不做總結(jié)了。但是有這么幾點需要注意(待補(bǔ)充):

    1.盡管Message有public的默認(rèn)構(gòu)造方法,但是你應(yīng)該通過Message.obtain()來從消息池中獲得空消息對象,以節(jié)省資源。

    2.如果你的message只需要攜帶簡單的int信息,請優(yōu)先使用Message.arg1和Message.arg2來傳遞信息,這比用Bundle更省內(nèi)存

    3.擅用message.what來標(biāo)識信息,以便用不同方式處理message。

    ?

    ?

    ---------------------------------------------------------------------------------------------華麗的分割線----------------------------------------------------------------------------------------------

    ?

    ??????? 我們知道,Android應(yīng)用程序是通過消息來驅(qū)動的,即在應(yīng)用程序的主線程(UI線程)中有一個消息循環(huán),負(fù)責(zé)處理消息隊列中的消息。我們也知道,Android應(yīng)用程序是支持多線程的,即可以創(chuàng)建子線程來執(zhí)行一些計算型的任務(wù),那么,這些子線程能不能像應(yīng)用程序的主線程一樣具有消息循環(huán)呢?這些子線程又能不能往應(yīng)用程序的主線程中發(fā)送消息呢?本文將分析Android應(yīng)用程序線程消息處理模型,為讀者解答這兩個問題

    ?? ? ? ?在開發(fā)Android應(yīng)用程序中,有時候我們需要在應(yīng)用程序中創(chuàng)建一些常駐的子線程來不定期地執(zhí)行一些不需要與應(yīng)用程序界面交互的計算型的任務(wù)。如果這些子線程具有消息循環(huán),那么它們就能夠常駐在應(yīng)用程序中不定期的執(zhí)行一些計算型任務(wù)了:當(dāng)我們需要用這些子線程來執(zhí)行任務(wù)時,就往這個子線程的消息隊列中發(fā)送一個消息,然后就可以在子線程的消息循環(huán)中執(zhí)行我們的計算型任務(wù)了。我們在前面一篇文章Android系統(tǒng)默認(rèn)Home應(yīng)用程序(Launcher)的啟動過程源代碼分析中,介紹Launcher的啟動過程時,在Step 15(LauncherModel.startLoader)中,Launcher就是通過往一個子線程的消息隊列中發(fā)送一個消息(sWorker.post(mLoaderTask)),然后子線程就會在它的消息循環(huán)中處理這個消息的時候執(zhí)行從PackageManagerService中獲取系統(tǒng)中已安裝應(yīng)用程序的信息列表的任務(wù),即調(diào)用Step 16中的LoaderTask.run函數(shù)。

    ?? ? ? ?在開發(fā)Android應(yīng)用程序中,有時候我們又需要在應(yīng)用程序中創(chuàng)建一些子線程來執(zhí)行一些需要與應(yīng)用程序界面進(jìn)交互的計算型任務(wù)。典型的應(yīng)用場景是當(dāng)我們要從網(wǎng)上下載文件時,為了不使主線程被阻塞,我們通常創(chuàng)建一個子線程來負(fù)責(zé)下載任務(wù),同時,在下載的過程,將下載進(jìn)度以百分比的形式在應(yīng)用程序的界面上顯示出來,這樣就既不會阻塞主線程的運(yùn)行,又能獲得良好的用戶體驗。但是,我們知道,Android應(yīng)用程序的子線程是不可以操作主線程的UI的,那么,這個負(fù)責(zé)下載任務(wù)的子線程應(yīng)該如何在應(yīng)用程序界面上顯示下載的進(jìn)度呢?如果我們能夠在子線程中往主線程的消息隊列中發(fā)送消息,那么問題就迎刃而解了,因為發(fā)往主線程消息隊列的消息最終是由主線程來處理的,在處理這個消息的時候,我們就可以在應(yīng)用程序界面上顯示下載進(jìn)度了。

    ?? ? ? ?上面提到的這兩種情況,Android系統(tǒng)都為我們提供了完善的解決方案,前者可以通過使用HandlerThread類來實現(xiàn),而后者可以使用AsyncTask類來實現(xiàn),本文就詳細(xì)這兩個類是如何實現(xiàn)的。不過,為了更好地理解HandlerThread類和AsyncTask類的實現(xiàn),我們先來看看應(yīng)用程序的主線程的消息循環(huán)模型是如何實現(xiàn)的。

    ?? ? ? ?1. 應(yīng)用程序主線程消息循環(huán)模型

    ?? ? ? ?在前面一篇文章Android應(yīng)用程序進(jìn)程啟動過程的源代碼分析一文中,我們已經(jīng)分析應(yīng)用程序進(jìn)程(主線程)的啟動過程了,這里主要是針對它的消息循環(huán)模型作一個總結(jié)。當(dāng)運(yùn)行在Android應(yīng)用程序框架層中的ActivityManagerService決定要為當(dāng)前啟動的應(yīng)用程序創(chuàng)建一個主線程的時候,它會在ActivityManagerService中的startProcessLocked成員函數(shù)調(diào)用Process類的靜態(tài)成員函數(shù)start為當(dāng)前應(yīng)用程序創(chuàng)建一個主線程:

    view plain
  • public?final?class?ActivityManagerService?extends?ActivityManagerNative??????
  • ????????implements?Watchdog.Monitor,?BatteryStatsImpl.BatteryCallback?{??????
  • ??????
  • ????......??????
  • ??????
  • ????private?final?void?startProcessLocked(ProcessRecord?app,??????
  • ????????????????String?hostingType,?String?hostingNameStr)?{??????
  • ??????
  • ????????......??????
  • ??????
  • ????????try?{??????
  • ????????????int?uid?=?app.info.uid;??????
  • ????????????int[]?gids?=?null;??????
  • ????????????try?{??????
  • ????????????????gids?=?mContext.getPackageManager().getPackageGids(??????
  • ????????????????????app.info.packageName);??????
  • ????????????}?catch?(PackageManager.NameNotFoundException?e)?{??????
  • ????????????????......??????
  • ????????????}??????
  • ??????????????????
  • ????????????......??????
  • ??????
  • ????????????int?debugFlags?=?0;??????
  • ??????????????????
  • ????????????......??????
  • ??????????????????
  • ????????????int?pid?=?Process.start("android.app.ActivityThread",??????
  • ????????????????mSimpleProcessManagement???app.processName?:?null,?uid,?uid,??????
  • ????????????????gids,?debugFlags,?null);??????
  • ??????????????????
  • ????????????......??????
  • ??????
  • ????????}?catch?(RuntimeException?e)?{??????
  • ??????????????????
  • ????????????......??????
  • ??????
  • ????????}??????
  • ????}??????
  • ??????
  • ????......??????
  • ??????
  • }??????
  • ?? ? ? ?這里我們主要關(guān)注Process.start函數(shù)的第一個參數(shù)“android.app.ActivityThread”,它表示要在當(dāng)前新建的線程中加載android.app.ActivityThread類,并且調(diào)用這個類的靜態(tài)成員函數(shù)main作為應(yīng)用程序的入口點。ActivityThread類定義在frameworks/base/core/java/android/app/ActivityThread.java文件中:

    view plain
  • public?final?class?ActivityThread?{????
  • ????......????
  • ????
  • ????public?static?final?void?main(String[]?args)?{????
  • ????????......??
  • ????
  • ????????Looper.prepareMainLooper();????
  • ???????????
  • ????????......????
  • ????
  • ????????ActivityThread?thread?=?new?ActivityThread();????
  • ????????thread.attach(false);????
  • ????
  • ????????......???
  • ????????Looper.loop();????
  • ????
  • ????????......???
  • ????
  • ????????thread.detach();????
  • ????????......????
  • ????}????
  • ????
  • ????......????
  • }????
  • ?? ? ? ?在這個main函數(shù)里面,除了創(chuàng)建一個ActivityThread實例外,就是在進(jìn)行消息循環(huán)了。

    ?? ? ? ?在進(jìn)行消息循環(huán)之前,首先會通過Looper類的靜態(tài)成員函數(shù)prepareMainLooper為當(dāng)前線程準(zhǔn)備一個消息循環(huán)對象。Looper類定義在frameworks/base/core/java/android/os/Looper.java文件中:

    view plain
  • public?class?Looper?{??
  • ????......??
  • ??
  • ????//?sThreadLocal.get()?will?return?null?unless?you've?called?prepare().??
  • ????private?static?final?ThreadLocal?sThreadLocal?=?new?ThreadLocal();??
  • ??
  • ????......??
  • ??
  • ????private?static?Looper?mMainLooper?=?null;??
  • ??
  • ????......??
  • ??
  • ????public?static?final?void?prepare()?{??
  • ????????if?(sThreadLocal.get()?!=?null)?{??
  • ????????????throw?new?RuntimeException("Only?one?Looper?may?be?created?per?thread");??
  • ????????}??
  • ????????sThreadLocal.set(new?Looper());??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????public?static?final?void?prepareMainLooper()?{??
  • ????????prepare();??
  • ????????setMainLooper(myLooper());??
  • ????????......??
  • ????}??
  • ??
  • ????private?synchronized?static?void?setMainLooper(Looper?looper)?{??
  • ????????mMainLooper?=?looper;??
  • ????}??
  • ??
  • ????public?synchronized?static?final?Looper?getMainLooper()?{??
  • ????????return?mMainLooper;??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????public?static?final?Looper?myLooper()?{??
  • ????????return?(Looper)sThreadLocal.get();??
  • ????}??
  • ??
  • ????......??
  • }??
  • ?? ? ? ?Looper類的靜態(tài)成員函數(shù)prepareMainLooper是專門應(yīng)用程序的主線程調(diào)用的,應(yīng)用程序的其它子線程都不應(yīng)該調(diào)用這個函數(shù)來在本線程中創(chuàng)建消息循環(huán)對象,而應(yīng)該調(diào)用prepare函數(shù)來在本線程中創(chuàng)建消息循環(huán)對象,下一節(jié)我們介紹一個線程類HandlerThread 時將會看到。

    ?? ? ? ?為什么要為應(yīng)用程序的主線程專門準(zhǔn)備一個創(chuàng)建消息循環(huán)對象的函數(shù)呢?這是為了讓其它地方能夠方便地通過Looper類的getMainLooper函數(shù)來獲得應(yīng)用程序主線程中的消息循環(huán)對象。獲得應(yīng)用程序主線程中的消息循環(huán)對象又有什么用呢?一般就是為了能夠向應(yīng)用程序主線程發(fā)送消息了。

    ?? ? ? ?在prepareMainLooper函數(shù)中,首先會調(diào)用prepare函數(shù)在本線程中創(chuàng)建一個消息循環(huán)對象,然后將這個消息循環(huán)對象放在線程局部變量sThreadLocal中:

    view plain
  • sThreadLocal.set(new?Looper());??
  • ?? ? ? ?接著再將這個消息循環(huán)對象通過調(diào)用setMainLooper函數(shù)來保存在Looper類的靜態(tài)成員變量mMainLooper中:

    view plain
  • mMainLooper?=?looper;??
  • ?? ? ? 這樣,其它地方才可以調(diào)用getMainLooper函數(shù)來獲得應(yīng)用程序主線程中的消息循環(huán)對象。

    ?? ? ? 消息循環(huán)對象創(chuàng)建好之后,回到ActivityThread類的main函數(shù)中,接下來,就是要進(jìn)入消息循環(huán)了:

    view plain
  • Looper.loop();???
  • ?? ? ? ?Looper類具體是如何通過loop函數(shù)進(jìn)入消息循環(huán)以及處理消息隊列中的消息,可以參考前面一篇文章Android應(yīng)用程序消息處理機(jī)制(Looper、Handler)分析,這里就不再分析了,我們只要知道ActivityThread類中的main函數(shù)執(zhí)行了這一步之后,就為應(yīng)用程序的主線程準(zhǔn)備好消息循環(huán)就可以了。

    ?? ? ? ?2. 應(yīng)用程序子線程消息循環(huán)模型

    ?? ? ? ?在Java框架中,如果我們想在當(dāng)前應(yīng)用程序中創(chuàng)建一個子線程,一般就是通過自己實現(xiàn)一個類,這個類繼承于Thread類,然后重載Thread類的run函數(shù),把我們想要在這個子線程執(zhí)行的任務(wù)都放在這個run函數(shù)里面實現(xiàn)。最后實例這個自定義的類,并且調(diào)用它的start函數(shù),這樣一個子線程就創(chuàng)建好了,并且會調(diào)用這個自定義類的run函數(shù)。但是當(dāng)這個run函數(shù)執(zhí)行完成后,子線程也就結(jié)束了,它沒有消息循環(huán)的概念。

    ?? ? ? ?前面說過,有時候我們需要在應(yīng)用程序中創(chuàng)建一些常駐的子線程來不定期地執(zhí)行一些計算型任務(wù),這時候就可以考慮使用Android系統(tǒng)提供的HandlerThread類了,它具有創(chuàng)建具有消息循環(huán)功能的子線程的作用。

    ?? ? ? ?HandlerThread類實現(xiàn)在frameworks/base/core/java/android/os/HandlerThread.java文件中,這里我們通過使用情景來有重點的分析它的實現(xiàn)。

    ?? ? ? ?在前面一篇文章Android系統(tǒng)默認(rèn)Home應(yīng)用程序(Launcher)的啟動過程源代碼分析中,我們分析了Launcher的啟動過程,其中在Step 15(LauncherModel.startLoader)和Step 16(LoaderTask.run)中,Launcher會通過創(chuàng)建一個HandlerThread類來實現(xiàn)在一個子線程加載系統(tǒng)中已經(jīng)安裝的應(yīng)用程序的任務(wù):

    view plain
  • public?class?LauncherModel?extends?BroadcastReceiver?{??
  • ????......??
  • ??
  • ????private?LoaderTask?mLoaderTask;??
  • ??
  • ????private?static?final?HandlerThread?sWorkerThread?=?new?HandlerThread("launcher-loader");??
  • ????static?{??
  • ????????sWorkerThread.start();??
  • ????}??
  • ????private?static?final?Handler?sWorker?=?new?Handler(sWorkerThread.getLooper());??
  • ??
  • ????......??
  • ??
  • ????public?void?startLoader(Context?context,?boolean?isLaunching)?{????
  • ????????......????
  • ??
  • ????????synchronized?(mLock)?{????
  • ????????????......????
  • ??
  • ????????????//?Don't?bother?to?start?the?thread?if?we?know?it's?not?going?to?do?anything????
  • ????????????if?(mCallbacks?!=?null?&&?mCallbacks.get()?!=?null)?{????
  • ????????????????......??
  • ??
  • ????????????????mLoaderTask?=?new?LoaderTask(context,?isLaunching);????
  • ????????????????sWorker.post(mLoaderTask);????
  • ????????????}????
  • ????????}????
  • ????}????
  • ??
  • ????......??
  • ??
  • ????private?class?LoaderTask?implements?Runnable?{????
  • ????????......????
  • ??
  • ????????public?void?run()?{????
  • ????????????......????
  • ??
  • ????????????keep_running:?{????
  • ????????????????......????
  • ??
  • ????????????????//?second?step????
  • ????????????????if?(loadWorkspaceFirst)?{????
  • ????????????????????......????
  • ????????????????????loadAndBindAllApps();????
  • ????????????????}?else?{????
  • ????????????????????......????
  • ????????????????}????
  • ??
  • ????????????????......????
  • ????????????}????
  • ??
  • ????????????......????
  • ????????}????
  • ??
  • ????????......????
  • ????}???
  • ??
  • ????......??
  • }??
  • ?? ? ? ?在這個LauncherModel類中,首先創(chuàng)建了一個HandlerThread對象:

    view plain
  • private?static?final?HandlerThread?sWorkerThread?=?new?HandlerThread("launcher-loader");??
  • ?? ? ? ?接著調(diào)用它的start成員函數(shù)來啟動一個子線程:

    view plain
  • static?{??
  • ????sWorkerThread.start();??
  • }??
  • ?? ? ? ?接著還通過這個HandlerThread對象的getLooper函數(shù)來獲得這個子線程中的消息循環(huán)對象,并且使用這個消息循環(huán)創(chuàng)建對象來創(chuàng)建一個Handler:

    view plain
  • private?static?final?Handler?sWorker?=?new?Handler(sWorkerThread.getLooper());??
  • ?? ? ? ?有了這個Handler對象sWorker之后,我們就可以往這個子線程中發(fā)送消息,然后在處理這個消息的時候執(zhí)行加載系統(tǒng)中已經(jīng)安裝的應(yīng)用程序的任務(wù)了,在startLoader函數(shù)中:

    view plain
  • mLoaderTask?=?new?LoaderTask(context,?isLaunching);????
  • sWorker.post(mLoaderTask);????
  • ?? ? ? ?這里的mLoaderTask是一個LoaderTask對象,它實現(xiàn)了Runnable接口,因此,可以把這個LoaderTask對象作為參數(shù)傳給sWorker.post函數(shù)。在sWorker.post函數(shù)里面,會把這個LoaderTask對象封裝成一個消息,并且放入這個子線程的消息隊列中去。當(dāng)這個子線程的消息循環(huán)處理這個消息的時候,就會調(diào)用這個LoaderTask對象的run函數(shù),因此,我們就可以在LoaderTask對象的run函數(shù)中通過調(diào)用loadAndBindAllApps來執(zhí)行加載系統(tǒng)中已經(jīng)安裝的應(yīng)用程序的任務(wù)了。

    ?? ? ? ?了解了HanderThread類的使用方法之后,我們就可以重點地來分析它的實現(xiàn)了:

    view plain
  • public?class?HandlerThread?extends?Thread?{??
  • ????......??
  • ????private?Looper?mLooper;??
  • ??
  • ????public?HandlerThread(String?name)?{??
  • ????????super(name);??
  • ????????......??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????public?void?run()?{??
  • ????????......??
  • ????????Looper.prepare();??
  • ????????synchronized?(this)?{??
  • ????????????mLooper?=?Looper.myLooper();??
  • ????????????......??
  • ????????}??
  • ????????......??
  • ????????Looper.loop();??
  • ????????......??
  • ????}??
  • ??
  • ????public?Looper?getLooper()?{??
  • ????????......??
  • ????????return?mLooper;??
  • ????}??
  • ??
  • ????......??
  • }??
  • ?? ? ? ?首先我們看到的是,Handler類繼承了Thread類,因此,通過它可以在應(yīng)用程序中創(chuàng)建一個子線程,其次我們看到在它的run函數(shù)中,會進(jìn)入一個消息循環(huán)中,因此,這個子線程可以常駐在應(yīng)用程序中,直到它接收收到一個退出消息為止。

    ?? ? ? ?在run函數(shù)中,首先是調(diào)用Looper類的靜態(tài)成員函數(shù)prepare來準(zhǔn)備一個消息循環(huán)對象:

    view plain
  • Looper.prepare();??
  • ?? ? ? ?然后通過Looper類的myLooper成員函數(shù)將這個子線程中的消息循環(huán)對象保存在HandlerThread類中的成員變量mLooper中:

    view plain
  • mLooper?=?Looper.myLooper();??
  • ?? ? ? ?這樣,其它地方就可以方便地通過它的getLooper函數(shù)來獲得這個消息循環(huán)對象了,有了這個消息循環(huán)對象后,就可以往這個子線程的消息隊列中發(fā)送消息,通知這個子線程執(zhí)行特定的任務(wù)了。

    ?? ? ? ?最在這個run函數(shù)通過Looper類的loop函數(shù)進(jìn)入消息循環(huán)中:

    view plain
  • Looper.loop();??
  • ?? ? ? ?這樣,一個具有消息循環(huán)的應(yīng)用程序子線程就準(zhǔn)備就緒了。

    ?? ? ? ?HandlerThread類的實現(xiàn)雖然非常簡單,當(dāng)然這得益于Java提供的Thread類和Android自己本身提供的Looper類,但是它的想法卻非常周到,為應(yīng)用程序開發(fā)人員提供了很大的方便。
    ?? ? ? ?3. 需要與UI交互的應(yīng)用程序子線程消息模型

    ?? ? ? ?前面說過,我們開發(fā)應(yīng)用程序的時候,經(jīng)常中需要創(chuàng)建一個子線程來在后臺執(zhí)行一個特定的計算任務(wù),而在這個任務(wù)計算的過程中,需要不斷地將計算進(jìn)度或者計算結(jié)果展現(xiàn)在應(yīng)用程序的界面中。典型的例子是從網(wǎng)上下載文件,為了不阻塞應(yīng)用程序的主線程,我們開辟一個子線程來執(zhí)行下載任務(wù),子線程在下載的同時不斷地將下載進(jìn)度在應(yīng)用程序界面上顯示出來,這樣做出來程序就非常友好。由于子線程不能直接操作應(yīng)用程序的UI,因此,這時候,我們就可以通過往應(yīng)用程序的主線程中發(fā)送消息來通知應(yīng)用程序主線程更新界面上的下載進(jìn)度。因為類似的這種情景在實際開發(fā)中經(jīng)常碰到,Android系統(tǒng)為開發(fā)人員提供了一個異步任務(wù)類(AsyncTask)來實現(xiàn)上面所說的功能,即它會在一個子線程中執(zhí)行計算任務(wù),同時通過主線程的消息循環(huán)來獲得更新應(yīng)用程序界面的機(jī)會。

    ?? ? ? ?為了更好地分析AsyncTask的實現(xiàn),我們先舉一個例子來說明它的用法。在前面一篇文章Android系統(tǒng)中的廣播(Broadcast)機(jī)制簡要介紹和學(xué)習(xí)計劃中,我們開發(fā)了一個應(yīng)用程序Broadcast,其中使用了AsyncTask來在一個線程在后臺在執(zhí)行計數(shù)任務(wù),計數(shù)過程通過廣播(Broadcast)來將中間結(jié)果在應(yīng)用程序界面上顯示出來。在這個例子中,使用廣播來在應(yīng)用程序主線程和子線程中傳遞數(shù)據(jù)不是最優(yōu)的方法,當(dāng)時只是為了分析Android系統(tǒng)的廣播機(jī)制而有意為之的。在本節(jié)內(nèi)容中,我們稍微這個例子作一個簡單的修改,就可以通過消息的方式來將計數(shù)過程的中間結(jié)果在應(yīng)用程序界面上顯示出來。

    ?? ? ? ?為了區(qū)別Android系統(tǒng)中的廣播(Broadcast)機(jī)制簡要介紹和學(xué)習(xí)計劃一文中使用的應(yīng)用程序Broadcast,我們將本節(jié)中使用的應(yīng)用程序命名為Counter。首先在Android源代碼工程中創(chuàng)建一個Android應(yīng)用程序工程,名字就為Counter,放在packages/experimental目錄下。關(guān)于如何獲得Android源代碼工程,請參考在Ubuntu上下載、編譯和安裝Android最新源代碼一文;關(guān)于如何在Android源代碼工程中創(chuàng)建應(yīng)用程序工程,請參考在Ubuntu上為Android系統(tǒng)內(nèi)置Java應(yīng)用程序測試Application Frameworks層的硬件服務(wù)一文。這個應(yīng)用程序工程定義了一個名為shy.luo.counter的package,這個例子的源代碼主要就是實現(xiàn)在這個目錄下的Counter.java文件中:

    view plain
  • package?shy.luo.counter;??
  • ??
  • import?android.app.Activity;??
  • import?android.content.ComponentName;??
  • import?android.content.Context;??
  • import?android.content.Intent;??
  • import?android.content.IntentFilter;??
  • import?android.os.Bundle;??
  • import?android.os.AsyncTask;??
  • import?android.util.Log;??
  • import?android.view.View;??
  • import?android.view.View.OnClickListener;??
  • import?android.widget.Button;??
  • import?android.widget.TextView;??
  • ??
  • public?class?Counter?extends?Activity?implements?OnClickListener?{??
  • ????private?final?static?String?LOG_TAG?=?"shy.luo.counter.Counter";??
  • ??
  • ????private?Button?startButton?=?null;??
  • ????private?Button?stopButton?=?null;??
  • ????private?TextView?counterText?=?null;??
  • ??
  • ????private?AsyncTask<Integer,?Integer,?Integer>?task?=?null;??
  • ????private?boolean?stop?=?false;??
  • ??
  • ????@Override??
  • ????public?void?onCreate(Bundle?savedInstanceState)?{??
  • ????????super.onCreate(savedInstanceState);??
  • ????????setContentView(R.layout.main);??
  • ??
  • ????????startButton?=?(Button)findViewById(R.id.button_start);??
  • ????????stopButton?=?(Button)findViewById(R.id.button_stop);??
  • ????????counterText?=?(TextView)findViewById(R.id.textview_counter);??
  • ??
  • ????????startButton.setOnClickListener(this);??
  • ????????stopButton.setOnClickListener(this);??
  • ??
  • ????????startButton.setEnabled(true);??
  • ????????stopButton.setEnabled(false);??
  • ??
  • ??
  • ????????Log.i(LOG_TAG,?"Main?Activity?Created.");??
  • ????}??
  • ??
  • ??
  • ????@Override??
  • ????public?void?onClick(View?v)?{??
  • ????????if(v.equals(startButton))?{??
  • ????????????if(task?==?null)?{??
  • ????????????????task?=?new?CounterTask();??
  • ????????????????task.execute(0);??
  • ??
  • ????????????????startButton.setEnabled(false);??
  • ????????????????stopButton.setEnabled(true);??
  • ????????????}??
  • ????????}?else?if(v.equals(stopButton))?{??
  • ????????????if(task?!=?null)?{??
  • ????????????????stop?=?true;??
  • ????????????????task?=?null;??
  • ??
  • ????????????????startButton.setEnabled(true);??
  • ????????????????stopButton.setEnabled(false);??
  • ????????????}??
  • ????????}??
  • ????}??
  • ??
  • ????class?CounterTask?extends?AsyncTask<Integer,?Integer,?Integer>?{??
  • ????????@Override??
  • ????????protected?Integer?doInBackground(Integer...?vals)?{??
  • ????????????Integer?initCounter?=?vals[0];??
  • ??
  • ????????????stop?=?false;??
  • ????????????while(!stop)?{??
  • ????????????????publishProgress(initCounter);??
  • ??
  • ????????????????try?{??
  • ????????????????????Thread.sleep(1000);??
  • ????????????????}?catch?(InterruptedException?e)?{??
  • ????????????????????e.printStackTrace();??
  • ????????????????}??
  • ??
  • ????????????????initCounter++;??
  • ????????????}??
  • ??
  • ????????????return?initCounter;??
  • ????????}??
  • ??
  • ????????@Override??
  • ????????protected?void?onProgressUpdate(Integer...?values)?{??
  • ????????????super.onProgressUpdate(values);??
  • ??
  • ????????????String?text?=?values[0].toString();??
  • ????????????counterText.setText(text);??
  • ????????}??
  • ??
  • ????????@Override??
  • ????????protected?void?onPostExecute(Integer?val)?{??
  • ????????????String?text?=?val.toString();??
  • ????????????counterText.setText(text);??
  • ????????}??
  • ????};??
  • }??
  • ?? ? ? ?這個計數(shù)器程序很簡單,它在界面上有兩個按鈕Start和Stop。點擊Start按鈕時,便會創(chuàng)建一個CounterTask實例task,然后調(diào)用它的execute函數(shù)就可以在應(yīng)用程序中啟動一個子線程,并且通過調(diào)用這個CounterTask類的doInBackground函數(shù)來執(zhí)行計數(shù)任務(wù)。在計數(shù)的過程中,會通過調(diào)用publishProgress函數(shù)來將中間結(jié)果傳遞到onProgressUpdate函數(shù)中去,在onProgressUpdate函數(shù)中,就可以把中間結(jié)果顯示在應(yīng)用程序界面了。點擊Stop按鈕時,便會通過設(shè)置變量stop為true,這樣,CounterTask類的doInBackground函數(shù)便會退出循環(huán),然后將結(jié)果返回到onPostExecute函數(shù)中去,在onPostExecute函數(shù),會把最終計數(shù)結(jié)果顯示在用程序界面中。

    ?? ? ? 在這個例子中,我們需要注意的是:

    ?? ? ? A. CounterTask類繼承于AsyncTask類,因此它也是一個異步任務(wù)類;

    ?? ? ? B.?CounterTask類的doInBackground函數(shù)是在后臺的子線程中運(yùn)行的,這時候它不可以操作應(yīng)用程序的界面;

    ?? ? ? C.?CounterTask類的onProgressUpdate和onPostExecute兩個函數(shù)是應(yīng)用程序的主線程中執(zhí)行,它們可以操作應(yīng)用程序的界面。

    ?? ? ? 關(guān)于C這一點的實現(xiàn)原理,我們在后面會分析到,這里我們先完整地介紹這個例子,以便讀者可以參考做一下實驗。

    ?? ? ? 接下來我們再看看應(yīng)用程序的配置文件AndroidManifest.xml:

    view plain
  • <?xml?version="1.0"?encoding="utf-8"?>??
  • <manifest?xmlns:android="http://schemas.android.com/apk/res/android"??
  • ??????package="shy.luo.counter"??
  • ??????android:versionCode="1"??
  • ??????android:versionName="1.0">??
  • ????<application?android:icon="@drawable/icon"?android:label="@string/app_name">??
  • ????????<activity?android:name=".Counter"??
  • ??????????????????android:label="@string/app_name">??
  • ????????????<intent-filter>??
  • ????????????????<action?android:name="android.intent.action.MAIN"?/>??
  • ????????????????<category?android:name="android.intent.category.LAUNCHER"?/>??
  • ????????????</intent-filter>??
  • ????????</activity>??
  • ????</application>??
  • </manifest>??
  • ?? ? ? 這個配置文件很簡單,我們就不介紹了。

    ?? ? ??再來看應(yīng)用程序的界面文件,它定義在res/layout/main.xml文件中:

    view plain
  • <?xml?version="1.0"?encoding="utf-8"?>????
  • <LinearLayout?xmlns:android="http://schemas.android.com/apk/res/android"????
  • ????android:orientation="vertical"????
  • ????android:layout_width="fill_parent"????
  • ????android:layout_height="fill_parent"?????
  • ????android:gravity="center">????
  • ????<LinearLayout????
  • ????????android:layout_width="fill_parent"????
  • ????????android:layout_height="wrap_content"????
  • ????????android:layout_marginBottom="10px"????
  • ????????android:orientation="horizontal"?????
  • ????????android:gravity="center">????
  • ????????<TextView??????
  • ????????android:layout_width="wrap_content"?????
  • ????????????android:layout_height="wrap_content"?????
  • ????????????android:layout_marginRight="4px"????
  • ????????????android:gravity="center"????
  • ????????????android:text="@string/counter">????
  • ????????</TextView>????
  • ????????<TextView??????
  • ????????????android:id="@+id/textview_counter"????
  • ????????android:layout_width="wrap_content"?????
  • ????????????android:layout_height="wrap_content"?????
  • ????????????android:gravity="center"????
  • ????????????android:text="0">????
  • ????????</TextView>????
  • ????</LinearLayout>????
  • ????<LinearLayout????
  • ????????android:layout_width="fill_parent"????
  • ????????android:layout_height="wrap_content"????
  • ????????android:orientation="horizontal"?????
  • ????????android:gravity="center">????
  • ????????<Button?????
  • ????????????android:id="@+id/button_start"????
  • ????????????android:layout_width="wrap_content"????
  • ????????????android:layout_height="wrap_content"????
  • ????????????android:gravity="center"????
  • ????????????android:text="@string/start">????
  • ????????</Button>????
  • ????????<Button?????
  • ????????????android:id="@+id/button_stop"????
  • ????????????android:layout_width="wrap_content"????
  • ????????????android:layout_height="wrap_content"????
  • ????????????android:gravity="center"????
  • ????????????android:text="@string/stop"?>????
  • ????????</Button>????
  • ?????</LinearLayout>??????
  • </LinearLayout>????
  • ?? ? ? 這個界面配置文件也很簡單,等一下我們在模擬器把這個應(yīng)用程序啟動起來后,就可以看到它的截圖了。

    ?? ? ??應(yīng)用程序用到的字符串資源文件位于res/values/strings.xml文件中:

    view plain
  • <?xml?version="1.0"?encoding="utf-8"?>????
  • <resources>????
  • ????<string?name="app_name">Counter</string>????
  • ????<string?name="counter">Counter:?</string>????
  • ????<string?name="start">Start?Counter</string>????
  • ????<string?name="stop">Stop?Counter</string>????
  • </resources>???
  • ?? ? ? 最后,我們還要在工程目錄下放置一個編譯腳本文件Android.mk:

    view plain
  • LOCAL_PATH:=?$(call?my-dir)??????????
  • include?$(CLEAR_VARS)??????????
  • ??????????
  • LOCAL_MODULE_TAGS?:=?optional??????????
  • ??????????
  • LOCAL_SRC_FILES?:=?$(call?all-subdir-java-files)??????????
  • ??????????
  • LOCAL_PACKAGE_NAME?:=?Counter??????????
  • ??????????
  • include?$(BUILD_PACKAGE)????
  • ?? ? ??接下來就要編譯了。有關(guān)如何單獨編譯Android源代碼工程的模塊,以及如何打包system.img,請參考如何單獨編譯Android源代碼中的模塊一文。
    ?? ? ? 執(zhí)行以下命令進(jìn)行編譯和打包:

    view plain
  • USER-NAME@MACHINE-NAME:~/Android$?mmm?packages/experimental/Counter????????????
  • USER-NAME@MACHINE-NAME:~/Android$?make?snod??
  • ?? ? ??這樣,打包好的Android系統(tǒng)鏡像文件system.img就包含我們前面創(chuàng)建的Counter應(yīng)用程序了。
    ?? ? ? 再接下來,就是運(yùn)行模擬器來運(yùn)行我們的例子了。關(guān)于如何在Android源代碼工程中運(yùn)行模擬器,請參考在Ubuntu上下載、編譯和安裝Android最新源代碼一文。
    ?? ? ? 執(zhí)行以下命令啟動模擬器:

    view plain
  • USER-NAME@MACHINE-NAME:~/Android$?emulator???
  • ?? ? ? 最后我們就可以在Launcher中找到Counter應(yīng)用程序圖標(biāo),把它啟動起來,點擊Start按鈕,就會看到應(yīng)用程序界面上的計數(shù)器跑起來了:

    ?? ? ? ?這樣,使用AsyncTask的例子就介紹完了,下面,我們就要根據(jù)上面對AsyncTask的使用情況來重點分析它的實現(xiàn)了。

    ?? ? ? ?AsyncTask類定義在frameworks/base/core/java/android/os/AsyncTask.java文件中:

    view plain
  • public?abstract?class?AsyncTask<Params,?Progress,?Result>?{??
  • ????......??
  • ??
  • ????private?static?final?BlockingQueue<Runnable>?sWorkQueue?=??
  • ????????????new?LinkedBlockingQueue<Runnable>(10);??
  • ??
  • ????private?static?final?ThreadFactory?sThreadFactory?=?new?ThreadFactory()?{??
  • ????????private?final?AtomicInteger?mCount?=?new?AtomicInteger(1);??
  • ??
  • ????????public?Thread?newThread(Runnable?r)?{??
  • ????????????return?new?Thread(r,?"AsyncTask?#"?+?mCount.getAndIncrement());??
  • ????????}??
  • ????};??
  • ??
  • ????......??
  • ??
  • ????private?static?final?ThreadPoolExecutor?sExecutor?=?new?ThreadPoolExecutor(CORE_POOL_SIZE,??
  • ????????MAXIMUM_POOL_SIZE,?KEEP_ALIVE,?TimeUnit.SECONDS,?sWorkQueue,?sThreadFactory);??
  • ??
  • ????private?static?final?int?MESSAGE_POST_RESULT?=?0x1;??
  • ????private?static?final?int?MESSAGE_POST_PROGRESS?=?0x2;??
  • ????private?static?final?int?MESSAGE_POST_CANCEL?=?0x3;??
  • ??
  • ????private?static?final?InternalHandler?sHandler?=?new?InternalHandler();??
  • ??
  • ????private?final?WorkerRunnable<Params,?Result>?mWorker;??
  • ????private?final?FutureTask<Result>?mFuture;??
  • ??
  • ????......??
  • ??
  • ????public?AsyncTask()?{??
  • ????????mWorker?=?new?WorkerRunnable<Params,?Result>()?{??
  • ????????????public?Result?call()?throws?Exception?{??
  • ????????????????......??
  • ????????????????return?doInBackground(mParams);??
  • ????????????}??
  • ????????};??
  • ??
  • ????????mFuture?=?new?FutureTask<Result>(mWorker)?{??
  • ????????????@Override??
  • ????????????protected?void?done()?{??
  • ????????????????Message?message;??
  • ????????????????Result?result?=?null;??
  • ??
  • ????????????????try?{??
  • ????????????????????result?=?get();??
  • ????????????????}?catch?(InterruptedException?e)?{??
  • ????????????????????android.util.Log.w(LOG_TAG,?e);??
  • ????????????????}?catch?(ExecutionException?e)?{??
  • ????????????????????throw?new?RuntimeException("An?error?occured?while?executing?doInBackground()",??
  • ????????????????????????e.getCause());??
  • ????????????????}?catch?(CancellationException?e)?{??
  • ????????????????????message?=?sHandler.obtainMessage(MESSAGE_POST_CANCEL,??
  • ????????????????????????new?AsyncTaskResult<Result>(AsyncTask.this,?(Result[])?null));??
  • ????????????????????message.sendToTarget();??
  • ????????????????????return;??
  • ????????????????}?catch?(Throwable?t)?{??
  • ????????????????????throw?new?RuntimeException("An?error?occured?while?executing?"??
  • ????????????????????????+?"doInBackground()",?t);??
  • ????????????????}??
  • ??
  • ????????????????message?=?sHandler.obtainMessage(MESSAGE_POST_RESULT,??
  • ????????????????????new?AsyncTaskResult<Result>(AsyncTask.this,?result));??
  • ????????????????message.sendToTarget();??
  • ????????????}??
  • ????????};??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????public?final?Result?get()?throws?InterruptedException,?ExecutionException?{??
  • ????????return?mFuture.get();??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????public?final?AsyncTask<Params,?Progress,?Result>?execute(Params...?params)?{??
  • ????????......??
  • ??
  • ????????mWorker.mParams?=?params;??
  • ????????sExecutor.execute(mFuture);??
  • ??
  • ????????return?this;??
  • ????}??
  • ??
  • ????......??
  • ??
  • ????protected?final?void?publishProgress(Progress...?values)?{??
  • ????????sHandler.obtainMessage(MESSAGE_POST_PROGRESS,??
  • ????????????new?AsyncTaskResult<Progress>(this,?values)).sendToTarget();??
  • ????}??
  • ??
  • ????????private?void?finish(Result?result)?{??
  • ????????????????......??
  • ????????????????onPostExecute(result);??
  • ????????????????......??
  • ????????}??
  • ??
  • ????......??
  • ??
  • ????private?static?class?InternalHandler?extends?Handler?{??
  • ????????@SuppressWarnings({"unchecked",?"RawUseOfParameterizedType"})??
  • ????????@Override??
  • ????????public?void?handleMessage(Message?msg)?{??
  • ????????????AsyncTaskResult?result?=?(AsyncTaskResult)?msg.obj;??
  • ????????????switch?(msg.what)?{??
  • ????????????????case?MESSAGE_POST_RESULT:??
  • ?????????????????//?There?is?only?one?result??
  • ?????????????????result.mTask.finish(result.mData[0]);??
  • ?????????????????break;??
  • ????????????????case?MESSAGE_POST_PROGRESS:??
  • ?????????????????result.mTask.onProgressUpdate(result.mData);??
  • ?????????????????break;??
  • ????????????????case?MESSAGE_POST_CANCEL:??
  • ?????????????????result.mTask.onCancelled();??
  • ?????????????????break;??
  • ????????????}??
  • ????????}??
  • ????}??
  • ??
  • ????private?static?abstract?class?WorkerRunnable<Params,?Result>?implements?Callable<Result>?{??
  • ????????Params[]?mParams;??
  • ????}??
  • ??
  • ????private?static?class?AsyncTaskResult<Data>?{??
  • ????????final?AsyncTask?mTask;??
  • ????????final?Data[]?mData;??
  • ??
  • ????????AsyncTaskResult(AsyncTask?task,?Data...?data)?{??
  • ????????????mTask?=?task;??
  • ????????????mData?=?data;??
  • ????????}??
  • ????}??
  • }??
  • ?? ? ? ?從AsyncTask的實現(xiàn)可以看出,當(dāng)我們第一次創(chuàng)建一個AsyncTask對象時,首先會執(zhí)行下面靜態(tài)初始化代碼創(chuàng)建一個線程池sExecutor:

    view plain
  • private?static?final?BlockingQueue<Runnable>?sWorkQueue?=??
  • ????new?LinkedBlockingQueue<Runnable>(10);??
  • ??
  • private?static?final?ThreadFactory?sThreadFactory?=?new?ThreadFactory()?{??
  • ????private?final?AtomicInteger?mCount?=?new?AtomicInteger(1);??
  • ??
  • ????public?Thread?newThread(Runnable?r)?{??
  • ????????return?new?Thread(r,?"AsyncTask?#"?+?mCount.getAndIncrement());??
  • ????}??
  • };??
  • ??
  • ......??
  • ??
  • private?static?final?ThreadPoolExecutor?sExecutor?=?new?ThreadPoolExecutor(CORE_POOL_SIZE,??
  • ????MAXIMUM_POOL_SIZE,?KEEP_ALIVE,?TimeUnit.SECONDS,?sWorkQueue,?sThreadFactory);??
  • ?? ? ? ?這里的ThreadPoolExecutor是Java提供的多線程機(jī)制之一,這里用的構(gòu)造函數(shù)原型為:

    view plain
  • ThreadPoolExecutor(int?corePoolSize,?int?maximumPoolSize,?long?keepAliveTime,?TimeUnit?unit,???
  • ????BlockingQueue<Runnable>?workQueue,?ThreadFactory?threadFactory)??
  • ?? ? ? ?各個參數(shù)的意義如下:

    ?? ? ? ?corePoolSize -- 線程池的核心線程數(shù)量

    ?? ? ? ?maximumPoolSize -- 線程池的最大線程數(shù)量

    ?? ? ? ?keepAliveTime -- 若線程池的線程數(shù)數(shù)量大于核心線程數(shù)量,那么空閑時間超過keepAliveTime的線程將被回收

    ?? ? ? ?unit -- 參數(shù)keepAliveTime使用的時間單位

    ?? ? ? ?workerQueue -- 工作任務(wù)隊列

    ?? ? ? ?threadFactory -- 用來創(chuàng)建線程池中的線程
    ?? ? ? ?簡單來說,ThreadPoolExecutor的運(yùn)行機(jī)制是這樣的:每一個工作任務(wù)用一個Runnable對象來表示,當(dāng)我們要把一個工作任務(wù)交給這個線程池來執(zhí)行的時候,就通過調(diào)用ThreadPoolExecutor的execute函數(shù)來把這個工作任務(wù)加入到線程池中去。此時,如果線程池中的線程數(shù)量小于corePoolSize,那么就會調(diào)用threadFactory接口來創(chuàng)建一個新的線程并且加入到線程池中去,再執(zhí)行這個工作任務(wù);如果線程池中的線程數(shù)量等于corePoolSize,但是工作任務(wù)隊列workerQueue未滿,則把這個工作任務(wù)加入到工作任務(wù)隊列中去等待執(zhí)行;如果線程池中的線程數(shù)量大于corePoolSize,但是小于maximumPoolSize,并且工作任務(wù)隊列workerQueue已經(jīng)滿了,那么就會調(diào)用threadFactory接口來創(chuàng)建一個新的線程并且加入到線程池中去,再執(zhí)行這個工作任務(wù);如果線程池中的線程量已經(jīng)等于maximumPoolSize了,并且工作任務(wù)隊列workerQueue也已經(jīng)滿了,這個工作任務(wù)就被拒絕執(zhí)行了。

    ?? ? ? ?創(chuàng)建好了線程池后,再創(chuàng)建一個消息處理器:

    view plain
  • private?static?final?InternalHandler?sHandler?=?new?InternalHandler();??
  • ?? ? ? ?注意,這行代碼是在應(yīng)用程序的主線程中執(zhí)行的,因此,這個消息處理器sHandler內(nèi)部引用的消息循環(huán)對象looper是應(yīng)用程序主線程的消息循環(huán)對象,消息處理器的實現(xiàn)機(jī)制具體可以參考前面一篇文章Android應(yīng)用程序消息處理機(jī)制(Looper、Handler)分析。

    ?? ? ? ?AsyncTask類的靜態(tài)初始化代碼執(zhí)行完成之后,才開始創(chuàng)建AsyncTask對象,即執(zhí)行AsyncTask類的構(gòu)造函數(shù):

    view plain
  • public?AsyncTask()?{??
  • ????mWorker?=?new?WorkerRunnable<Params,?Result>()?{??
  • ????????public?Result?call()?throws?Exception?{??
  • ????????????......??
  • ????????????return?doInBackground(mParams);??
  • ????????}??
  • ????};??
  • ??
  • ????mFuture?=?new?FutureTask<Result>(mWorker)?{??
  • ????????@Override??
  • ????????protected?void?done()?{??
  • ????????????Message?message;??
  • ????????????Result?result?=?null;??
  • ??
  • ????????????try?{??
  • ????????????????result?=?get();??
  • ????????????}?catch?(InterruptedException?e)?{??
  • ????????????????android.util.Log.w(LOG_TAG,?e);??
  • ????????????}?catch?(ExecutionException?e)?{??
  • ????????????????throw?new?RuntimeException("An?error?occured?while?executing?doInBackground()",??
  • ????????????????????e.getCause());??
  • ????????????}?catch?(CancellationException?e)?{??
  • ????????????????message?=?sHandler.obtainMessage(MESSAGE_POST_CANCEL,??
  • ????????????????????new?AsyncTaskResult<Result>(AsyncTask.this,?(Result[])?null));??
  • ????????????????message.sendToTarget();??
  • ????????????????return;??
  • ????????????}?catch?(Throwable?t)?{??
  • ????????????????throw?new?RuntimeException("An?error?occured?while?executing?"??
  • ????????????????????+?"doInBackground()",?t);??
  • ????????????}??
  • ??
  • ????????????message?=?sHandler.obtainMessage(MESSAGE_POST_RESULT,??
  • ????????????????new?AsyncTaskResult<Result>(AsyncTask.this,?result));??
  • ????????????message.sendToTarget();??
  • ????????}??
  • ????};??
  • }??
  • ?? ? ? ?在AsyncTask類的構(gòu)造函數(shù)里面,主要是創(chuàng)建了兩個對象,分別是一個WorkerRunnable對象mWorker和一個FutureTask對象mFuture。

    ?? ? ? ?WorkerRunnable類實現(xiàn)了Runnable接口,此外,它的內(nèi)部成員變量mParams用于保存從AsyncTask對象的execute函數(shù)傳進(jìn)來的參數(shù)列表:

    view plain
  • private?static?abstract?class?WorkerRunnable<Params,?Result>?implements?Callable<Result>?{??
  • ????Params[]?mParams;??
  • }??
  • ?? ? ? ?FutureTask類也實現(xiàn)了Runnable接口,所以它可以作為一個工作任務(wù)通過調(diào)用AsyncTask類的execute函數(shù)添加到sExecuto線程池中去:

    view plain
  • public?final?AsyncTask<Params,?Progress,?Result>?execute(Params...?params)?{??
  • ????......??
  • ??
  • ????mWorker.mParams?=?params;??
  • ????sExecutor.execute(mFuture);??
  • ??
  • ????return?this;??
  • }??
  • ?? ? ? 這里的FutureTask對象mFuture是用來封裝前面的WorkerRunnable對象mWorker。當(dāng)mFuture加入到線程池中執(zhí)行時,它調(diào)用的是mWorker對象的call函數(shù):

    view plain
  • mWorker?=?new?WorkerRunnable<Params,?Result>()?{??
  • ????public?Result?call()?throws?Exception?{??
  • ???????????......??
  • ???????????return?doInBackground(mParams);??
  • ????????}??
  • };??
  • ?? ? ? ?在call函數(shù)里面,會調(diào)用AsyncTask類的doInBackground函數(shù)來執(zhí)行真正的任務(wù),這個函數(shù)是要由AsyncTask的子類來實現(xiàn)的,注意,這個函數(shù)是在應(yīng)用程序的子線程中執(zhí)行的,它不可以操作應(yīng)用程序的界面。

    ?? ? ? ?我們可以通過mFuture對象來操作當(dāng)前執(zhí)行的任務(wù),例如查詢當(dāng)前任務(wù)的狀態(tài),它是正在執(zhí)行中,還是完成了,還是被取消了,如果是完成了,還可以通過它獲得任務(wù)的執(zhí)行結(jié)果,如果還沒有完成,可以取消任務(wù)的執(zhí)行。

    ?? ? ? ?當(dāng)工作任務(wù)mWorker執(zhí)行完成的時候,mFuture對象中的done函數(shù)就會被被調(diào)用,根據(jù)任務(wù)的完成狀況,執(zhí)行相應(yīng)的操作,例如,如果是因為異常而完成時,就會拋異常,如果是正常完成,就會把任務(wù)執(zhí)行結(jié)果封裝成一個AsyncTaskResult對象:

    view plain
  • private?static?class?AsyncTaskResult<Data>?{??
  • ????final?AsyncTask?mTask;??
  • ????final?Data[]?mData;??
  • ??
  • ????AsyncTaskResult(AsyncTask?task,?Data...?data)?{??
  • ????????mTask?=?task;??
  • ????????mData?=?data;??
  • ????}??
  • }??
  • ?? ? ? ?其中,成員變量mData保存的是任務(wù)執(zhí)行結(jié)果,而成員變量mTask指向前面我們創(chuàng)建的AsyncTask對象。
    ?? ? ? ?最后把這個AsyncTaskResult對象封裝成一個消息,并且通過消息處理器sHandler加入到應(yīng)用程序主線程的消息隊列中:

    view plain
  • message?=?sHandler.obtainMessage(MESSAGE_POST_RESULT,??
  • ????new?AsyncTaskResult<Result>(AsyncTask.this,?result));??
  • message.sendToTarget();??
  • ?? ? ? ?這個消息最終就會在InternalHandler類的handleMessage函數(shù)中處理了:

    view plain
  • private?static?class?InternalHandler?extends?Handler?{??
  • ????@SuppressWarnings({"unchecked",?"RawUseOfParameterizedType"})??
  • ????@Override??
  • ????public?void?handleMessage(Message?msg)?{??
  • ????????AsyncTaskResult?result?=?(AsyncTaskResult)?msg.obj;??
  • ????????switch?(msg.what)?{??
  • ????????case?MESSAGE_POST_RESULT:??
  • ????????????//?There?is?only?one?result??
  • ????????????result.mTask.finish(result.mData[0]);??
  • ????????????break;??
  • ????????......??
  • ????????}??
  • ????}??
  • }??
  • ?? ? ? ?在這個函數(shù)里面,最終會調(diào)用前面創(chuàng)建的這個AsyncTask對象的finish函數(shù)來進(jìn)一步處理:

    view plain
  • private?void?finish(Result?result)?{??
  • ???????......??
  • ???????onPostExecute(result);??
  • ???????......??
  • }??
  • ?? ? ? ?這個函數(shù)調(diào)用AsyncTask類的onPostExecute函數(shù)來進(jìn)一步處理,AsyncTask類的onPostExecute函數(shù)一般是要由其子類來重載的,注意,這個函數(shù)是在應(yīng)用程序的主線程中執(zhí)行的,因此,它可以操作應(yīng)用程序的界面。
    ?? ? ? ?在任務(wù)執(zhí)行的過程當(dāng)中,即執(zhí)行doInBackground函數(shù)時候,可能通過調(diào)用publishProgress函數(shù)來將中間結(jié)果封裝成一個消息發(fā)送到應(yīng)用程序主線程中的消息隊列中去:

    view plain
  • protected?final?void?publishProgress(Progress...?values)?{??
  • ????sHandler.obtainMessage(MESSAGE_POST_PROGRESS,??
  • ????????new?AsyncTaskResult<Progress>(this,?values)).sendToTarget();??
  • }??
  • ?? ? ? ?這個消息最終也是由InternalHandler類的handleMessage函數(shù)來處理的:

    view plain
  • private?static?class?InternalHandler?extends?Handler?{??
  • ????@SuppressWarnings({"unchecked",?"RawUseOfParameterizedType"})??
  • ????@Override??
  • ????public?void?handleMessage(Message?msg)?{??
  • ????????AsyncTaskResult?result?=?(AsyncTaskResult)?msg.obj;??
  • ????????switch?(msg.what)?{??
  • ????????......??
  • ????????case?MESSAGE_POST_PROGRESS:??
  • ?????????????????result.mTask.onProgressUpdate(result.mData);??
  • ?????????????????break;??
  • ????????......??
  • ????????}??
  • ????}??
  • }??
  • ?? ? ? ?這里它調(diào)用前面創(chuàng)建的AsyncTask對象的onPorgressUpdate函數(shù)來進(jìn)一步處理,這個函數(shù)一般是由AsyncTask的子類來實現(xiàn)的,注意,這個函數(shù)是在應(yīng)用程序的主線程中執(zhí)行的,因此,它和前面的onPostExecute函數(shù)一樣,可以操作應(yīng)用程序的界面。

    ?? ? ? 這樣,AsyncTask類的主要實現(xiàn)就介紹完了,結(jié)合前面開發(fā)的應(yīng)用程序Counter來分析,會更好地理解它的實現(xiàn)原理。

    ?? ? ? 至此,Android應(yīng)用程序線程消息循環(huán)模型就分析完成了,理解它有利于我們在開發(fā)Android應(yīng)用程序時,能夠充分利用多線程的并發(fā)性來提高應(yīng)用程序的性能以及獲得良好的用戶體驗。

    ?

    轉(zhuǎn)自:http://www.cnblogs.com/qingblog/archive/2012/06/27/2566021.html

    總結(jié)

    以上是生活随笔為你收集整理的Android消息处理机制的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    国产亚洲精品久久久久久国模美 | 久久久久人妻一区精品色欧美 | 一本一道久久综合久久 | 欧美日韩人成综合在线播放 | 欧美人与禽猛交狂配 | 无码人妻久久一区二区三区不卡 | 精品国产成人一区二区三区 | 久久婷婷五月综合色国产香蕉 | 国产成人无码午夜视频在线观看 | 国产人妻人伦精品1国产丝袜 | 麻豆成人精品国产免费 | 免费播放一区二区三区 | 亚洲精品综合一区二区三区在线 | 亚洲小说图区综合在线 | 兔费看少妇性l交大片免费 | 国产亚av手机在线观看 | 亚洲精品国产a久久久久久 | 人人澡人摸人人添 | 无码国模国产在线观看 | 国产欧美熟妇另类久久久 | 少妇性l交大片欧洲热妇乱xxx | 精品日本一区二区三区在线观看 | 18黄暴禁片在线观看 | 欧美午夜特黄aaaaaa片 | 欧美兽交xxxx×视频 | 99久久人妻精品免费一区 | 久久久久久av无码免费看大片 | 亚洲精品成人福利网站 | 噜噜噜亚洲色成人网站 | 午夜精品一区二区三区在线观看 | 亚洲一区二区三区国产精华液 | 欧美性猛交xxxx富婆 | 亚洲综合另类小说色区 | 青春草在线视频免费观看 | 狠狠色丁香久久婷婷综合五月 | 丰满诱人的人妻3 | 国产精品久久久久影院嫩草 | 鲁一鲁av2019在线 | 成 人 网 站国产免费观看 | 在线a亚洲视频播放在线观看 | 亚洲理论电影在线观看 | 欧美人与物videos另类 | 蜜臀av在线播放 久久综合激激的五月天 | 亚洲精品一区二区三区在线观看 | 精品少妇爆乳无码av无码专区 | 在线观看国产一区二区三区 | 国产激情艳情在线看视频 | 最近的中文字幕在线看视频 | 女高中生第一次破苞av | 久久视频在线观看精品 | 人人澡人人透人人爽 | 人妻人人添人妻人人爱 | 国产激情一区二区三区 | 丰满妇女强制高潮18xxxx | 亚洲精品一区二区三区大桥未久 | 久久综合激激的五月天 | √天堂中文官网8在线 | 国产女主播喷水视频在线观看 | 三上悠亚人妻中文字幕在线 | 亚洲成av人在线观看网址 | 日本大香伊一区二区三区 | 131美女爱做视频 | 欧美日韩精品 | 亚洲精品欧美二区三区中文字幕 | 精品国产精品久久一区免费式 | 亚洲精品综合五月久久小说 | 国产精品久久久午夜夜伦鲁鲁 | 一本久久a久久精品vr综合 | 日本乱人伦片中文三区 | 婷婷六月久久综合丁香 | 动漫av一区二区在线观看 | 成人免费视频视频在线观看 免费 | 亚洲s码欧洲m码国产av | 亚洲乱码国产乱码精品精 | 国内综合精品午夜久久资源 | 亚洲乱码日产精品bd | 无码av免费一区二区三区试看 | 全球成人中文在线 | 中文字幕人妻丝袜二区 | 国精产品一区二区三区 | 久久国产精品二国产精品 | 亚洲综合另类小说色区 | 国产精品沙发午睡系列 | 日韩精品久久久肉伦网站 | 亚洲欧美国产精品久久 | 久久人人爽人人爽人人片ⅴ | 久久国产精品偷任你爽任你 | 国产精品亚洲五月天高清 | 精品国产一区二区三区四区 | 国产亚洲精品久久久久久久久动漫 | 欧美日韩一区二区综合 | 亚洲午夜福利在线观看 | 国产97在线 | 亚洲 | 国产激情一区二区三区 | 国产热a欧美热a在线视频 | 97资源共享在线视频 | 人妻天天爽夜夜爽一区二区 | 一本精品99久久精品77 | 久久久久久久女国产乱让韩 | 色婷婷欧美在线播放内射 | 无码人妻丰满熟妇区五十路百度 | 天天拍夜夜添久久精品大 | 亚洲日韩av一区二区三区中文 | 在线精品亚洲一区二区 | 三级4级全黄60分钟 | 无码国产乱人伦偷精品视频 | 久久午夜无码鲁丝片秋霞 | 国产绳艺sm调教室论坛 | 小鲜肉自慰网站xnxx | 久久综合久久自在自线精品自 | 亚洲热妇无码av在线播放 | 久久久精品成人免费观看 | 在线观看国产一区二区三区 | a在线观看免费网站大全 | 中文字幕色婷婷在线视频 | 无套内谢老熟女 | 亚洲精品国产第一综合99久久 | 巨爆乳无码视频在线观看 | 国产激情无码一区二区app | 丰满少妇人妻久久久久久 | 欧美性生交xxxxx久久久 | 中文字幕久久久久人妻 | 亚洲国产成人av在线观看 | 蜜桃av抽搐高潮一区二区 | 国产亚洲精品久久久久久国模美 | 东京无码熟妇人妻av在线网址 | 日日天干夜夜狠狠爱 | 好屌草这里只有精品 | 激情综合激情五月俺也去 | 亚洲精品国产精品乱码不卡 | 亚洲人交乣女bbw | 无码一区二区三区在线观看 | 亚洲成av人综合在线观看 | 成人免费视频视频在线观看 免费 | 东京热无码av男人的天堂 | 国产精品福利视频导航 | 久久成人a毛片免费观看网站 | 色偷偷人人澡人人爽人人模 | 玩弄人妻少妇500系列视频 | 最新国产乱人伦偷精品免费网站 | 丰满少妇女裸体bbw | 无遮无挡爽爽免费视频 | 一个人看的视频www在线 | 99精品视频在线观看免费 | 秋霞成人午夜鲁丝一区二区三区 | 国产成人久久精品流白浆 | 荫蒂被男人添的好舒服爽免费视频 | 97夜夜澡人人爽人人喊中国片 | 精品乱子伦一区二区三区 | 久久97精品久久久久久久不卡 | 精品无码一区二区三区爱欲 | 精品无码一区二区三区爱欲 | 青青青手机频在线观看 | 色妞www精品免费视频 | 少妇愉情理伦片bd | 蜜臀av在线播放 久久综合激激的五月天 | 免费看男女做好爽好硬视频 | 一区二区三区乱码在线 | 欧洲 | 奇米影视7777久久精品人人爽 | 人人澡人人透人人爽 | 久久99精品久久久久久 | 日韩人妻无码一区二区三区久久99 | 99久久无码一区人妻 | 人人澡人摸人人添 | 久久aⅴ免费观看 | 亚洲中文字幕无码中字 | 日韩亚洲欧美中文高清在线 | 动漫av一区二区在线观看 | 欧美日韩久久久精品a片 | 无码纯肉视频在线观看 | 国产网红无码精品视频 | 97无码免费人妻超级碰碰夜夜 | 一二三四在线观看免费视频 | 久久综合九色综合欧美狠狠 | 免费人成网站视频在线观看 | 大屁股大乳丰满人妻 | 东京无码熟妇人妻av在线网址 | 成人免费视频在线观看 | 国产精品毛片一区二区 | 成年美女黄网站色大免费视频 | 国精产品一区二区三区 | 亚洲精品一区二区三区大桥未久 | 呦交小u女精品视频 | 国产极品视觉盛宴 | 国产精品高潮呻吟av久久 | aⅴ在线视频男人的天堂 | 国产97色在线 | 免 | 日本一区二区三区免费高清 | 国产亚av手机在线观看 | 日本饥渴人妻欲求不满 | 日本一区二区三区免费高清 | 国产精品va在线观看无码 | 中文字幕无码视频专区 | 国产精品毛多多水多 | 男女猛烈xx00免费视频试看 | 久久精品国产大片免费观看 | 日日橹狠狠爱欧美视频 | 欧美日韩视频无码一区二区三 | 夜先锋av资源网站 | 亚洲欧美国产精品专区久久 | 国产av人人夜夜澡人人爽麻豆 | 亚洲精品久久久久久一区二区 | 伊人久久婷婷五月综合97色 | 国产精品永久免费视频 | 高潮毛片无遮挡高清免费 | aⅴ亚洲 日韩 色 图网站 播放 | 色五月丁香五月综合五月 | 台湾无码一区二区 | 欧美日韩在线亚洲综合国产人 | 国产人妻大战黑人第1集 | 欧美国产亚洲日韩在线二区 | 国产精品美女久久久网av | 欧美xxxxx精品 | 六十路熟妇乱子伦 | 久激情内射婷内射蜜桃人妖 | 偷窥日本少妇撒尿chinese | 精品无码一区二区三区的天堂 | 中文字幕久久久久人妻 | 欧美 丝袜 自拍 制服 另类 | 午夜福利不卡在线视频 | 男女超爽视频免费播放 | 无套内射视频囯产 | 亚洲а∨天堂久久精品2021 | 少妇人妻偷人精品无码视频 | 男人扒开女人内裤强吻桶进去 | 久久午夜夜伦鲁鲁片无码免费 | 理论片87福利理论电影 | 久久99热只有频精品8 | 亚洲成a人片在线观看日本 | 97色伦图片97综合影院 | 天天做天天爱天天爽综合网 | 欧美三级不卡在线观看 | 亚洲日韩av一区二区三区四区 | 欧美熟妇另类久久久久久多毛 | 婷婷丁香六月激情综合啪 | 国产熟妇另类久久久久 | 欧美自拍另类欧美综合图片区 | 日韩av激情在线观看 | 精品亚洲成av人在线观看 | 欧美大屁股xxxxhd黑色 | 成人av无码一区二区三区 | 中文字幕人成乱码熟女app | 高潮喷水的毛片 | 无码av岛国片在线播放 | 精品无码国产自产拍在线观看蜜 | 亚洲精品久久久久中文第一幕 | 小鲜肉自慰网站xnxx | 国产人妻精品一区二区三区 | 国产无遮挡又黄又爽又色 | 99久久久国产精品无码免费 | 呦交小u女精品视频 | 亚洲性无码av中文字幕 | 亚洲va中文字幕无码久久不卡 | 欧美乱妇无乱码大黄a片 | 国产激情一区二区三区 | 国产九九九九九九九a片 | 亚洲中文字幕无码中文字在线 | 内射爽无广熟女亚洲 | 18禁止看的免费污网站 | 日韩精品乱码av一区二区 | 国产成人精品优优av | 国产口爆吞精在线视频 | 国产精品爱久久久久久久 | 色综合久久久无码网中文 | 国产福利视频一区二区 | 精品无人国产偷自产在线 | 日日摸日日碰夜夜爽av | 国产无遮挡又黄又爽免费视频 | 装睡被陌生人摸出水好爽 | 精品久久久中文字幕人妻 | 国产精品久久国产三级国 | 黑人粗大猛烈进出高潮视频 | 蜜臀aⅴ国产精品久久久国产老师 | 无码午夜成人1000部免费视频 | 国产又爽又黄又刺激的视频 | 国内揄拍国内精品少妇国语 | 中文毛片无遮挡高清免费 | 伊人久久婷婷五月综合97色 | 亚洲国产精品一区二区第一页 | 国语自产偷拍精品视频偷 | 亚洲色www成人永久网址 | 又大又紧又粉嫩18p少妇 | 图片区 小说区 区 亚洲五月 | 日本乱人伦片中文三区 | 国产精品人妻一区二区三区四 | 少妇无套内谢久久久久 | 18黄暴禁片在线观看 | 国产内射老熟女aaaa | 大地资源中文第3页 | 亚洲精品鲁一鲁一区二区三区 | 欧美阿v高清资源不卡在线播放 | 日本熟妇人妻xxxxx人hd | 久久综合网欧美色妞网 | 国产精品久久久久7777 | 亚欧洲精品在线视频免费观看 | 伊人色综合久久天天小片 | 东北女人啪啪对白 | 中文字幕av日韩精品一区二区 | 少妇人妻大乳在线视频 | 日日橹狠狠爱欧美视频 | 国产精品自产拍在线观看 | 青青久在线视频免费观看 | 伊人久久婷婷五月综合97色 | 久9re热视频这里只有精品 | 久9re热视频这里只有精品 | 国产精品美女久久久久av爽李琼 | 未满小14洗澡无码视频网站 | 精品午夜福利在线观看 | 国产亚洲tv在线观看 | 在线观看国产一区二区三区 | 亚洲七七久久桃花影院 | 欧美丰满少妇xxxx性 | 精品人妻中文字幕有码在线 | 蜜桃视频插满18在线观看 | 亚洲码国产精品高潮在线 | 国产一区二区不卡老阿姨 | 日韩视频 中文字幕 视频一区 | 天天拍夜夜添久久精品 | 性欧美videos高清精品 | 中文字幕av无码一区二区三区电影 | 97精品国产97久久久久久免费 | 欧美 丝袜 自拍 制服 另类 | 国产精品国产自线拍免费软件 | 国产精品永久免费视频 | 一本大道伊人av久久综合 | 福利一区二区三区视频在线观看 | a在线亚洲男人的天堂 | 帮老师解开蕾丝奶罩吸乳网站 | 精品水蜜桃久久久久久久 | 国产乱人伦app精品久久 国产在线无码精品电影网 国产国产精品人在线视 | 无码人妻丰满熟妇区五十路百度 | 亚洲色www成人永久网址 | 国产成人无码午夜视频在线观看 | 日本丰满熟妇videos | 99久久精品国产一区二区蜜芽 | 亚洲色偷偷男人的天堂 | 国产亚洲精品久久久久久久 | 少妇厨房愉情理9仑片视频 | 精品一二三区久久aaa片 | 正在播放东北夫妻内射 | 婷婷综合久久中文字幕蜜桃三电影 | 激情五月综合色婷婷一区二区 | 波多野结衣一区二区三区av免费 | 色欲人妻aaaaaaa无码 | 午夜精品久久久内射近拍高清 | 中文字幕亚洲情99在线 | 国产网红无码精品视频 | 丝袜足控一区二区三区 | 亚洲成在人网站无码天堂 | 国产另类ts人妖一区二区 | 国产精品鲁鲁鲁 | 免费观看激色视频网站 | 亚洲自偷自拍另类第1页 | 国产电影无码午夜在线播放 | 中文字幕精品av一区二区五区 | 又粗又大又硬毛片免费看 | 永久免费精品精品永久-夜色 | 男人和女人高潮免费网站 | 亚洲va中文字幕无码久久不卡 | 欧美丰满少妇xxxx性 | 日本丰满熟妇videos | 97se亚洲精品一区 | 亚洲日韩av一区二区三区四区 | 蜜桃av抽搐高潮一区二区 | 欧美 亚洲 国产 另类 | 久在线观看福利视频 | 久青草影院在线观看国产 | 日本一卡二卡不卡视频查询 | 国产99久久精品一区二区 | 中文字幕无线码免费人妻 | 国产精品亚洲五月天高清 | 欧美 亚洲 国产 另类 | 日韩人妻无码中文字幕视频 | 亚洲成a人片在线观看无码3d | 小sao货水好多真紧h无码视频 | 毛片内射-百度 | 人妻少妇精品无码专区二区 | 成人免费视频视频在线观看 免费 | 永久免费精品精品永久-夜色 | 日韩 欧美 动漫 国产 制服 | 日本丰满护士爆乳xxxx | 国产精品沙发午睡系列 | 一区二区三区高清视频一 | 日韩精品无码一区二区中文字幕 | 99精品国产综合久久久久五月天 | 中文字幕 亚洲精品 第1页 | a国产一区二区免费入口 | 欧美 日韩 亚洲 在线 | 女人高潮内射99精品 | 亚洲中文字幕av在天堂 | 国产黄在线观看免费观看不卡 | 九月婷婷人人澡人人添人人爽 | 亚洲中文字幕乱码av波多ji | 中文字幕无码免费久久9一区9 | 国产国产精品人在线视 | 欧美丰满老熟妇xxxxx性 | 亚洲精品一区二区三区大桥未久 | 国产又粗又硬又大爽黄老大爷视 | 女高中生第一次破苞av | 国产成人无码区免费内射一片色欲 | 亚洲人成网站在线播放942 | 日日摸夜夜摸狠狠摸婷婷 | 2020久久香蕉国产线看观看 | 久久国产劲爆∧v内射 | 偷窥日本少妇撒尿chinese | 亚洲中文字幕在线观看 | 国产精品久久久久9999小说 | 一个人看的www免费视频在线观看 | 乱人伦人妻中文字幕无码久久网 | аⅴ资源天堂资源库在线 | 国产精品人妻一区二区三区四 | 少妇人妻av毛片在线看 | 熟女俱乐部五十路六十路av | 欧洲熟妇精品视频 | 免费乱码人妻系列无码专区 | 亚洲日韩乱码中文无码蜜桃臀网站 | 人妻有码中文字幕在线 | 无码纯肉视频在线观看 | 久久亚洲a片com人成 | 中文字幕乱码人妻无码久久 | 噜噜噜亚洲色成人网站 | 乌克兰少妇性做爰 | 高清无码午夜福利视频 | 九九久久精品国产免费看小说 | 亚洲日韩av一区二区三区中文 | 国产麻豆精品精东影业av网站 | 久久精品一区二区三区四区 | 亚洲呦女专区 | 波多野结衣av一区二区全免费观看 | 久久国内精品自在自线 | 无码精品人妻一区二区三区av | 久久人人爽人人爽人人片ⅴ | 国内精品人妻无码久久久影院蜜桃 | 色婷婷久久一区二区三区麻豆 | 性欧美videos高清精品 | 久久久精品欧美一区二区免费 | 亚洲一区二区三区无码久久 | 无码人妻黑人中文字幕 | 无码人妻精品一区二区三区不卡 | 久久久精品人妻久久影视 | 高中生自慰www网站 | 中国女人内谢69xxxx | 国产又爽又猛又粗的视频a片 | 国产极品视觉盛宴 | 欧美精品无码一区二区三区 | √天堂中文官网8在线 | 日韩人妻无码中文字幕视频 | 人妻无码αv中文字幕久久琪琪布 | 国产亚洲精品精品国产亚洲综合 | 国产亚洲美女精品久久久2020 | 国产人妻精品一区二区三区不卡 | 久久久国产精品无码免费专区 | 亚洲成av人在线观看网址 | 亚洲国产成人a精品不卡在线 | av在线亚洲欧洲日产一区二区 | 老子影院午夜精品无码 | 国产xxx69麻豆国语对白 | 亚洲国产精华液网站w | 麻豆av传媒蜜桃天美传媒 | 久热国产vs视频在线观看 | 中文字幕人妻无码一区二区三区 | 欧美人妻一区二区三区 | 水蜜桃色314在线观看 | 夜夜高潮次次欢爽av女 | 亚洲中文字幕无码中文字在线 | 国产av无码专区亚洲a∨毛片 | 亚洲国产欧美日韩精品一区二区三区 | 在线成人www免费观看视频 | 亚洲国产成人av在线观看 | 成人无码精品一区二区三区 | 国产精品嫩草久久久久 | 亚洲精品无码国产 | 国产成人精品一区二区在线小狼 | 国产av人人夜夜澡人人爽麻豆 | 色婷婷综合激情综在线播放 | 麻豆成人精品国产免费 | 久久亚洲精品成人无码 | 久久精品国产日本波多野结衣 | 中文字幕人妻丝袜二区 | 日韩在线不卡免费视频一区 | 熟妇人妻激情偷爽文 | 帮老师解开蕾丝奶罩吸乳网站 | 天堂久久天堂av色综合 | 亚洲国产成人av在线观看 | 国内综合精品午夜久久资源 | 性欧美熟妇videofreesex | 牲交欧美兽交欧美 | 久久久精品456亚洲影院 | 成 人 免费观看网站 | 日本爽爽爽爽爽爽在线观看免 | 国产一精品一av一免费 | 帮老师解开蕾丝奶罩吸乳网站 | 一本大道伊人av久久综合 | 中文字幕+乱码+中文字幕一区 | 欧美精品免费观看二区 | 动漫av一区二区在线观看 | 中文字幕av无码一区二区三区电影 | 亚洲熟妇色xxxxx欧美老妇y | 亚洲va欧美va天堂v国产综合 | 欧美熟妇另类久久久久久多毛 | 国产九九九九九九九a片 | 熟女俱乐部五十路六十路av | a在线观看免费网站大全 | 久久精品国产99精品亚洲 | 亚洲gv猛男gv无码男同 | 午夜精品久久久久久久久 | 久久国产劲爆∧v内射 | 日本护士毛茸茸高潮 | 亚洲最大成人网站 | 无套内射视频囯产 | 国产绳艺sm调教室论坛 | 国产激情艳情在线看视频 | 丰腴饱满的极品熟妇 | 精品无码成人片一区二区98 | 久久精品人妻少妇一区二区三区 | 久久久久亚洲精品中文字幕 | 久久99精品国产麻豆蜜芽 | 精品偷拍一区二区三区在线看 | 久久综合久久自在自线精品自 | 国产精品无码mv在线观看 | 内射老妇bbwx0c0ck | 波多野结衣一区二区三区av免费 | 国产精品igao视频网 | 亚洲成a人一区二区三区 | 蜜臀aⅴ国产精品久久久国产老师 | 内射后入在线观看一区 | 亚洲日韩一区二区 | 麻豆国产人妻欲求不满谁演的 | 精品久久综合1区2区3区激情 | 在线成人www免费观看视频 | 日本精品人妻无码77777 天堂一区人妻无码 | 国产精品亚洲专区无码不卡 | 秋霞特色aa大片 | 亚洲另类伦春色综合小说 | 思思久久99热只有频精品66 | 久久亚洲中文字幕精品一区 | 亚洲第一网站男人都懂 | 亚无码乱人伦一区二区 | 国产手机在线αⅴ片无码观看 | 一本一道久久综合久久 | 国产sm调教视频在线观看 | 午夜理论片yy44880影院 | 狠狠cao日日穞夜夜穞av | 女人被男人躁得好爽免费视频 | 国内揄拍国内精品少妇国语 | 午夜福利电影 | 牲欲强的熟妇农村老妇女视频 | 国产精品亚洲五月天高清 | 国产精品人人爽人人做我的可爱 | 日韩精品无码免费一区二区三区 | 人人妻人人澡人人爽欧美精品 | 亚洲欧美日韩国产精品一区二区 | 无码av岛国片在线播放 | 人妻中文无码久热丝袜 | 国产成人一区二区三区别 | 久久久无码中文字幕久... | 88国产精品欧美一区二区三区 | 中文字幕人妻丝袜二区 | 四虎4hu永久免费 | 久久久久免费精品国产 | 中文字幕无码热在线视频 | 国产精品久久久久久久9999 | 色噜噜亚洲男人的天堂 | 装睡被陌生人摸出水好爽 | 77777熟女视频在线观看 а天堂中文在线官网 | 日本大香伊一区二区三区 | 给我免费的视频在线观看 | 久久久久se色偷偷亚洲精品av | 最新版天堂资源中文官网 | 国产性生大片免费观看性 | 免费男性肉肉影院 | 红桃av一区二区三区在线无码av | 午夜福利试看120秒体验区 | 男女性色大片免费网站 | 国产乱码精品一品二品 | 日韩在线不卡免费视频一区 | 少妇愉情理伦片bd | 性色欲网站人妻丰满中文久久不卡 | 人人澡人人透人人爽 | 国产午夜视频在线观看 | 国产精品无码mv在线观看 | 在线亚洲高清揄拍自拍一品区 | 黑森林福利视频导航 | 日韩在线不卡免费视频一区 | 久久99精品国产.久久久久 | 亚洲欧洲日本无在线码 | 2019午夜福利不卡片在线 | 国产深夜福利视频在线 | 亚洲国产日韩a在线播放 | 人妻人人添人妻人人爱 | 国产亚洲欧美日韩亚洲中文色 | 久久人人爽人人爽人人片ⅴ | 76少妇精品导航 | 高清不卡一区二区三区 | 窝窝午夜理论片影院 | 中文字幕人妻丝袜二区 | 学生妹亚洲一区二区 | 精品国产青草久久久久福利 | 中文字幕精品av一区二区五区 | 夜精品a片一区二区三区无码白浆 | 亚洲色成人中文字幕网站 | 欧美精品免费观看二区 | 久久综合激激的五月天 | 粉嫩少妇内射浓精videos | 荫蒂被男人添的好舒服爽免费视频 | 亚洲国产精品一区二区第一页 | 国产人妻久久精品二区三区老狼 | 中文字幕乱码中文乱码51精品 | aⅴ在线视频男人的天堂 | 在线播放亚洲第一字幕 | 亚洲毛片av日韩av无码 | 精品aⅴ一区二区三区 | 成人免费视频一区二区 | 人妻天天爽夜夜爽一区二区 | 日日夜夜撸啊撸 | 免费无码的av片在线观看 | 爽爽影院免费观看 | 狠狠躁日日躁夜夜躁2020 | 精品久久久中文字幕人妻 | 天天躁日日躁狠狠躁免费麻豆 | 欧美 亚洲 国产 另类 | 欧美老妇交乱视频在线观看 | 久久午夜夜伦鲁鲁片无码免费 | 又紧又大又爽精品一区二区 | 国产亚洲人成在线播放 | 99riav国产精品视频 | 无码人妻久久一区二区三区不卡 | 国产亚洲精品久久久久久国模美 | 88国产精品欧美一区二区三区 | 日日鲁鲁鲁夜夜爽爽狠狠 | 成熟女人特级毛片www免费 | 丰满诱人的人妻3 | www国产精品内射老师 | 任你躁国产自任一区二区三区 | 色综合视频一区二区三区 | 久久亚洲精品中文字幕无男同 | 色欲综合久久中文字幕网 | 国产深夜福利视频在线 | 男女下面进入的视频免费午夜 | 久久这里只有精品视频9 | 国产舌乚八伦偷品w中 | 麻豆md0077饥渴少妇 | 久久久精品欧美一区二区免费 | 少妇邻居内射在线 | 国产无遮挡又黄又爽免费视频 | 东京热男人av天堂 | 久久久www成人免费毛片 | 欧美性猛交内射兽交老熟妇 | 欧美性生交活xxxxxdddd | 久久精品国产一区二区三区肥胖 | 亚洲 高清 成人 动漫 | 熟妇女人妻丰满少妇中文字幕 | 狠狠躁日日躁夜夜躁2020 | 久久97精品久久久久久久不卡 | 日韩精品无码一本二本三本色 | 欧美人与物videos另类 | 最新国产乱人伦偷精品免费网站 | 亚洲成av人综合在线观看 | 99久久精品日本一区二区免费 | 欧美熟妇另类久久久久久多毛 | 综合人妻久久一区二区精品 | 免费看男女做好爽好硬视频 | 国产精品99爱免费视频 | 久热国产vs视频在线观看 | 久在线观看福利视频 | 亚洲 激情 小说 另类 欧美 | 成人精品视频一区二区 | 在线成人www免费观看视频 | 久久精品99久久香蕉国产色戒 | 精品无人区无码乱码毛片国产 | 国产精品久久久久影院嫩草 | 在线精品国产一区二区三区 | 免费观看激色视频网站 | 亚洲日本va午夜在线电影 | 成人无码精品一区二区三区 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 色综合天天综合狠狠爱 | 国产一区二区三区四区五区加勒比 | 无码福利日韩神码福利片 | 欧美老妇交乱视频在线观看 | 国产舌乚八伦偷品w中 | 国产精品亚洲lv粉色 | 岛国片人妻三上悠亚 | 亚洲天堂2017无码 | 亚洲gv猛男gv无码男同 | 特级做a爰片毛片免费69 | 沈阳熟女露脸对白视频 | 中文精品无码中文字幕无码专区 | 久久综合激激的五月天 | 午夜免费福利小电影 | 少妇激情av一区二区 | 97精品国产97久久久久久免费 | 波多野结衣av一区二区全免费观看 | 人人爽人人爽人人片av亚洲 | 国内老熟妇对白xxxxhd | 一本色道久久综合狠狠躁 | 伊人久久大香线焦av综合影院 | 日产精品99久久久久久 | 亚洲人成网站在线播放942 | 久久精品国产一区二区三区 | 草草网站影院白丝内射 | 午夜福利一区二区三区在线观看 | 国产乱人偷精品人妻a片 | 久久久久成人片免费观看蜜芽 | 成人精品一区二区三区中文字幕 | 无码中文字幕色专区 | 成人三级无码视频在线观看 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 久久精品女人天堂av免费观看 | 中国女人内谢69xxxx | 国产精品99久久精品爆乳 | 国产亲子乱弄免费视频 | 亚洲精品美女久久久久久久 | 中文字幕人妻无码一夲道 | 国产情侣作爱视频免费观看 | av在线亚洲欧洲日产一区二区 | 成人无码精品一区二区三区 | 精品国产aⅴ无码一区二区 | 在线亚洲高清揄拍自拍一品区 | 成人性做爰aaa片免费看不忠 | 狂野欧美性猛xxxx乱大交 | 国产深夜福利视频在线 | 麻豆人妻少妇精品无码专区 | 国产麻豆精品一区二区三区v视界 | 国产又粗又硬又大爽黄老大爷视 | 国产电影无码午夜在线播放 | 国产人成高清在线视频99最全资源 | 亚洲第一无码av无码专区 | 无码国产乱人伦偷精品视频 | 特黄特色大片免费播放器图片 | 人妻插b视频一区二区三区 | 内射白嫩少妇超碰 | 精品久久久中文字幕人妻 | 婷婷六月久久综合丁香 | 国产av剧情md精品麻豆 | 国产精品自产拍在线观看 | 亚洲 高清 成人 动漫 | 国产av无码专区亚洲awww | 久久久久久a亚洲欧洲av冫 | 内射老妇bbwx0c0ck | 免费观看又污又黄的网站 | 久久精品国产99精品亚洲 | 久久久久久久人妻无码中文字幕爆 | 成人免费视频视频在线观看 免费 | 窝窝午夜理论片影院 | 久久久成人毛片无码 | 麻花豆传媒剧国产免费mv在线 | 永久免费观看国产裸体美女 | 亚洲精品成人福利网站 | 精品无码一区二区三区的天堂 | 男女作爱免费网站 | 撕开奶罩揉吮奶头视频 | 色窝窝无码一区二区三区色欲 | 中文字幕 人妻熟女 | 美女黄网站人色视频免费国产 | 东北女人啪啪对白 | 无码人妻久久一区二区三区不卡 | 国产一精品一av一免费 | 久久亚洲精品中文字幕无男同 | 亚洲国产欧美国产综合一区 | 久久久精品人妻久久影视 | 国产三级久久久精品麻豆三级 | 国语精品一区二区三区 | 理论片87福利理论电影 | 久久五月精品中文字幕 | 欧美日韩综合一区二区三区 | 无码一区二区三区在线观看 | 人妻少妇精品无码专区动漫 | 131美女爱做视频 | 狠狠亚洲超碰狼人久久 | 国产av无码专区亚洲a∨毛片 | 久久成人a毛片免费观看网站 | 奇米影视7777久久精品 | 人人妻人人藻人人爽欧美一区 | 性欧美疯狂xxxxbbbb | 日韩亚洲欧美中文高清在线 | 日本欧美一区二区三区乱码 | 麻豆国产人妻欲求不满 | 国产成人无码av片在线观看不卡 | 国产农村妇女高潮大叫 | 55夜色66夜色国产精品视频 | 国产精品.xx视频.xxtv | 亚洲狠狠色丁香婷婷综合 | 久久人人97超碰a片精品 | 日韩在线不卡免费视频一区 | www国产亚洲精品久久网站 | 无套内谢老熟女 | 久久久精品成人免费观看 | 97精品人妻一区二区三区香蕉 | 久久午夜无码鲁丝片午夜精品 | 国产免费久久精品国产传媒 | 国产精品人妻一区二区三区四 | 亚洲人成影院在线无码按摩店 | 亚洲区小说区激情区图片区 | 久久精品国产日本波多野结衣 | 国产精品久久久久影院嫩草 | 免费无码午夜福利片69 | 国产免费观看黄av片 | 中文字幕av日韩精品一区二区 | 中文精品久久久久人妻不卡 | 天堂а√在线中文在线 | 少妇久久久久久人妻无码 | 欧美性猛交xxxx富婆 | 小鲜肉自慰网站xnxx | 欧美zoozzooz性欧美 | 久青草影院在线观看国产 | 精品国产av色一区二区深夜久久 | 最新国产乱人伦偷精品免费网站 | 网友自拍区视频精品 | 欧美zoozzooz性欧美 | 国产成人精品优优av | 牲欲强的熟妇农村老妇女 | 亚洲中文字幕无码中文字在线 | 最近免费中文字幕中文高清百度 | 狠狠cao日日穞夜夜穞av | 日韩人妻系列无码专区 | 国产精品人妻一区二区三区四 | 亚洲日韩一区二区三区 | 亚洲精品国偷拍自产在线观看蜜桃 | 国产又爽又猛又粗的视频a片 | 欧美亚洲国产一区二区三区 | 男女超爽视频免费播放 | 欧美丰满熟妇xxxx性ppx人交 | 无人区乱码一区二区三区 | 在线精品亚洲一区二区 | 国产精品视频免费播放 | 欧美日韩久久久精品a片 | 国产特级毛片aaaaaa高潮流水 | 国产精品-区区久久久狼 | 国产欧美精品一区二区三区 | 国产片av国语在线观看 | 人人妻人人澡人人爽人人精品 | 日韩成人一区二区三区在线观看 | 黑森林福利视频导航 | 天天av天天av天天透 | 亚洲一区二区三区含羞草 | 亚洲成a人片在线观看日本 | 特黄特色大片免费播放器图片 | 亚洲精品一区二区三区大桥未久 | 色情久久久av熟女人妻网站 | 国产成人一区二区三区在线观看 | 国产亚av手机在线观看 | 久久久久成人片免费观看蜜芽 | 中文字幕乱妇无码av在线 | 中文字幕av无码一区二区三区电影 | aⅴ在线视频男人的天堂 | 无码福利日韩神码福利片 | 一本大道伊人av久久综合 | 国产成人无码av片在线观看不卡 | 国产 浪潮av性色四虎 | 久久伊人色av天堂九九小黄鸭 | 国产成人精品优优av | 亚洲日韩乱码中文无码蜜桃臀网站 | 日日摸夜夜摸狠狠摸婷婷 | 天天拍夜夜添久久精品 | 无码人妻精品一区二区三区下载 | 一本色道久久综合狠狠躁 | 乌克兰少妇xxxx做受 | 少妇人妻偷人精品无码视频 | av人摸人人人澡人人超碰下载 | 欧洲熟妇色 欧美 | 亚洲小说春色综合另类 | 中文字幕无码日韩专区 | 丰满少妇熟乱xxxxx视频 | 又大又黄又粗又爽的免费视频 | 无码一区二区三区在线观看 | 夜夜躁日日躁狠狠久久av | 亚洲人交乣女bbw | 日日夜夜撸啊撸 | 国产欧美熟妇另类久久久 | 国内揄拍国内精品少妇国语 | 18精品久久久无码午夜福利 | 亚洲中文字幕无码中文字在线 | 成人亚洲精品久久久久软件 | 成人动漫在线观看 | 免费视频欧美无人区码 | 天堂无码人妻精品一区二区三区 | 一个人看的www免费视频在线观看 | 国产成人精品必看 | 人人爽人人澡人人高潮 | 日本在线高清不卡免费播放 | 丁香啪啪综合成人亚洲 | 性欧美熟妇videofreesex | 午夜肉伦伦影院 | 国产在线无码精品电影网 | 亚洲国产精品美女久久久久 | 日本www一道久久久免费榴莲 | 国产精品内射视频免费 | 狂野欧美性猛xxxx乱大交 | 久久精品国产99精品亚洲 | 国产熟妇高潮叫床视频播放 | 亚洲午夜久久久影院 | 亚洲欧美精品伊人久久 | 国产乱人伦偷精品视频 | 国产亚洲精品久久久久久 | 偷窥村妇洗澡毛毛多 | 亚洲成a人一区二区三区 | 一本精品99久久精品77 | 国产人妻精品一区二区三区不卡 | 欧美精品一区二区精品久久 | 欧美一区二区三区视频在线观看 | 成在人线av无码免观看麻豆 | 久久久成人毛片无码 | 荫蒂被男人添的好舒服爽免费视频 | 亚洲精品一区二区三区婷婷月 | 97资源共享在线视频 | 人妻少妇精品久久 | 亚洲伊人久久精品影院 | 色综合久久88色综合天天 | 无码av最新清无码专区吞精 | 无码人妻黑人中文字幕 | 久久综合香蕉国产蜜臀av | 精品国偷自产在线视频 | 国产无遮挡又黄又爽免费视频 | 久久国内精品自在自线 | 西西人体www44rt大胆高清 | 亚洲精品综合一区二区三区在线 | 日日摸夜夜摸狠狠摸婷婷 | 领导边摸边吃奶边做爽在线观看 | 国产无av码在线观看 | 国产精品无套呻吟在线 | 未满小14洗澡无码视频网站 | 成人无码视频在线观看网站 | 人妻少妇被猛烈进入中文字幕 | 99久久久无码国产精品免费 | 夜精品a片一区二区三区无码白浆 | 国产卡一卡二卡三 | 久久久久久亚洲精品a片成人 | 国产性生大片免费观看性 | 日韩av无码中文无码电影 | 中国女人内谢69xxxx | 国产美女精品一区二区三区 | 国产真实伦对白全集 | 国内精品久久久久久中文字幕 | 国内精品九九久久久精品 | 亚洲人亚洲人成电影网站色 | 亚洲熟熟妇xxxx | 日本熟妇乱子伦xxxx | 青青久在线视频免费观看 | 国产人妻精品午夜福利免费 | 国产亚洲美女精品久久久2020 | 少妇性荡欲午夜性开放视频剧场 | 亚拍精品一区二区三区探花 | 亚洲 高清 成人 动漫 | 国产精品亚洲专区无码不卡 | 国产成人综合在线女婷五月99播放 | 欧美丰满熟妇xxxx性ppx人交 | 色综合视频一区二区三区 | 国产麻豆精品精东影业av网站 | 亚洲va欧美va天堂v国产综合 | 狠狠躁日日躁夜夜躁2020 | а√资源新版在线天堂 | 黑人巨大精品欧美黑寡妇 | 国产精品99久久精品爆乳 | 国产精品高潮呻吟av久久 | 夜精品a片一区二区三区无码白浆 | 鲁一鲁av2019在线 | www国产精品内射老师 | 欧美日韩亚洲国产精品 | 天堂亚洲免费视频 | 人人妻人人澡人人爽欧美精品 | 亚洲大尺度无码无码专区 | aⅴ亚洲 日韩 色 图网站 播放 | 毛片内射-百度 | 国产深夜福利视频在线 | 亚洲国产欧美日韩精品一区二区三区 | 欧美激情一区二区三区成人 | 18禁止看的免费污网站 | 亚洲人成影院在线无码按摩店 | 欧美自拍另类欧美综合图片区 | 少妇性荡欲午夜性开放视频剧场 | 99精品国产综合久久久久五月天 | 国产人成高清在线视频99最全资源 | 人妻互换免费中文字幕 | 国产真实伦对白全集 | 国产精品美女久久久网av | 久久久婷婷五月亚洲97号色 | 亚洲日本va午夜在线电影 | 亚洲精品无码国产 | 国产av久久久久精东av | 欧美日韩一区二区三区自拍 | 亚洲爆乳大丰满无码专区 | 国产精品亚洲lv粉色 | aa片在线观看视频在线播放 | 日本精品少妇一区二区三区 | 成人一在线视频日韩国产 | 久久国产劲爆∧v内射 | 永久免费观看美女裸体的网站 | 久久久精品欧美一区二区免费 | 日产精品高潮呻吟av久久 | 青青青爽视频在线观看 | 日韩人妻无码一区二区三区久久99 | 国产激情无码一区二区app | 久久人人97超碰a片精品 | 国产午夜亚洲精品不卡 | 蜜桃av抽搐高潮一区二区 | 日韩精品无码免费一区二区三区 | 欧美精品一区二区精品久久 | 亚洲色www成人永久网址 | 久久精品女人的天堂av | 夜精品a片一区二区三区无码白浆 | 国产成人av免费观看 | 精品无码成人片一区二区98 | 丰满少妇熟乱xxxxx视频 | 亚洲日韩精品欧美一区二区 | 97久久精品无码一区二区 | 精品一区二区三区波多野结衣 | 国产精品自产拍在线观看 | 国产精品久久久久久亚洲影视内衣 | 国产人妻精品一区二区三区 | 无码精品人妻一区二区三区av | 国产偷自视频区视频 | 香蕉久久久久久av成人 | 亚洲熟妇色xxxxx欧美老妇 | 亚洲精品午夜国产va久久成人 | 2020久久香蕉国产线看观看 | 国产熟女一区二区三区四区五区 | 欧美35页视频在线观看 | 久久久久免费看成人影片 | 日本一卡二卡不卡视频查询 | 俄罗斯老熟妇色xxxx | 国产绳艺sm调教室论坛 | 少妇无套内谢久久久久 | 日韩少妇白浆无码系列 | 国产香蕉尹人综合在线观看 | 丁香花在线影院观看在线播放 | 麻豆国产97在线 | 欧洲 | 日本爽爽爽爽爽爽在线观看免 | 日本丰满护士爆乳xxxx | 国产精品久久久久7777 | 丰满人妻一区二区三区免费视频 | 中文字幕无码av波多野吉衣 | 国产亚洲精品久久久ai换 | 精品成人av一区二区三区 | 无码吃奶揉捏奶头高潮视频 | 丰满少妇熟乱xxxxx视频 | 亚洲日韩乱码中文无码蜜桃臀网站 | 正在播放老肥熟妇露脸 | 亚洲s码欧洲m码国产av | 2020久久香蕉国产线看观看 | 欧美人与牲动交xxxx | 久久天天躁狠狠躁夜夜免费观看 | 亚洲成a人片在线观看无码3d | 国产人妻久久精品二区三区老狼 | 99久久精品无码一区二区毛片 | 一本久久a久久精品亚洲 | 国产亚洲精品久久久ai换 | 人妻少妇精品久久 | 性色欲网站人妻丰满中文久久不卡 | 一本久久a久久精品亚洲 | 成人一在线视频日韩国产 | 亚洲va中文字幕无码久久不卡 | 亚洲色欲久久久综合网东京热 | 成年美女黄网站色大免费全看 | 精品无码av一区二区三区 | 国产成人精品视频ⅴa片软件竹菊 | 狠狠色丁香久久婷婷综合五月 | 亚洲中文字幕无码中文字在线 | 国产成人精品优优av | 日日摸日日碰夜夜爽av | 国产麻豆精品一区二区三区v视界 | 老司机亚洲精品影院无码 | 成人无码视频免费播放 | 天干天干啦夜天干天2017 | 久久国产精品精品国产色婷婷 | 97久久国产亚洲精品超碰热 | 人妻少妇精品视频专区 | 久久精品99久久香蕉国产色戒 | 国产无套内射久久久国产 | 欧美人与善在线com | 波多野结衣高清一区二区三区 | 久久国内精品自在自线 | 日韩精品a片一区二区三区妖精 | 国产综合久久久久鬼色 | 最新国产麻豆aⅴ精品无码 | 亚洲精品国产品国语在线观看 | 国产明星裸体无码xxxx视频 | 无码人妻出轨黑人中文字幕 | 色情久久久av熟女人妻网站 | 偷窥村妇洗澡毛毛多 | 久久精品一区二区三区四区 | 日日摸日日碰夜夜爽av | 欧美真人作爱免费视频 | 人妻人人添人妻人人爱 | 又大又紧又粉嫩18p少妇 | 波多野结衣乳巨码无在线观看 | 一个人看的视频www在线 | 国产激情一区二区三区 | 久久精品人人做人人综合试看 | 99久久人妻精品免费二区 | 亚洲aⅴ无码成人网站国产app | 无码成人精品区在线观看 | 日本在线高清不卡免费播放 | 国产亚洲美女精品久久久2020 | 久久久国产一区二区三区 | 麻豆精品国产精华精华液好用吗 | 国产人妻精品午夜福利免费 | 国产精品亚洲а∨无码播放麻豆 | 国产真人无遮挡作爱免费视频 | 67194成是人免费无码 | 国产成人无码av在线影院 | 亚洲国产精品久久人人爱 | 国产乱人偷精品人妻a片 | 东京热男人av天堂 | 无码免费一区二区三区 | 欧美真人作爱免费视频 | 日韩av激情在线观看 | 国产极品视觉盛宴 | 人人澡人人妻人人爽人人蜜桃 | 日韩在线不卡免费视频一区 | 伊人色综合久久天天小片 | 亚洲人成影院在线观看 | 无码福利日韩神码福利片 | 未满小14洗澡无码视频网站 | 欧美日韩一区二区综合 | 无码av中文字幕免费放 | 宝宝好涨水快流出来免费视频 | 红桃av一区二区三区在线无码av | 国产国语老龄妇女a片 | 精品人妻中文字幕有码在线 | 欧美亚洲国产一区二区三区 | 熟女俱乐部五十路六十路av | 免费播放一区二区三区 | 亚洲国产欧美在线成人 | 亚洲第一网站男人都懂 | 欧美精品在线观看 | 国产成人综合色在线观看网站 | 亚洲精品成人福利网站 | 麻豆人妻少妇精品无码专区 | 欧美性色19p | 精品国偷自产在线 | 亚洲s码欧洲m码国产av | 无套内射视频囯产 | 麻豆国产丝袜白领秘书在线观看 | 久久久成人毛片无码 | av人摸人人人澡人人超碰下载 | 国产熟女一区二区三区四区五区 | 国产精品亚洲lv粉色 | 中文字幕人妻无码一夲道 | 性做久久久久久久久 | 天天av天天av天天透 | 久久精品中文字幕大胸 | 97人妻精品一区二区三区 | 国产欧美熟妇另类久久久 | 国产在线精品一区二区高清不卡 | 麻豆果冻传媒2021精品传媒一区下载 | 97色伦图片97综合影院 | aⅴ亚洲 日韩 色 图网站 播放 | 亚洲综合无码一区二区三区 | 欧美丰满老熟妇xxxxx性 | 99久久99久久免费精品蜜桃 | 欧美激情一区二区三区成人 | 日本一卡二卡不卡视频查询 | 天堂亚洲免费视频 | 网友自拍区视频精品 | 亚洲大尺度无码无码专区 | 亚洲阿v天堂在线 | 国产明星裸体无码xxxx视频 | 乱中年女人伦av三区 | 国产人妻精品一区二区三区 | 成年美女黄网站色大免费视频 | aⅴ在线视频男人的天堂 | 窝窝午夜理论片影院 | 少妇一晚三次一区二区三区 | 国产精品国产三级国产专播 | 水蜜桃色314在线观看 | 无码av最新清无码专区吞精 | 丝袜足控一区二区三区 | 精品偷自拍另类在线观看 | 亚洲日韩中文字幕在线播放 | 狠狠色噜噜狠狠狠狠7777米奇 | 成人免费视频视频在线观看 免费 | 99国产欧美久久久精品 | 少妇厨房愉情理9仑片视频 | 欧美亚洲国产一区二区三区 | 久久国产精品精品国产色婷婷 | 国产亚洲精品久久久久久久久动漫 | 网友自拍区视频精品 | 亚洲精品无码国产 | 黑人巨大精品欧美黑寡妇 | 男女猛烈xx00免费视频试看 | 亚洲人成网站色7799 | 少妇激情av一区二区 | 国产极品美女高潮无套在线观看 | 亚洲精品一区国产 | 欧美成人免费全部网站 | 日韩av无码中文无码电影 | 青青青爽视频在线观看 | 国产麻豆精品精东影业av网站 | 中文字幕无线码免费人妻 | 一本色道久久综合亚洲精品不卡 | 午夜不卡av免费 一本久久a久久精品vr综合 | 国产99久久精品一区二区 | 青青青手机频在线观看 | 久久综合狠狠综合久久综合88 | 久久亚洲a片com人成 | 97精品人妻一区二区三区香蕉 | 久久精品中文闷骚内射 | 在线欧美精品一区二区三区 | 亚洲精品一区二区三区婷婷月 | 亚洲成av人影院在线观看 | 日日鲁鲁鲁夜夜爽爽狠狠 | 亚洲精品成人福利网站 | 国产一精品一av一免费 | 亚洲 日韩 欧美 成人 在线观看 | 捆绑白丝粉色jk震动捧喷白浆 | 亚洲成色在线综合网站 | 青青草原综合久久大伊人精品 | 人妻互换免费中文字幕 | 国产精品久久久久7777 | 人人妻人人澡人人爽欧美精品 | 国产成人无码a区在线观看视频app | 美女黄网站人色视频免费国产 | 日本丰满护士爆乳xxxx | 天天拍夜夜添久久精品 | 国内精品久久毛片一区二区 | 熟妇女人妻丰满少妇中文字幕 | 疯狂三人交性欧美 | 亚洲熟妇色xxxxx欧美老妇 | 久久这里只有精品视频9 | 欧美日韩在线亚洲综合国产人 | 亚洲精品久久久久avwww潮水 | 亚洲爆乳无码专区 | 色窝窝无码一区二区三区色欲 | 欧美变态另类xxxx | 精品aⅴ一区二区三区 | 奇米影视7777久久精品人人爽 | 国产国语老龄妇女a片 | 亚洲精品国偷拍自产在线麻豆 | 宝宝好涨水快流出来免费视频 | 成人无码影片精品久久久 | 欧美日韩一区二区三区自拍 | 波多野结衣av在线观看 | 熟妇女人妻丰满少妇中文字幕 | 又湿又紧又大又爽a视频国产 | 久久亚洲中文字幕无码 | 精品一区二区三区波多野结衣 | 日韩人妻无码一区二区三区久久99 | 亚洲春色在线视频 | 亚洲精品一区二区三区婷婷月 | 西西人体www44rt大胆高清 | 亚洲高清偷拍一区二区三区 | 粗大的内捧猛烈进出视频 | 国产av无码专区亚洲awww | 中文字幕乱码人妻二区三区 | 激情人妻另类人妻伦 | 久久国产精品精品国产色婷婷 | 色欲人妻aaaaaaa无码 | 日本高清一区免费中文视频 | 99久久99久久免费精品蜜桃 | 蜜臀av无码人妻精品 | 大色综合色综合网站 | 无遮无挡爽爽免费视频 | 永久免费观看美女裸体的网站 | 男人的天堂2018无码 | 国产高清不卡无码视频 | 精品偷拍一区二区三区在线看 | 内射白嫩少妇超碰 | 国产一精品一av一免费 | 99久久久无码国产精品免费 | 亚洲热妇无码av在线播放 | 中文字幕无码日韩专区 | 国产精品无码一区二区三区不卡 | 狠狠cao日日穞夜夜穞av | 久久aⅴ免费观看 | 国产美女精品一区二区三区 | 亚洲日本va中文字幕 | 老司机亚洲精品影院 | 高清国产亚洲精品自在久久 | 欧美freesex黑人又粗又大 | 亚洲精品国产第一综合99久久 | 国产成人精品无码播放 | 九一九色国产 | 久久国语露脸国产精品电影 | 亚洲色www成人永久网址 | 精品厕所偷拍各类美女tp嘘嘘 | 永久免费观看美女裸体的网站 | 亚洲大尺度无码无码专区 | 妺妺窝人体色www在线小说 | 日韩欧美中文字幕在线三区 | 成人亚洲精品久久久久软件 | 亚洲a无码综合a国产av中文 | v一区无码内射国产 | 国产猛烈高潮尖叫视频免费 | 2019午夜福利不卡片在线 | 亚洲乱亚洲乱妇50p | 老司机亚洲精品影院无码 | 麻豆md0077饥渴少妇 | 午夜理论片yy44880影院 | 成熟女人特级毛片www免费 | 人人超人人超碰超国产 | 曰本女人与公拘交酡免费视频 | 久久成人a毛片免费观看网站 | 国产极品美女高潮无套在线观看 | 欧美成人午夜精品久久久 | 丁香啪啪综合成人亚洲 | 国产热a欧美热a在线视频 | 麻豆果冻传媒2021精品传媒一区下载 | 中文字幕无线码免费人妻 | 久久综合网欧美色妞网 | 成人免费视频视频在线观看 免费 | 国产婷婷色一区二区三区在线 | 国产两女互慰高潮视频在线观看 | 亚洲乱码日产精品bd | 国产欧美精品一区二区三区 | 天天摸天天透天天添 | 精品成人av一区二区三区 | 亚洲一区av无码专区在线观看 | 国产午夜无码精品免费看 | 亚洲人成影院在线无码按摩店 | 国语自产偷拍精品视频偷 | 亚洲日韩av一区二区三区四区 | yw尤物av无码国产在线观看 | 全球成人中文在线 | 最新国产乱人伦偷精品免费网站 | 亚洲啪av永久无码精品放毛片 | 美女黄网站人色视频免费国产 | 99在线 | 亚洲 | 性做久久久久久久免费看 | 十八禁真人啪啪免费网站 | 色欲综合久久中文字幕网 | 18精品久久久无码午夜福利 | 人人妻人人澡人人爽欧美一区九九 | 国精品人妻无码一区二区三区蜜柚 | 在线精品亚洲一区二区 | 国产偷自视频区视频 | 欧美激情内射喷水高潮 | 女人色极品影院 | 亚洲综合久久一区二区 | 国产成人精品视频ⅴa片软件竹菊 | 国产偷国产偷精品高清尤物 | 丁香花在线影院观看在线播放 | 最新国产麻豆aⅴ精品无码 | 日韩av激情在线观看 | 亚洲日韩av一区二区三区中文 | 精品久久久中文字幕人妻 | 四虎国产精品一区二区 | 动漫av一区二区在线观看 | 宝宝好涨水快流出来免费视频 | 中文字幕无码日韩欧毛 | 国产凸凹视频一区二区 | 久久www免费人成人片 | 国产午夜亚洲精品不卡下载 | 无码任你躁久久久久久久 | 奇米影视7777久久精品人人爽 | 亚洲一区二区三区含羞草 | 曰韩少妇内射免费播放 | 高潮毛片无遮挡高清免费 | 欧美黑人巨大xxxxx | 国产舌乚八伦偷品w中 | 久久99精品久久久久久 | 亚洲国产精品无码一区二区三区 | 国产超碰人人爽人人做人人添 | 成人免费视频视频在线观看 免费 | 亚洲乱码中文字幕在线 | 亚洲精品无码国产 | 无码人妻少妇伦在线电影 | 午夜精品一区二区三区的区别 | 亚洲の无码国产の无码步美 | 亚洲成色www久久网站 | 香港三级日本三级妇三级 | 色欲av亚洲一区无码少妇 | 18精品久久久无码午夜福利 | 婷婷丁香六月激情综合啪 | 无遮挡啪啪摇乳动态图 | www成人国产高清内射 | 亚洲大尺度无码无码专区 | 精品人妻人人做人人爽夜夜爽 | 亚洲第一网站男人都懂 | 国产精品无码一区二区桃花视频 | 丁香花在线影院观看在线播放 | 大肉大捧一进一出好爽视频 | 亚洲成a人片在线观看无码 | 少妇高潮一区二区三区99 | 任你躁国产自任一区二区三区 | 999久久久国产精品消防器材 | 免费国产成人高清在线观看网站 | 日韩无码专区 | 亚洲国产成人a精品不卡在线 | 国产无套内射久久久国产 | 国产精品多人p群无码 | 扒开双腿吃奶呻吟做受视频 | 成人性做爰aaa片免费看 | 国内精品一区二区三区不卡 | 水蜜桃亚洲一二三四在线 | 国内精品久久毛片一区二区 | 亚洲综合久久一区二区 | 欧美怡红院免费全部视频 | 精品国产精品久久一区免费式 | 1000部夫妻午夜免费 | 国产精品久免费的黄网站 | 国产精品国产自线拍免费软件 | 久久99精品国产麻豆蜜芽 | 永久免费精品精品永久-夜色 | 玩弄中年熟妇正在播放 | 精品水蜜桃久久久久久久 | 国产尤物精品视频 | 久久zyz资源站无码中文动漫 | 强伦人妻一区二区三区视频18 | 亚洲成色在线综合网站 | 国产午夜无码精品免费看 | 日韩亚洲欧美中文高清在线 | 蜜臀aⅴ国产精品久久久国产老师 | 久久无码人妻影院 | 国产精品视频免费播放 | 国産精品久久久久久久 | 搡女人真爽免费视频大全 | 成人一区二区免费视频 | 性生交大片免费看l | 国产绳艺sm调教室论坛 | 精品国产麻豆免费人成网站 | 国产精品亚洲а∨无码播放麻豆 | 无码人妻丰满熟妇区毛片18 | 丰满少妇高潮惨叫视频 | 国产偷抇久久精品a片69 | 亚洲成a人一区二区三区 | 亚洲欧洲日本综合aⅴ在线 | 国产乱人伦av在线无码 | 性生交大片免费看l | 午夜熟女插插xx免费视频 | 波多野结衣高清一区二区三区 | 搡女人真爽免费视频大全 | 国产麻豆精品一区二区三区v视界 | 最近中文2019字幕第二页 | 永久免费观看国产裸体美女 | 欧美亚洲国产一区二区三区 | 亚洲精品鲁一鲁一区二区三区 | 少妇高潮一区二区三区99 | 小鲜肉自慰网站xnxx | www成人国产高清内射 | 亚洲小说春色综合另类 | 亚洲中文字幕av在天堂 | 亚洲精品成人av在线 | 老太婆性杂交欧美肥老太 | 丰满少妇女裸体bbw | 久久亚洲精品中文字幕无男同 | 樱花草在线播放免费中文 | 中文字幕人妻无码一夲道 | 中文字幕无码热在线视频 | 少妇被黑人到高潮喷出白浆 | 少妇性荡欲午夜性开放视频剧场 | 国产性生大片免费观看性 | 在线观看国产午夜福利片 | 天干天干啦夜天干天2017 | 捆绑白丝粉色jk震动捧喷白浆 | 天堂一区人妻无码 | 久久无码专区国产精品s | 内射白嫩少妇超碰 | 天天综合网天天综合色 | 国内揄拍国内精品人妻 | 香蕉久久久久久av成人 | 亚洲熟熟妇xxxx | 精品偷拍一区二区三区在线看 | 国产无套粉嫩白浆在线 | 成人一区二区免费视频 | 久久久久久久人妻无码中文字幕爆 | 丰满岳乱妇在线观看中字无码 | 六月丁香婷婷色狠狠久久 | 亚洲娇小与黑人巨大交 | 成人无码视频免费播放 | 好爽又高潮了毛片免费下载 | 久久久国产精品无码免费专区 | 欧美真人作爱免费视频 | 九九久久精品国产免费看小说 | 久久99精品国产.久久久久 | 99视频精品全部免费免费观看 | 亚洲va欧美va天堂v国产综合 | 午夜福利不卡在线视频 | 欧美精品在线观看 | 女人被男人爽到呻吟的视频 | 亚洲熟妇色xxxxx欧美老妇 | 一个人看的www免费视频在线观看 | 久久精品国产大片免费观看 | 亚洲а∨天堂久久精品2021 | 亚洲精品无码人妻无码 | 欧美黑人性暴力猛交喷水 | 中文字幕乱码人妻二区三区 | 色综合久久久无码中文字幕 | 久久久久成人片免费观看蜜芽 | 国产亚av手机在线观看 | 成年女人永久免费看片 | 无码一区二区三区在线观看 | 精品夜夜澡人妻无码av蜜桃 | aⅴ亚洲 日韩 色 图网站 播放 | 欧美日韩亚洲国产精品 | 久久精品人人做人人综合试看 | 99在线 | 亚洲 | 久久五月精品中文字幕 | 99国产欧美久久久精品 | 中国女人内谢69xxxxxa片 | 日日摸夜夜摸狠狠摸婷婷 | 99久久精品国产一区二区蜜芽 | 国产真实伦对白全集 | 久久精品国产日本波多野结衣 | 中文字幕av无码一区二区三区电影 | 一本久道久久综合婷婷五月 | 国产精品理论片在线观看 | 天堂在线观看www | 99在线 | 亚洲 | 熟妇激情内射com | 中文精品久久久久人妻不卡 | 人人妻人人澡人人爽欧美精品 | 欧美丰满少妇xxxx性 | 国产猛烈高潮尖叫视频免费 | 国产成人综合美国十次 | 午夜丰满少妇性开放视频 | 在线观看欧美一区二区三区 | 综合人妻久久一区二区精品 | 日欧一片内射va在线影院 | 丰满人妻翻云覆雨呻吟视频 | 久久午夜无码鲁丝片秋霞 | 日韩欧美成人免费观看 | 偷窥村妇洗澡毛毛多 | 无码国产色欲xxxxx视频 | 亚洲色无码一区二区三区 | 熟妇人妻激情偷爽文 | 久久精品丝袜高跟鞋 | 久久www免费人成人片 | 国产在线aaa片一区二区99 | 99久久无码一区人妻 | 久久久久成人精品免费播放动漫 | 啦啦啦www在线观看免费视频 | 一二三四社区在线中文视频 | 国产激情艳情在线看视频 | 露脸叫床粗话东北少妇 | 久久99精品国产麻豆蜜芽 | 欧美日韩色另类综合 | 欧洲熟妇色 欧美 | 麻豆成人精品国产免费 | 久久99久久99精品中文字幕 | 久久久久久九九精品久 | 激情爆乳一区二区三区 | 三级4级全黄60分钟 | 人人妻人人澡人人爽欧美一区 | 全黄性性激高免费视频 | 精品人妻人人做人人爽夜夜爽 | 亚洲狠狠色丁香婷婷综合 | 国产舌乚八伦偷品w中 | 国产亚洲人成a在线v网站 | 国产乱人无码伦av在线a | 久久zyz资源站无码中文动漫 | 亚洲gv猛男gv无码男同 | 内射老妇bbwx0c0ck | 国产亚洲精品久久久久久大师 | 国产人妻人伦精品 | 波多野结衣一区二区三区av免费 | 无码国产乱人伦偷精品视频 | 国产亚洲精品精品国产亚洲综合 | 国产av剧情md精品麻豆 | 国产精品无码一区二区桃花视频 | 国产人妖乱国产精品人妖 | 国产又粗又硬又大爽黄老大爷视 | 婷婷综合久久中文字幕蜜桃三电影 | 精品国产一区二区三区四区 | 窝窝午夜理论片影院 | 人人妻人人澡人人爽欧美一区 | 九九在线中文字幕无码 | 男人扒开女人内裤强吻桶进去 | 国产内射爽爽大片视频社区在线 | 日本免费一区二区三区最新 | av在线亚洲欧洲日产一区二区 | 精品水蜜桃久久久久久久 | 玩弄人妻少妇500系列视频 | 亚洲爆乳无码专区 | 黑人玩弄人妻中文在线 | 无码帝国www无码专区色综合 | 国产成人人人97超碰超爽8 | 露脸叫床粗话东北少妇 | 99久久精品无码一区二区毛片 | 亚洲精品无码人妻无码 | 亚洲色www成人永久网址 | 久久午夜无码鲁丝片 | 国产成人无码av一区二区 | 色老头在线一区二区三区 | 无码国产色欲xxxxx视频 | 久久99久久99精品中文字幕 | 人妻体内射精一区二区三四 | 少妇人妻偷人精品无码视频 | 色噜噜亚洲男人的天堂 | 亚洲a无码综合a国产av中文 | 无码乱肉视频免费大全合集 | 夜精品a片一区二区三区无码白浆 | 一区二区三区高清视频一 | 成人综合网亚洲伊人 | 呦交小u女精品视频 | 中文字幕乱码人妻无码久久 | 国产精品久久久久无码av色戒 | 噜噜噜亚洲色成人网站 | 成 人 免费观看网站 | 窝窝午夜理论片影院 | 精品 日韩 国产 欧美 视频 | 无码人妻出轨黑人中文字幕 | 久久久久亚洲精品男人的天堂 | 国产黄在线观看免费观看不卡 | 丰满少妇女裸体bbw | 欧美午夜特黄aaaaaa片 | 欧美人与物videos另类 | 亚洲成av人在线观看网址 | 在线观看免费人成视频 | 久久精品中文字幕大胸 | 99久久人妻精品免费一区 | 无码人中文字幕 | 国产精品第一区揄拍无码 | 东京热一精品无码av | 伊在人天堂亚洲香蕉精品区 | 亚洲色大成网站www国产 | 欧美35页视频在线观看 | 5858s亚洲色大成网站www | 97精品国产97久久久久久免费 | 日韩亚洲欧美精品综合 | 中文字幕无码视频专区 | 国产av一区二区精品久久凹凸 | 99精品视频在线观看免费 | 国产艳妇av在线观看果冻传媒 | 国产农村乱对白刺激视频 | 国产色xx群视频射精 | 亚洲自偷自偷在线制服 | 人妻互换免费中文字幕 | 欧美 日韩 人妻 高清 中文 | 成人无码精品一区二区三区 | 国产精品久久久av久久久 | 国产精品美女久久久久av爽李琼 | 激情综合激情五月俺也去 | 国产艳妇av在线观看果冻传媒 | 亚洲精品一区三区三区在线观看 | 国产精品久久久久久亚洲毛片 | 思思久久99热只有频精品66 | 国产又爽又猛又粗的视频a片 | 熟妇人妻中文av无码 | 久久国产精品萌白酱免费 | 欧美真人作爱免费视频 | 夜夜夜高潮夜夜爽夜夜爰爰 | 国产成人综合色在线观看网站 | 东北女人啪啪对白 | 丰满人妻精品国产99aⅴ | 国产成人综合色在线观看网站 | 国产免费观看黄av片 | 久久午夜无码鲁丝片午夜精品 | 特黄特色大片免费播放器图片 | 午夜熟女插插xx免费视频 | 狠狠色噜噜狠狠狠狠7777米奇 | 少妇人妻偷人精品无码视频 | 在线精品国产一区二区三区 | 国精产品一区二区三区 | 日韩人妻无码一区二区三区久久99 | 日本一区二区三区免费播放 | 131美女爱做视频 | 国产激情艳情在线看视频 | 亚洲精品国产品国语在线观看 | 蜜臀av无码人妻精品 | 2019午夜福利不卡片在线 | 精品欧美一区二区三区久久久 | 99久久人妻精品免费二区 | 国产成人无码a区在线观看视频app | 国产成人午夜福利在线播放 | 亚洲成av人影院在线观看 | 在线欧美精品一区二区三区 | 狠狠综合久久久久综合网 | 久久99久久99精品中文字幕 | 国产av人人夜夜澡人人爽麻豆 | 2019午夜福利不卡片在线 | 亚洲成在人网站无码天堂 | 无码午夜成人1000部免费视频 | 国产精品久免费的黄网站 | 少妇人妻av毛片在线看 | 亚洲乱码中文字幕在线 | 国产成人无码午夜视频在线观看 | 99久久婷婷国产综合精品青草免费 | 无码播放一区二区三区 | 国产女主播喷水视频在线观看 | 亚洲成a人一区二区三区 | 亚洲中文字幕久久无码 | 久久久久se色偷偷亚洲精品av | 日本熟妇大屁股人妻 | 99riav国产精品视频 | 日本免费一区二区三区最新 |