日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android开发中常见的设计模式

發布時間:2025/1/21 Android 132 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android开发中常见的设计模式 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

對于開發人員來說,設計模式有時候就是一道坎,但是設計模式又非常有用,過了這道坎,它可以讓你水平提高一個檔次。而在android開發中,必要的了解一些設計模式又是非常有必要的。對于想系統的學習設計模式的同學,這里推薦2本書。一本是Head First系列的Head Hirst Design Pattern,英文好的可以看英文,可以多讀幾遍。另外一本是大話設計模式。

單例模式

首先了解一些單例模式的概念。

確保某一個類只有一個實例,而且自行實例化并向整個系統提供這個實例。

這樣做有以下幾個優點

  • 對于那些比較耗內存的類,只實例化一次可以大大提高性能,尤其是在移動開發中。
  • 保持程序運行的時候該中始終只有一個實例存在內存中

其實單例有很多種實現方式,但是個人比較傾向于其中1種。可以見單例模式

代碼如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class Singleton {
private static volatile Singleton instance = null;

private Singleton(){
}

public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}

要保證單例,需要做一下幾步

  • 必須防止外部可以調用構造函數進行實例化,因此構造函數必須私有化。
  • 必須定義一個靜態函數獲得該單例
  • 單例使用volatile修飾
  • 使用synchronized 進行同步處理,并且雙重判斷是否為null,我們看到synchronized (Singleton.class)里面又進行了是否為null的判斷,這是因為一個線程進入了該代碼,如果另一個線程在等待,這時候前一個線程創建了一個實例出來完畢后,另一個線程獲得鎖進入該同步代碼,實例已經存在,沒必要再次創建,因此這個判斷是否是null還是必須的。

至于單例的并發測試,可以使用CountDownLatch,使用await()等待鎖釋放,使用countDown()釋放鎖從而達到并發的效果。可以見下面的代碼

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public static void main(String[] args) {
final CountDownLatch latch = new CountDownLatch(1);
int threadCount = 1000;
for (int i = 0; i < threadCount; i++) {
new Thread() {
@Override
public void run() {
try {
latch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Singleton.getInstance().hashCode());
}
}.start();
}
latch.countDown();
}

看看打印出來的hashCode會不會出現不一樣即可,理論上是全部都一樣的。

而在Android中,很多地方用到了單例。

比如Android-Universal-Image-Loader中的單例

1
2
3
4
5
6
7
8
9
10
11
12
private volatile static ImageLoader instance;
/** Returns singleton class instance */
public static ImageLoader getInstance() {
if (instance == null) {
synchronized (ImageLoader.class) {
if (instance == null) {
instance = new ImageLoader();
}
}
}
return instance;
}

比如EventBus中的單例

1
2
3
4
5
6
7
8
9
10
11
private static volatile EventBus defaultInstance;
public static EventBus getDefault() {
if (defaultInstance == null) {
synchronized (EventBus.class) {
if (defaultInstance == null) {
defaultInstance = new EventBus();
}
}
}
return defaultInstance;
}

上面的單例都是比較規規矩矩的,當然實際上有很多單例都是變了一個樣子,單本質還是單例。

如InputMethodManager 中的單例

1
2
3
4
5
6
7
8
9
10
11
static InputMethodManager sInstance;
public static InputMethodManager getInstance() {
synchronized (InputMethodManager.class) {
if (sInstance == null) {
IBinder b = ServiceManager.getService(Context.INPUT_METHOD_SERVICE);
IInputMethodManager service = IInputMethodManager.Stub.asInterface(b);
sInstance = new InputMethodManager(service, Looper.getMainLooper());
}
return sInstance;
}
}

AccessibilityManager 中的單例,看代碼這么長,其實就是進行了一些判斷,還是一個單例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
private static AccessibilityManager sInstance;
public static AccessibilityManager getInstance(Context context) {
synchronized (sInstanceSync) {
if (sInstance == null) {
final int userId;
if (Binder.getCallingUid() == Process.SYSTEM_UID
|| context.checkCallingOrSelfPermission(
Manifest.permission.INTERACT_ACROSS_USERS)
== PackageManager.PERMISSION_GRANTED
|| context.checkCallingOrSelfPermission(
Manifest.permission.INTERACT_ACROSS_USERS_FULL)
== PackageManager.PERMISSION_GRANTED) {
userId = UserHandle.USER_CURRENT;
} else {
userId = UserHandle.myUserId();
}
IBinder iBinder = ServiceManager.getService(Context.ACCESSIBILITY_SERVICE);
IAccessibilityManager service = IAccessibilityManager.Stub.asInterface(iBinder);
sInstance = new AccessibilityManager(context, service, userId);
}
}
return sInstance;
}

當然單例還有很多種寫法,比如惡漢式,有興趣的自己去了解就好了。

最后,我們應用一下單例模式。典型的一個應用就是管理我們的Activity,下面這個可以作為一個工具類,代碼也很簡單,也不做什么解釋了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
public class ActivityManager {

private static volatile ActivityManager instance;
private Stack<Activity> mActivityStack = new Stack<Activity>();

private ActivityManager(){

}

public static ActivityManager getInstance(){
if (instance == null) {
synchronized (ActivityManager.class) {
if (instance == null) {
instance = new ActivityManager();
}
}
return instance;
}

public void addActicity(Activity act){
mActivityStack.push(act);
}

public void removeActivity(Activity act){
mActivityStack.remove(act);
}

public void killMyProcess(){
int nCount = mActivityStack.size();
for (int i = nCount - 1; i >= 0; i--) {
Activity activity = mActivityStack.get(i);
activity.finish();
}

mActivityStack.clear();
android.os.Process.killProcess(android.os.Process.myPid());
}
}

這個類可以在開源中國的幾個客戶端中找到類似的源碼

  • Git@OSC中的AppManager

  • android-app中的AppManager

以上兩個類是一樣的,沒區別。

Build模式

了解了單例模式,接下來介紹另一個常見的模式——Builder模式。

那么什么是Builder模式呢。你通過搜索,會發現大部分網上的定義都是

將一個復雜對象的構建與它的表示分離,使得同樣的構建過程可以創建不同的表示

但是看完這個定義,并沒有什么卵用,你依然不知道什么是Builder設計模式。在此個人的態度是學習設計模式這種東西,不要過度在意其定義,定義往往是比較抽象的,學習它最好的例子就是通過樣例代碼。

我們通過一個例子來引出Builder模式。假設有一個Person類,我們通過該Person類來構建一大批人,這個Person類里有很多屬性,最常見的比如name,age,weight,height等等,并且我們允許這些值不被設置,也就是允許為null,該類的定義如下。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
public class Person {
private String name;
private int age;
private double height;
private double weight;

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public double getHeight() {
return height;
}

public void setHeight(double height) {
this.height = height;
}

public double getWeight() {
return weight;
}

public void setWeight(double weight) {
this.weight = weight;
}
}

然后我們為了方便可能會定義一個構造方法。

1
2
3
4
5
6
public Person(String name, int age, double height, double weight) {
this.name = name;
this.age = age;
this.height = height;
this.weight = weight;
}

或許為了方便new對象,你還會定義一個空的構造方法

1
2
public Person() {
}

甚至有時候你很懶,只想傳部分參數,你還會定義如下類似的構造方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public Person(String name) {
this.name = name;
}

public Person(String name, int age) {
this.name = name;
this.age = age;
}

public Person(String name, int age, double height) {
this.name = name;
this.age = age;
this.height = height;
}

于是你就可以這樣創建各個需要的對象

1
2
3
4
5
Person p1=new Person();
Person p2=new Person("張三");
Person p3=new Person("李四",18);
Person p4=new Person("王五",21,180);
Person p5=new Person("趙六",17,170,65.4);

可以想象一下這樣創建的壞處,最直觀的就是四個參數的構造函數的最后面的兩個參數到底是什么意思,可讀性不怎么好,如果不點擊看源碼,鬼知道哪個是weight哪個是height。還有一個問題就是當有很多參數時,編寫這個構造函數就會顯得異常麻煩,這時候如果換一個角度,試試Builder模式,你會發現代碼的可讀性一下子就上去了。

我們給Person增加一個靜態內部類Builder類,并修改Person類的構造函數,代碼如下。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
public class Person {
private String name;
private int age;
private double height;
private double weight;

privatePerson(Builder builder) {
this.name=builder.name;
this.age=builder.age;
this.height=builder.height;
this.weight=builder.weight;
}
public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public double getHeight() {
return height;
}

public void setHeight(double height) {
this.height = height;
}

public double getWeight() {
return weight;
}

public void setWeight(double weight) {
this.weight = weight;
}

static class Builder{
private String name;
private int age;
private double height;
private double weight;
public Builder name(String name){
this.name=name;
return this;
}
public Builder age(int age){
this.age=age;
return this;
}
public Builder height(double height){
this.height=height;
return this;
}

public Builder weight(double weight){
this.weight=weight;
return this;
}

public Person build(){
return new Person(this);
}
}
}

從上面的代碼中我們可以看到,我們在Builder類里定義了一份與Person類一模一樣的變量,通過一系列的成員函數進行設置屬性值,但是返回值都是this,也就是都是Builder對象,最后提供了一個build函數用于創建Person對象,返回的是Person對象,對應的構造函數在Person類中進行定義,也就是構造函數的入參是Builder對象,然后依次對自己的成員變量進行賦值,對應的值都是Builder對象中的值。此外Builder類中的成員函數返回Builder對象自身的另一個作用就是讓它支持鏈式調用,使代碼可讀性大大增強。

于是我們就可以這樣創建Person類。

1
2
3
4
5
6
7
Person.Builder builder=new Person.Builder();
Person person=builder
.name("張三")
.age(18)
.height(178.5)
.weight(67.4)
.build();

有沒有覺得創建過程一下子就變得那么清晰了。對應的值是什么屬性一目了然,可讀性大大增強。

其實在Android中, Builder模式也是被大量的運用。比如常見的對話框的創建

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
AlertDialog.Builder builder=new AlertDialog.Builder(this);
AlertDialog dialog=builder.setTitle("標題")
.setIcon(android.R.drawable.ic_dialog_alert)
.setView(R.layout.myview)
.setPositiveButton(R.string.positive, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {

}
})
.setNegativeButton(R.string.negative, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {

}
})
.create();
dialog.show();

其實在java中有兩個常見的類也是Builder模式,那就是StringBuilder和StringBuffer,只不過其實現過程簡化了一點罷了。

我們再找找Builder模式在各個框架中的應用。

如Gson中的GsonBuilder,代碼太長了,就不貼了,有興趣自己去看源碼,這里只貼出其Builder的使用方法。

1
2
3
4
5
6
GsonBuilder builder=new GsonBuilder();
Gson gson=builder.setPrettyPrinting()
.disableHtmlEscaping()
.generateNonExecutableJson()
.serializeNulls()
.create();

還有EventBus中也有一個Builder,只不過這個Builder外部訪問不到而已,因為它的構造函數不是public的,但是你可以在EventBus這個類中看到他的應用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public static EventBusBuilder builder() {
return new EventBusBuilder();
}
public EventBus() {
this(DEFAULT_BUILDER);
}
EventBus(EventBusBuilder builder) {
subscriptionsByEventType = new HashMap<Class<?>, CopyOnWriteArrayList<Subscription>>();
typesBySubscriber = new HashMap<Object, List<Class<?>>>();
stickyEvents = new ConcurrentHashMap<Class<?>, Object>();
mainThreadPoster = new HandlerPoster(this, Looper.getMainLooper(), 10);
backgroundPoster = new BackgroundPoster(this);
asyncPoster = new AsyncPoster(this);
subscriberMethodFinder = new SubscriberMethodFinder(builder.skipMethodVerificationForClasses);
logSubscriberExceptions = builder.logSubscriberExceptions;
logNoSubscriberMessages = builder.logNoSubscriberMessages;
sendSubscriberExceptionEvent = builder.sendSubscriberExceptionEvent;
sendNoSubscriberEvent = builder.sendNoSubscriberEvent;
throwSubscriberException = builder.throwSubscriberException;
eventInheritance = builder.eventInheritance;
executorService = builder.executorService;
}

再看看著名的網絡請求框架OkHttp

1
2
3
4
5
Request.Builder builder=new Request.Builder();
Request request=builder.addHeader("","")
.url("")
.post(body)
.build();

除了Request外,Response也是通過Builder模式創建的。貼一下Response的構造函數

1
2
3
4
5
6
7
8
9
10
11
12
private Response(Builder builder) {
this.request = builder.request;
this.protocol = builder.protocol;
this.code = builder.code;
this.message = builder.message;
this.handshake = builder.handshake;
this.headers = builder.headers.build();
this.body = builder.body;
this.networkResponse = builder.networkResponse;
this.cacheResponse = builder.cacheResponse;
this.priorResponse = builder.priorResponse;
}

可見各大框架中大量的運用了Builder模式。最后總結一下

  • 定義一個靜態內部類Builder,內部的成員變量和外部類一樣
  • Builder類通過一系列的方法用于成員變量的賦值,并返回當前對象本身(this)
  • Builder類提供一個build方法或者create方法用于創建對應的外部類,該方法內部調用了外部類的一個私有構造函數,該構造函數的參數就是內部類Builder
  • 外部類提供一個私有構造函數供內部類調用,在該構造函數中完成成員變量的賦值,取值為Builder對象中對應的值

觀察者模式

前面介紹了單例模式和Builder模式,這部分著重介紹一下觀察者模式。先看下這個模式的定義。

定義對象間的一種一對多的依賴關系,當一個對象的狀態發送改變時,所有依賴于它的對象都能得到通知并被自動更新

還是那句話,定義往往是抽象的,要深刻的理解定義,你需要自己動手實踐一下。

先來講幾個情景。

  • 情景1

    有一種短信服務,比如天氣預報服務,一旦你訂閱該服務,你只需按月付費,付完費后,每天一旦有天氣信息更新,它就會及時向你發送最新的天氣信息。

  • 情景2

    雜志的訂閱,你只需向郵局訂閱雜志,繳納一定的費用,當有新的雜志時,郵局會自動將雜志送至你預留的地址。

觀察上面兩個情景,有一個共同點,就是我們無需每時每刻關注我們感興趣的東西,我們只需做的就是訂閱感興趣的事物,比如天氣預報服務,雜志等,一旦我們訂閱的事物發生變化,比如有新的天氣預報信息,新的雜志等,被訂閱的事物就會即時通知到訂閱者,即我們。而這些被訂閱的事物可以擁有多個訂閱者,也就是一對多的關系。當然,嚴格意義上講,這個一對多可以包含一對一,因為一對一是一對多的特例,沒有特殊說明,本文的一對多包含了一對一。

現在你反過頭來看看觀察者模式的定義,你是不是豁然開朗了。

然后我們看一下觀察者模式的幾個重要組成。

  • 觀察者,我們稱它為Observer,有時候我們也稱它為訂閱者,即Subscriber
  • 被觀察者,我們稱它為Observable,即可以被觀察的東西,有時候還會稱之為主題,即Subject

至于觀察者模式的具體實現,這里帶帶大家實現一下場景一,其實java中提供了Observable類和Observer接口供我們快速的實現該模式,但是為了加深印象,我們不使用這兩個類。

場景1中我們感興趣的事情是天氣預報,于是,我們應該定義一個Weather實體類。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class Weather {
private String description;

public Weather(String description) {
this.description = description;
}

public String getDescription() {
return description;
}

public void setDescription(String description) {
this.description = description;
}

@Override
public String toString() {
return "Weather{" +
"description='" + description + '\'' +
'}';
}
}

然后定義我們的被觀察者,我們想要這個被觀察者能夠通用,將其定義成泛型。內部應該暴露register和unregister方法供觀察者訂閱和取消訂閱,至于觀察者的保存,直接用ArrayList即可,此外,當有主題內容發送改變時,會即時通知觀察者做出反應,因此應該暴露一個notifyObservers方法,以上方法的具體實現見如下代碼。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public class Observable<T> {
List<Observer<T>> mObservers = new ArrayList<Observer<T>>();

public void register(Observer<T> observer) {
if (observer == null) {
throw new NullPointerException("observer == null");
}
synchronized (this) {
if (!mObservers.contains(observer))
mObservers.add(observer);
}
}

public synchronized void unregister(Observer<T> observer) {
mObservers.remove(observer);
}

public void notifyObservers(T data) {
for (Observer<T> observer : mObservers) {
observer.onUpdate(this, data);
}
}

}

而我們的觀察者,只需要實現一個觀察者的接口Observer,該接口也是泛型的。其定義如下。

1
2
3
public interface Observer<T> {
void onUpdate(Observable<T> observable,T data);
}

一旦訂閱的主題發送變換就會回調該接口。

我們來使用一下,我們定義了一個天氣變換的主題,也就是被觀察者,還有兩個觀察者觀察天氣變換,一旦變換了,就打印出天氣信息,注意一定要調用被觀察者的register進行注冊,否則會收不到變換信息。而一旦不敢興趣了,直接調用unregister方法進行取消注冊即可

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public class Main {
public static void main(String [] args){
Observable<Weather> observable=new Observable<Weather>();
Observer<Weather> observer1=new Observer<Weather>() {
@Override
public void onUpdate(Observable<Weather> observable, Weather data) {
System.out.println("觀察者1:"+data.toString());
}
};
Observer<Weather> observer2=new Observer<Weather>() {
@Override
public void onUpdate(Observable<Weather> observable, Weather data) {
System.out.println("觀察者2:"+data.toString());
}
};

observable.register(observer1);
observable.register(observer2);


Weather weather=new Weather("晴轉多云");
observable.notifyObservers(weather);

Weather weather1=new Weather("多云轉陰");
observable.notifyObservers(weather1);

observable.unregister(observer1);

Weather weather2=new Weather("臺風");
observable.notifyObservers(weather2);

}
}

最后的輸出結果也是沒有什么問題的,如下

觀察者1:Weather{description=’晴轉多云’}
觀察者2:Weather{description=’晴轉多云’}
觀察者1:Weather{description=’多云轉陰’}
觀察者2:Weather{description=’多云轉陰’}
觀察者2:Weather{description=’臺風’}

接下來我們看看觀察者模式在android中的應用。我們從最簡單的開始。還記得我們為一個Button設置點擊事件的代碼嗎。

1
2
3
4
5
6
7
Button btn=new Button(this);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Log.e("TAG","click");
}
});

其實嚴格意義上講,這個最多算是回調,但是我們可以將其看成是一對一的觀察者模式,即只有一個觀察者。

其實只要是set系列的設置監聽器的方法最多都只能算回調,但是有一些監聽器式add進去的,這種就是觀察者模式了,比如RecyclerView中的addOnScrollListener方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
private List<OnScrollListener> mScrollListeners;
public void addOnScrollListener(OnScrollListener listener) {
if (mScrollListeners == null) {
mScrollListeners = new ArrayList<OnScrollListener>();
}
mScrollListeners.add(listener);
}
public void removeOnScrollListener(OnScrollListener listener) {
if (mScrollListeners != null) {
mScrollListeners.remove(listener);
}
}
public void clearOnScrollListeners() {
if (mScrollListeners != null) {
mScrollListeners.clear();
}
}

然后有滾動事件時便會觸發觀察者進行方法回調

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public abstract static class OnScrollListener {
public void onScrollStateChanged(RecyclerView recyclerView, int newState){}
public void onScrolled(RecyclerView recyclerView, int dx, int dy){}
}

void dispatchOnScrolled(int hresult, int vresult) {
//...
if (mScrollListeners != null) {
for (int i = mScrollListeners.size() - 1; i >= 0; i--) {
mScrollListeners.get(i).onScrolled(this, hresult, vresult);
}
}
}
void dispatchOnScrollStateChanged(int state) {
//...
if (mScrollListeners != null) {
for (int i = mScrollListeners.size() - 1; i >= 0; i--) {
mScrollListeners.get(i).onScrollStateChanged(this, state);
}
}
}

類似的方法很多很多,都是add監聽器系列的方法,這里也不再舉例。

還有一個地方就是Android的廣播機制,其本質也是觀察者模式,這里為了簡單方便,直接拿本地廣播的代碼說明,即LocalBroadcastManager。

我們平時使用本地廣播主要就是下面四個方法

1
2
3
4
LocalBroadcastManager localBroadcastManager=LocalBroadcastManager.getInstance(this);
localBroadcastManager.registerReceiver(BroadcastReceiver receiver, IntentFilter filter);
localBroadcastManager.unregisterReceiver(BroadcastReceiver receiver);
localBroadcastManager.sendBroadcast(Intent intent)

調用registerReceiver方法注冊廣播,調用unregisterReceiver方法取消注冊,之后直接使用sendBroadcast發送廣播,發送廣播之后,注冊的廣播會收到對應的廣播信息,這就是典型的觀察者模式。具體的源代碼這里也不貼。

android系統中的觀察者模式還有很多很多,有興趣的自己去挖掘,接下來我們看一下一些開源框架中的觀察者模式。一說到開源框架,你首先想到的應該是EventBus。沒錯,EventBus也是基于觀察者模式的。

觀察者模式的三個典型方法它都具有,即注冊,取消注冊,發送事件

1
2
3
4
EventBus.getDefault().register(Object subscriber);
EventBus.getDefault().unregister(Object subscriber);

EventBus.getDefault().post(Object event);

內部源碼也不展開了。接下來看一下重量級的庫,它就是RxJava,由于學習曲線的陡峭,這個庫讓很多人望而止步。

創建一個被觀察者

1
2
3
4
5
6
7
8
9
Observable<String> myObservable = Observable.create(
new Observable.OnSubscribe<String>() {
@Override
public void call(Subscriber<? super String> sub) {
sub.onNext("Hello, world!");
sub.onCompleted();
}
}
);

創建一個觀察者,也就是訂閱者

1
2
3
4
5
6
7
8
9
10
Subscriber<String> mySubscriber = new Subscriber<String>() {
@Override
public void onNext(String s) { System.out.println(s); }

@Override
public void onCompleted() { }

@Override
public void onError(Throwable e) { }
};

觀察者進行事件的訂閱

1
myObservable.subscribe(mySubscriber);

具體源碼也不展開,不過RxJava這個開源庫的源碼個人還是建議很值得去看一看的。

總之,在Android中觀察者模式還是被用得很頻繁的。

原型模式

這部分介紹的模式其實很簡單,即原型模式,按照慣例,先看定義。

用原型實例指定創建對象的種類,并通過拷貝這些原型創建新的對象。

這是什么鬼哦,本寶寶看不懂!不必過度在意這些定義,自己心里明白就ok了。沒代碼你說個jb。

首先我們定義一個Person類

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
public class Person{
private String name;
private int age;
private double height;
private double weight;

public Person(){

}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public double getHeight() {
return height;
}

public void setHeight(double height) {
this.height = height;
}

public double getWeight() {
return weight;
}

public void setWeight(double weight) {
this.weight = weight;
}

@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", height=" + height +
", weight=" + weight +
'}';
}
}

要實現原型模式,只需要按照下面的幾個步驟去實現即可。

  • 實現Cloneable接口
1
2
3
public class Person implements Cloneable{

}
  • 重寫Object的clone方法
1
2
3
4
@Override
public Object clone(){
return null;
}
  • 實現clone方法中的拷貝邏輯
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Override
public Object clone(){
Person person=null;
try {
person=(Person)super.clone();
person.name=this.name;
person.weight=this.weight;
person.height=this.height;
person.age=this.age;
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return person;
}

測試一下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class Main {
public static void main(String [] args){
Person p=new Person();
p.setAge(18);
p.setName("張三");
p.setHeight(178);
p.setWeight(65);
System.out.println(p);

Person p1= (Person) p.clone();
System.out.println(p1);

p1.setName("李四");
System.out.println(p);
System.out.println(p1);
}
}

輸出結果如下

Person{name=’張三’, age=18, height=178.0, weight=65.0}
Person{name=’張三’, age=18, height=178.0, weight=65.0}
Person{name=’張三’, age=18, height=178.0, weight=65.0}
Person{name=’李四’, age=18, height=178.0, weight=65.0}

試想一下,兩個不同的人,除了姓名不一樣,其他三個屬性都一樣,用原型模式進行拷貝就會顯得異常簡單,這也是原型模式的應用場景之一。

一個對象需要提供給其他對象訪問,而且各個調用者可能都需要修改其值時,可以考慮使用原型模式拷貝多個對象供調用者使用,即保護性拷貝。

但是假設Person類里還有一個屬性叫興趣愛好,是一個List集合,就像這樣子

1
2
3
4
5
6
7
8
9
private ArrayList<String> hobbies=new ArrayList<String>();

public ArrayList<String> getHobbies() {
return hobbies;
}

public void setHobbies(ArrayList<String> hobbies) {
this.hobbies = hobbies;
}

在進行拷貝的時候要格外注意,如果你直接按之前的代碼那樣拷貝

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Override
public Object clone(){
Person person=null;
try {
person=(Person)super.clone();
person.name=this.name;
person.weight=this.weight;
person.height=this.height;
person.age=this.age;
person.hobbies=this.hobbies;
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return person;
}

會帶來一個問題

使用測試代碼進行測試

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public class Main {
public static void main(String [] args){
Person p=new Person();
p.setAge(18);
p.setName("張三");
p.setHeight(178);
p.setWeight(65);
ArrayList <String> hobbies=new ArrayList<String>();
hobbies.add("籃球");
hobbies.add("編程");
hobbies.add("長跑");
p.setHobbies(hobbies);
System.out.println(p);

Person p1= (Person) p.clone();
System.out.println(p1);

p1.setName("李四");
p1.getHobbies().add("游泳");
System.out.println(p);
System.out.println(p1);
}
}

我們拷貝了一個對象,并添加了一個興趣愛好進去,看下打印結果

Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑]}
Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑]}
Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑, 游泳]}
Person{name=’李四’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑, 游泳]}

你會發現原來的對象的hobby也發生了變換。

其實導致這個問題的本質原因是我們只進行了淺拷貝,也就是只拷貝了引用,最終兩個對象指向的引用是同一個,一個發生變化另一個也會發生變換,顯然解決方法就是使用深拷貝。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
@Override
public Object clone(){
Person person=null;
try {
person=(Person)super.clone();
person.name=this.name;
person.weight=this.weight;
person.height=this.height;
person.age=this.age;

person.hobbies=(ArrayList<String>)this.hobbies.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return person;
}

注意person.hobbies=(ArrayList)this.hobbies.clone();,不再是直接引用而是進行了一份拷貝。再運行一下,就會發現原來的對象不會再發生變化了。

Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑]}
Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑]}
Person{name=’張三’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑]}
Person{name=’李四’, age=18, height=178.0, weight=65.0, hobbies=[籃球, 編程, 長跑, 游泳]}

其實有時候我們會更多的看到原型模式的另一種寫法。

  • 在clone函數里調用構造函數,構造函數的入參是該類對象。
1
2
3
4
@Override
public Object clone(){
return new Person(this);
}
  • 在構造函數中完成拷貝邏輯
1
2
3
4
5
6
7
public Person(Person person){
this.name=person.name;
this.weight=person.weight;
this.height=person.height;
this.age=person.age;
this.hobbies= new ArrayList<String>(hobbies);
}

其實都差不多,只是寫法不一樣。

現在來挖挖android中的原型模式。

先看Bundle類,該類實現了Cloneable接口

1
2
3
4
5
6
7
8
9
public Object clone() {
return new Bundle(this);
}
public Bundle(Bundle b) {
super(b);

mHasFds = b.mHasFds;
mFdsKnown = b.mFdsKnown;
}

然后是Intent類,該類也實現了Cloneable接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
@Override
public Object clone() {
return new Intent(this);
}
public Intent(Intent o) {
this.mAction = o.mAction;
this.mData = o.mData;
this.mType = o.mType;
this.mPackage = o.mPackage;
this.mComponent = o.mComponent;
this.mFlags = o.mFlags;
this.mContentUserHint = o.mContentUserHint;
if (o.mCategories != null) {
this.mCategories = new ArraySet<String>(o.mCategories);
}
if (o.mExtras != null) {
this.mExtras = new Bundle(o.mExtras);
}
if (o.mSourceBounds != null) {
this.mSourceBounds = new Rect(o.mSourceBounds);
}
if (o.mSelector != null) {
this.mSelector = new Intent(o.mSelector);
}
if (o.mClipData != null) {
this.mClipData = new ClipData(o.mClipData);
}
}

用法也顯得十分簡單,一旦我們要用的Intent與現有的一個Intent很多東西都是一樣的,那我們就可以直接拷貝現有的Intent,再修改不同的地方,便可以直接使用。

1
2
3
4
5
6
Uri uri = Uri.parse("smsto:10086");
Intent shareIntent = new Intent(Intent.ACTION_SENDTO, uri);
shareIntent.putExtra("sms_body", "hello");

Intent intent = (Intent)shareIntent.clone() ;
startActivity(intent);

網絡請求中一個最常見的開源庫OkHttp中,也應用了原型模式。它就在OkHttpClient這個類中,它實現了Cloneable接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/** Returns a shallow copy of this OkHttpClient. */
@Override
public OkHttpClient clone() {
return new OkHttpClient(this);
}
private OkHttpClient(OkHttpClient okHttpClient) {
this.routeDatabase = okHttpClient.routeDatabase;
this.dispatcher = okHttpClient.dispatcher;
this.proxy = okHttpClient.proxy;
this.protocols = okHttpClient.protocols;
this.connectionSpecs = okHttpClient.connectionSpecs;
this.interceptors.addAll(okHttpClient.interceptors);
this.networkInterceptors.addAll(okHttpClient.networkInterceptors);
this.proxySelector = okHttpClient.proxySelector;
this.cookieHandler = okHttpClient.cookieHandler;
this.cache = okHttpClient.cache;
this.internalCache = cache != null ? cache.internalCache : okHttpClient.internalCache;
this.socketFactory = okHttpClient.socketFactory;
this.sslSocketFactory = okHttpClient.sslSocketFactory;
this.hostnameVerifier = okHttpClient.hostnameVerifier;
this.certificatePinner = okHttpClient.certificatePinner;
this.authenticator = okHttpClient.authenticator;
this.connectionPool = okHttpClient.connectionPool;
this.network = okHttpClient.network;
this.followSslRedirects = okHttpClient.followSslRedirects;
this.followRedirects = okHttpClient.followRedirects;
this.retryOnConnectionFailure = okHttpClient.retryOnConnectionFailure;
this.connectTimeout = okHttpClient.connectTimeout;
this.readTimeout = okHttpClient.readTimeout;
this.writeTimeout = okHttpClient.writeTimeout;
}

正如開頭的注釋Returns a shallow copy of this OkHttpClient,該clone方法返回了一個當前對象的淺拷貝對象。

至于其他框架中的原型模式,請讀者自行發現。

策略模式

看下策略模式的定義

策略模式定義了一些列的算法,并將每一個算法封裝起來,而且使它們還可以相互替換。策略模式讓算法獨立于使用它的客戶而獨立變換。

乍一看,也沒看出個所以然來。舉個栗子吧。

假設我們要出去旅游,而去旅游出行的方式有很多,有步行,有坐火車,有坐飛機等等。而如果不使用任何模式,我們的代碼可能就是這樣子的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
public class TravelStrategy {
enum Strategy{
WALK,PLANE,SUBWAY
}
private Strategy strategy;
public TravelStrategy(Strategy strategy){
this.strategy=strategy;
}

public void travel(){
if(strategy==Strategy.WALK){
print("walk");
}else if(strategy==Strategy.PLANE){
print("plane");
}else if(strategy==Strategy.SUBWAY){
print("subway");
}
}

public void print(String str){
System.out.println("出行旅游的方式為:"+str);
}

public static void main(String[] args) {
TravelStrategy walk=new TravelStrategy(Strategy.WALK);
walk.travel();

TravelStrategy plane=new TravelStrategy(Strategy.PLANE);
plane.travel();

TravelStrategy subway=new TravelStrategy(Strategy.SUBWAY);
subway.travel();
}
}

這樣做有一個致命的缺點,一旦出行的方式要增加,我們就不得不增加新的else if語句,而這違反了面向對象的原則之一,對修改封閉。而這時候,策略模式則可以完美的解決這一切。

首先,需要定義一個策略接口。

1
2
3
4

public interface Strategy {
void travel();
}

然后根據不同的出行方式實行對應的接口

1
2
3
4
5
6
7
8
public class WalkStrategy implements Strategy{

@Override
public void travel() {
System.out.println("walk");
}

}
1
2
3
4
5
6
7
8
public class PlaneStrategy implements Strategy{

@Override
public void travel() {
System.out.println("plane");
}

}
1
2
3
4
5
6
7
8
public class SubwayStrategy implements Strategy{

@Override
public void travel() {
System.out.println("subway");
}

}

此外還需要一個包裝策略的類,并調用策略接口中的方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class TravelContext {
Strategy strategy;

public Strategy getStrategy() {
return strategy;
}

public void setStrategy(Strategy strategy) {
this.strategy = strategy;
}

public void travel() {
if (strategy != null) {
strategy.travel();
}
}
}

測試一下代碼

1
2
3
4
5
6
7
8
9
10
11
public class Main {
public static void main(String[] args) {
TravelContext travelContext=new TravelContext();
travelContext.setStrategy(new PlaneStrategy());
travelContext.travel();
travelContext.setStrategy(new WalkStrategy());
travelContext.travel();
travelContext.setStrategy(new SubwayStrategy());
travelContext.travel();
}
}

輸出結果如下

plane
walk
subway

可以看到,應用了策略模式后,如果我們想增加新的出行方式,完全不必要修改現有的類,我們只需要實現策略接口即可,這就是面向對象中的對擴展開放準則。假設現在我們增加了一種自行車出行的方式。只需新增一個類即可。

1
2
3
4
5
6
7
8
public class BikeStrategy implements Strategy{

@Override
public void travel() {
System.out.println("bike");
}

}

之后設置策略即可

1
2
3
4
5
6
7
public class Main {
public static void main(String[] args) {
TravelContext travelContext=new TravelContext();
travelContext.setStrategy(new BikeStrategy());
travelContext.travel();
}
}

而在Android的系統源碼中,策略模式也是應用的相當廣泛的.最典型的就是屬性動畫中的應用.

我們知道,在屬性動畫中,有一個東西叫做插值器,它的作用就是根據時間流逝的百分比來來計算出當前屬性值改變的百分比.

我們使用屬性動畫的時候,可以通過set方法對插值器進行設置.可以看到內部維持了一個時間插值器的引用,并設置了getter和setter方法,默認情況下是先加速后減速的插值器,set方法如果傳入的是null,則是線性插值器。而時間插值器TimeInterpolator是個接口,有一個接口繼承了該接口,就是Interpolator這個接口,其作用是為了保持兼容

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
private static final TimeInterpolator sDefaultInterpolator =
new AccelerateDecelerateInterpolator();
private TimeInterpolator mInterpolator = sDefaultInterpolator;
@Override
public void setInterpolator(TimeInterpolator value) {
if (value != null) {
mInterpolator = value;
} else {
mInterpolator = new LinearInterpolator();
}
}

@Override
public TimeInterpolator getInterpolator() {
return mInterpolator;
}
1
2
3
4
5
6
public interface Interpolator extends TimeInterpolator {
// A new interface, TimeInterpolator, was introduced for the new android.animation
// package. This older Interpolator interface extends TimeInterpolator so that users of
// the new Animator-based animations can use either the old Interpolator implementations or
// new classes that implement TimeInterpolator directly.
}

此外還有一個BaseInterpolator插值器實現了Interpolator接口,并且是一個抽象類

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
abstract public class BaseInterpolator implements Interpolator {
private int mChangingConfiguration;
/**
* @hide
*/
public int getChangingConfiguration() {
return mChangingConfiguration;
}

/**
* @hide
*/
void setChangingConfiguration(int changingConfiguration) {
mChangingConfiguration = changingConfiguration;
}
}

平時我們使用的時候,通過設置不同的插值器,實現不同的動畫速率變換效果,比如線性變換,回彈,自由落體等等。這些都是插值器接口的具體實現,也就是具體的插值器策略。我們略微來看幾個策略。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class LinearInterpolator extends BaseInterpolator implements NativeInterpolatorFactory {

public LinearInterpolator() {
}

public LinearInterpolator(Context context, AttributeSet attrs) {
}

public float getInterpolation(float input) {
return input;
}

/** @hide */
@Override
public long createNativeInterpolator() {
return NativeInterpolatorFactoryHelper.createLinearInterpolator();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public class AccelerateDecelerateInterpolator extends BaseInterpolator
implements NativeInterpolatorFactory {
public AccelerateDecelerateInterpolator() {
}

@SuppressWarnings({"UnusedDeclaration"})
public AccelerateDecelerateInterpolator(Context context, AttributeSet attrs) {
}

public float getInterpolation(float input) {
return (float)(Math.cos((input + 1) * Math.PI) / 2.0f) + 0.5f;
}

/** @hide */
@Override
public long createNativeInterpolator() {
return NativeInterpolatorFactoryHelper.createAccelerateDecelerateInterpolator();
}
}

內部使用的時候直接調用getInterpolation方法就可以返回對應的值了,也就是屬性值改變的百分比。

屬性動畫中另外一個應用策略模式的地方就是估值器,它的作用是根據當前屬性改變的百分比來計算改變后的屬性值。該屬性和插值器是類似的,有幾個默認的實現。其中TypeEvaluator是一個接口。

1
2
3
4
5
public interface TypeEvaluator<T> {

public T evaluate(float fraction, T startValue, T endValue);

}
1
2
3
4
5
6
7
public class IntEvaluator implements TypeEvaluator<Integer> {

public Integer evaluate(float fraction, Integer startValue, Integer endValue) {
int startInt = startValue;
return (int)(startInt + fraction * (endValue - startInt));
}
}
1
2
3
4
5
6
7
public class FloatEvaluator implements TypeEvaluator<Number> {

public Float evaluate(float fraction, Number startValue, Number endValue) {
float startFloat = startValue.floatValue();
return startFloat + fraction * (endValue.floatValue() - startFloat);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class PointFEvaluator implements TypeEvaluator<PointF> {

private PointF mPoint;


public PointFEvaluator() {
}

public PointFEvaluator(PointF reuse) {
mPoint = reuse;
}

@Override
public PointF evaluate(float fraction, PointF startValue, PointF endValue) {
float x = startValue.x + (fraction * (endValue.x - startValue.x));
float y = startValue.y + (fraction * (endValue.y - startValue.y));

if (mPoint != null) {
mPoint.set(x, y);
return mPoint;
} else {
return new PointF(x, y);
}
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public class ArgbEvaluator implements TypeEvaluator {
private static final ArgbEvaluator sInstance = new ArgbEvaluator();

public static ArgbEvaluator getInstance() {
return sInstance;
}

public Object evaluate(float fraction, Object startValue, Object endValue) {
int startInt = (Integer) startValue;
int startA = (startInt >> 24) & 0xff;
int startR = (startInt >> 16) & 0xff;
int startG = (startInt >> 8) & 0xff;
int startB = startInt & 0xff;

int endInt = (Integer) endValue;
int endA = (endInt >> 24) & 0xff;
int endR = (endInt >> 16) & 0xff;
int endG = (endInt >> 8) & 0xff;
int endB = endInt & 0xff;

return (int)((startA + (int)(fraction * (endA - startA))) << 24) |
(int)((startR + (int)(fraction * (endR - startR))) << 16) |
(int)((startG + (int)(fraction * (endG - startG))) << 8) |
(int)((startB + (int)(fraction * (endB - startB))));
}
}

上面的都是一些系統實現好的估值策略,在內部調用估值器的evaluate方法即可返回改變后的值了。我們也可以自定義估值策略。這里就不展開了。

當然,在開源框架中,策略模式也是無處不在的。

首先在Volley中,策略模式就能看到。

有一個重試策略接口

1
2
3
4
5
6
7
8
9
10
11
public interface RetryPolicy {


public int getCurrentTimeout();//獲取當前請求用時(用于 Log)


public int getCurrentRetryCount();//獲取已經重試的次數(用于 Log)


public void retry(VolleyError error) throws VolleyError;//確定是否重試,參數為這次異常的具體信息。在請求異常時此接口會被調用,可在此函數實現中拋出傳入的異常表示停止重試。
}

在Volley中,該接口有一個默認的實現DefaultRetryPolicy,Volley 默認的重試策略實現類。主要通過在 retry(…) 函數中判斷重試次數是否達到上限確定是否繼續重試。

1
2
3
public class DefaultRetryPolicy implements RetryPolicy {
...
}

而策略的設置是在Request類中

1
2
3
4
5
6
7
8
9
10
11

public abstract class Request<T> implements Comparable<Request<T>> {
private RetryPolicy mRetryPolicy;
public Request<?> setRetryPolicy(RetryPolicy retryPolicy) {
mRetryPolicy = retryPolicy;
return this;
}
public RetryPolicy getRetryPolicy() {
return mRetryPolicy;
}
}

此外,各大網絡請求框架,或多或少都會使用到緩存,緩存一般會定義一個Cache接口,然后實現不同的緩存策略,如內存緩存,磁盤緩存等等,這個緩存的實現,其實也可以使用策略模式。直接看Volley,里面也有緩存。

定義了一個緩存接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80

/**
* An interface for a cache keyed by a String with a byte array as data.
*/
public interface Cache {
/**
* Retrieves an entry from the cache.
* @param key Cache key
* @return An {@link Entry} or null in the event of a cache miss
*/
public Entry get(String key);

/**
* Adds or replaces an entry to the cache.
* @param key Cache key
* @param entry Data to store and metadata for cache coherency, TTL, etc.
*/
public void put(String key, Entry entry);

/**
* Performs any potentially long-running actions needed to initialize the cache;
* will be called from a worker thread.
*/
public void initialize();

/**
* Invalidates an entry in the cache.
* @param key Cache key
* @param fullExpire True to fully expire the entry, false to soft expire
*/
public void invalidate(String key, boolean fullExpire);

/**
* Removes an entry from the cache.
* @param key Cache key
*/
public void remove(String key);

/**
* Empties the cache.
*/
public void clear();

/**
* Data and metadata for an entry returned by the cache.
*/
public static class Entry {
/** The data returned from cache. */
public byte[] data;

/** ETag for cache coherency. */
public String etag;

/** Date of this response as reported by the server. */
public long serverDate;

/** The last modified date for the requested object. */
public long lastModified;

/** TTL for this record. */
public long ttl;

/** Soft TTL for this record. */
public long softTtl;

/** Immutable response headers as received from server; must be non-null. */
public Map<String, String> responseHeaders = Collections.emptyMap();

/** True if the entry is expired. */
public boolean isExpired() {
return this.ttl < System.currentTimeMillis();
}

/** True if a refresh is needed from the original data source. */
public boolean refreshNeeded() {
return this.softTtl < System.currentTimeMillis();
}
}

}

它有兩個實現類NoCacheDiskBasedCache,使用的時候設置對應的緩存策略即可。

在android開發中,ViewPager是一個使用非常常見的控件,它的使用往往需要伴隨一個Indicator指示器。如果讓你重新實現一個ViewPager,并且帶有Indicator,這時候,你會不會想到用策略模式呢?在你自己寫的ViewPager中(不是系統的,當然你也可以繼承系統的)持有一個策略接口Indicator的變量,通過set方法設置策略,然后ViewPager滑動的時候調用策略接口的對應方法改變指示器。默認提供幾個Indicator接口的實現類,比如圓形指示器CircleIndicator、線性指示器LineIndicator、Tab指示器TabIndicator、圖標指示器IconIndicator 等等等等。有興趣的話自己去實現一個吧。

總結

以上是生活随笔為你收集整理的Android开发中常见的设计模式的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

成人a在线 | 精品色综合 | 人人爽久久涩噜噜噜网站 | 国产精品免费观看国产网曝瓜 | 国产特级毛片aaaaaaa高清 | 日韩中文在线播放 | av免费在线看网站 | 免费观看黄色av | 国产免费久久av | 男女视频91 | 四虎影视欧美 | 亚洲精品一区二区三区新线路 | 免费看精品久久片 | 人人干人人草 | 国产在线小视频 | 国产日女人 | 东方av在 | 91在线观看欧美日韩 | 欧美日韩精品国产 | 天天操天天射天天爱 | 狠狠色狠狠色合久久伊人 | 久久久久国产精品www | 精品久久久久久久 | 黄色一级影院 | 又黄又刺激视频 | www.狠狠插.com | 免费国产在线视频 | 欧美成人一区二区 | 亚洲欧美日韩精品久久奇米一区 | 国产精品久久久久久久久岛 | 成人av在线电影 | 国产精品免费久久久久久久久久中文 | 久草在线免费电影 | 91在线文字幕| 超碰在线成人 | 国产一级高清视频 | 久久久久久久久久久久av | 久久伊人综合 | 国产高清免费在线观看 | 狠狠色丁香久久婷婷综合五月 | www.99av| 激情视频免费在线观看 | 91精品久| 精品国产成人在线 | 天天超碰| 成年人视频在线免费播放 | 久久国产剧场电影 | 少妇18xxxx性xxxx片 | 又黄又刺激视频 | 日韩精品极品视频 | 四虎影视精品成人 | 久久人人精品 | 日本三级不卡 | 欧美日韩免费一区二区 | 国产一区二区在线观看免费 | 黄色一级片视频 | 久久99精品久久久久久清纯直播 | 中文字幕黄网 | 手机av电影在线观看 | 婷婷久久五月天 | 日韩视频一区二区 | 在线免费观看黄网站 | 天天射天天射天天射 | 亚洲国产黄色 | 亚洲国产美女精品久久久久∴ | 国产中文字幕在线观看 | 免费看片日韩 | 女人18毛片a级毛片一区二区 | 韩日精品在线观看 | www色网站| 又爽又黄又无遮挡网站动态图 | 国产精品久久久久久久久久久久久 | 99精品在线播放 | 日韩特级片 | 91av手机在线观看 | 亚洲精品在线资源 | 日韩高清成人 | 国产成人精品一区二区三区在线观看 | 成人影音在线 | 日韩成人精品在线观看 | 久久久精品国产一区二区 | 久久久久综合视频 | 欧美一区二区三区在线 | 免费高清av在线看 | 在线观看黄色的网站 | 韩国三级av在线 | 亚洲国产免费 | 久草视频在线新免费 | 成人国产精品免费 | 色精品视频| 五月香视频在线观看 | 成人黄色片在线播放 | 成年人视频免费在线播放 | 亚洲色图美腿丝袜 | 成人免费视频网站 | 日韩网站在线观看 | 人人玩人人添人人澡超碰 | 一区在线观看 | 亚洲一区二区三区四区精品 | a级国产毛片 | 女人18毛片90分钟 | 菠萝菠萝蜜在线播放 | 一区二区三区四区免费视频 | 少妇视频在线播放 | 91激情视频在线 | 丁香六月在线 | 亚洲黄色激情小说 | 免费看十八岁美女 | 国产一级在线观看 | 麻豆传媒视频在线播放 | 最新av在线播放 | 国产免费观看久久 | 青青河边草手机免费 | 激情片av | 成人一级在线观看 | 性日韩欧美在线视频 | 国产综合福利在线 | 天天干天天干天天干 | 日韩中文字幕国产 | 亚洲日本黄色 | 日韩午夜小视频 | 97av视频| 亚洲一区免费在线 | 亚洲精品一区二区三区新线路 | 国产精品成人在线观看 | 日韩啪视频| 91亚色免费视频 | 在线成人免费 | 精品国产一区二区三区日日嗨 | 91精品在线视频 | 黄色免费大片 | 亚洲视频精选 | 992tv成人免费看片 | 中文字幕在线一区二区三区 | 天天爱天天爽 | 国产在线看一区 | 中文字幕区 | 色狠狠久久av五月综合 | 夜夜夜夜夜夜操 | 欧美视频xxx | 中文字幕制服丝袜av久久 | 亚洲 综合 精品 | 国产高清日韩 | 欧美成人理伦片 | 久久久久久久久久久久99 | 手机在线黄色网址 | 色干干| 久久视频在线观看免费 | 97中文字幕| 特级黄色片免费看 | 国产成人精品久久二区二区 | 久操视频在线观看 | 婷婷激情综合网 | 亚洲精品xxx| 婷婷久久综合网 | 亚洲精品综合久久 | 欧美亚洲精品一区 | 91精品国自产在线偷拍蜜桃 | av观看久久久 | 日韩视频一区二区 | 国产精品精品久久久 | 四虎4hu永久免费 | 懂色av懂色av粉嫩av分享吧 | 在线色资源 | 国产精品黄色影片导航在线观看 | 亚洲婷婷伊人 | 免费亚洲精品 | 一区二区中文字幕在线观看 | 最新日韩电影 | 午夜精品视频在线 | 亚洲视频精选 | 狠狠综合 | 国产视频第二页 | 一区二区三区免费在线观看 | 在线视频 日韩 | 国产精品一区二区三区在线播放 | 最近高清中文在线字幕在线观看 | 亚洲网久久 | 欧美日韩成人 | 色视频国产直接看 | 91精品国产99久久久久久红楼 | 97超级碰碰碰视频在线观看 | 国产高清在线免费观看 | 最新av网址大全 | 玖玖爱免费视频 | av先锋影音少妇 | 丝袜一区在线 | 91久久久国产精品 | 伊人激情网| 黄色91在线观看 | 久久久国产精品网站 | 国产99久久99热这里精品5 | 婷婷激情网站 | 中文字幕网站视频在线 | av在线播放快速免费阴 | 中文在线免费视频 | 精品久久久网 | 91麻豆精品国产91久久久久久 | 色夜影院 | 天天综合91 | 三级黄免费看 | 亚洲jizzjizz日本少妇 | 欧美亚洲国产日韩 | www欧美色 | 成人黄大片视频在线观看 | 激情影音| 欧美日韩国语 | 国产成人免费观看久久久 | 欧美精品一区二区在线播放 | 久久久久久网站 | 亚洲无毛专区 | 中文字幕在线久一本久 | 色99久久| 麻豆 videos | 伊人久久国产 | 久久五月婷婷丁香社区 | 亚洲日韩欧美视频 | 亚洲国产wwwccc36天堂 | 91精品一| 欧美日韩国产精品一区二区亚洲 | 免费在线看成人av | 久久99国产精品二区护士 | 在线免费精品视频 | 久久久久久毛片精品免费不卡 | 日日天天狠狠 | 国产精品视频永久免费播放 | 最近字幕在线观看第一季 | 麻豆播放 | 日韩中文在线视频 | 成人在线免费小视频 | 日本中文在线观看 | 久久久久久高潮国产精品视 | av噜噜噜在线播放 | 一二区精品 | 国产中文欧美日韩在线 | 色婷婷免费视频 | 久久久网页 | 西西444www大胆高清图片 | 午夜影院一区 | 9草在线 | 在线精品观看 | 99r国产精品 | 色小说在线 | 国产精品一区二区免费看 | 超碰在线94 | 一级一片免费看 | a成人v | 久久精品一区二区三区视频 | 91精品国自产在线观看欧美 | 日韩有码欧美 | 久久婷婷一区二区三区 | 久久久国产精品电影 | 久草视频免费在线播放 | 精品国产乱码久久久久久1区2匹 | 国产精彩视频一区 | 色插综合| 色欧美综合 | 久久亚洲精品电影 | 福利视频导航网址 | 久久精品视频日本 | 欧洲一区二区三区精品 | 免费观看一区二区三区视频 | 一区二区三区四区不卡 | 亚洲三级性片 | 久久精品视频观看 | 国产一区二区精品 | www.狠狠干 | 97超碰人人澡 | 天天鲁一鲁摸一摸爽一爽 | 五月婷婷久草 | 亚洲精品日韩一区二区电影 | 国产高清视频在线观看 | 日韩啪啪小视频 | 免费观看黄色12片一级视频 | 日韩精品一区在线观看 | 99精品国产兔费观看久久99 | 91精品国产高清自在线观看 | 91网页版在线观看 | 狠色狠色综合久久 | 精品久久久影院 | 久久精品91久久久久久再现 | 国产视频一区精品 | 97香蕉久久超级碰碰高清版 | www.夜色.com| 91麻豆精品国产91久久久久 | 色噜噜狠狠狠狠色综合久不 | 国产亚洲资源 | 午夜av免费在线观看 | 日韩一区视频在线 | 色婷五月天 | 最近中文字幕完整视频高清1 | 午夜三级在线 | 国产精品精品久久久久久 | 亚洲日本国产精品 | 日日日天天天 | 偷拍视频一区 | 亚洲精品五月 | 深夜视频久久 | 中文字幕欧美日韩va免费视频 | 天躁狠狠躁| www·22com天天操 | 久久在线免费 | 精品主播网红福利资源观看 | 在线视频欧美精品 | 久久久精品成人 | 激情欧美在线观看 | 看全黄大色黄大片 | 国产成人一级 | av黄色成人 | 亚洲一级电影 | av在线一| 日三级在线 | 亚洲国产网站 | 国产精品欧美激情在线观看 | 操老逼免费视频 | 日日爱999 | 国产精品婷婷午夜在线观看 | 欧美成人一二区 | 欧美性视频网站 | 91精品国产高清 | 日韩免费高清 | 国产精品久久久久久久久大全 | 黄污污网站| 亚洲a网| 婷婷日日| 欧美日韩一区二区在线 | 国产白浆视频 | 国内精品在线观看视频 | 日韩在线视频精品 | 人成在线免费视频 | 四虎最新域名 | 色婷婷狠狠五月综合天色拍 | 99亚洲精品视频 | 在线91观看 | 午夜精品久久久久久久爽 | 亚洲一区二区91 | 日韩特黄一级欧美毛片特黄 | 精品亚洲成a人在线观看 | 中文在线字幕免费观 | 麻豆视频一区二区 | 美女视频黄免费的 | 午夜av大片 | 亚洲精品18日本一区app | www.五月天婷婷.com | 91成品视频| 亚洲精品中文字幕在线 | 欧美永久视频 | 日韩一区二区三免费高清在线观看 | 色久av | 黄网站污 | 亚洲1级片 | 天天色天天上天天操 | 一区二区三区免费在线观看视频 | 狠狠干夜夜操天天爽 | 99国产视频 | 国产精品国产亚洲精品看不卡15 | 日韩专区中文字幕 | 久久99精品久久久久久清纯直播 | 日韩欧美高清在线 | 国产麻豆精品95视频 | 激情狠狠干 | 日韩三级精品 | av在线小说 | 91精品在线麻豆 | 超碰在线免费福利 | 六月丁香婷 | 91av视频| 伊甸园永久入口www 99热 精品在线 | 久草精品视频在线播放 | 国产精品精品国产色婷婷 | 国产精品日韩久久久久 | 激情伊人五月天久久综合 | 久久久美女 | 在线免费观看一区二区三区 | 欧美日本在线视频 | 射九九 | 五月婷婷视频在线观看 | 国语精品免费视频 | 欧洲性视频 | 天天射天天添 | 超黄视频网站 | 深夜成人av | 国产一二三区av | 国产99久久久精品视频 | 天天干天天射天天操 | 国产91精品一区二区麻豆亚洲 | 国产福利在线 | 天天干天天怕 | 亚洲国产小视频在线观看 | 精品国产欧美一区二区 | 黄av免费 | 精品久久久久久综合日本 | 久久久久99精品成人片三人毛片 | 91中文字幕在线播放 | 在线观看91av | 亚洲精品视频在线 | 人人狠狠 | 欧美午夜精品久久久久 | 韩国一区二区av | wwwwwww黄 | 成人久久久久久久久久 | 国产99久久久久久免费看 | 久久精品牌麻豆国产大山 | 中文字幕中文字幕在线一区 | 久久96国产精品久久99漫画 | 久久精品成人 | 日韩av区 | 久久久久精 | 欧美成亚洲 | 国产小视频国产精品 | 日韩高清在线一区二区 | 国产经典三级 | 91福利专区| 亚洲人成在线观看 | 国产青春久久久国产毛片 | 国产精品18久久久久vr手机版特色 | 天天干天天操天天做 | 亚洲成人999| 天天操天天射天天添 | 美女网站在线观看 | 日日躁夜夜躁aaaaxxxx | 超级碰碰碰碰 | 日韩欧美视频免费在线观看 | 亚洲黄色高清 | 国产乱对白刺激视频在线观看女王 | av大全在线免费观看 | 天天插天天狠天天透 | 91福利影院在线观看 | 国产经典 欧美精品 | 日本午夜免费福利视频 | 在线观看午夜av | 日韩电影中文字幕在线观看 | 亚洲日本va中文字幕 | 国产在线观看高清视频 | 福利视频网址 | av高清一区二区三区 | 成人在线视频免费观看 | 国内精品在线观看视频 | 99色免费 | 国产精品成人一区二区 | 日韩电影在线视频 | 很黄很黄的网站免费的 | 欧美嫩草影院 | 五月婷婷中文 | 国产经典三级 | 国产亚洲一区二区在线观看 | 久久久精品一区二区 | 美腿丝袜一区二区三区 | av高清在线| 新版资源中文在线观看 | 日韩在线视频免费播放 | 最近2019好看的中文字幕免费 | 亚洲成人av一区 | 日韩精品一区二区三区第95 | 亚洲影院一区 | 国产91九色视频 | 日韩高清二区 | 在线免费av网 | 国产精品久久久久久模特 | 国产人成在线观看 | 国产亚洲精品免费 | 亚洲电影久久久 | 狠狠色丁香婷婷综合最新地址 | 成人高清av在线 | 一级免费观看 | 久久精品久久久久久久 | 国产精品一区二区果冻传媒 | 久草在线综合 | 日韩电影一区二区三区 | 五月婷久 | 国内精品久久久久影院一蜜桃 | 国产亚洲精品久久久久久 | 久久久久久精 | 最近日韩免费视频 | 精品国产一区二区三区四区vr | 欧美日韩视频精品 | 久9在线| 亚a在线 | 精品超碰 | 婷婷在线免费 | 免费在线观看av网址 | www.狠狠色.com | 亚洲国产精品va在线看黑人 | 欧美久久久久久久久久久 | 日日爱视频 | 在线国产高清 | 黄色在线观看免费 | 在线观看亚洲精品 | 国产黄色高清 | 亚洲播放一区 | 蜜臀一区二区三区精品免费视频 | 一区二区三区免费在线观看视频 | 伊人久久影视 | 成年人黄色大片在线 | 一区二区激情 | 在线成人小视频 | 日日操天天爽 | 精品一二三四在线 | 91香蕉视频在线 | 91干干干| 免费网站观看www在线观看 | 五月婷婷一区二区三区 | 欧美精品一区二区三区一线天视频 | 亚洲国产精品小视频 | 久久久久黄色 | 国产精品久久久久久久免费大片 | 天天操天天操天天操 | 国产精品短视频 | 人人网av | av一级在线 | 精品一区 在线 | 欧美日韩亚洲一 | 色在线视频网 | 91九色蝌蚪国产 | 欧美性网站 | 又黄又刺激的视频 | 国产精品一区二区三区99 | 尤物97国产精品久久精品国产 | 中文字幕中文字幕在线一区 | 国产精品一区二区三区久久久 | 高清免费在线视频 | 亚洲影视资源 | 精品在线观看国产 | 免费看十八岁美女 | 欧美一二三区在线观看 | 狠狠躁日日躁狂躁夜夜躁 | 日韩中文字幕国产 | 九九九九热精品免费视频点播观看 | 在线播放视频一区 | 久久这里只有精品视频首页 | 青春草国产视频 | 激情伊人五月天 | 亚洲精品人人 | 久久久久久久久毛片精品 | 国产69久久 | 成人毛片100免费观看 | 国产精品日韩欧美一区二区 | 久久国产精品99久久久久久丝袜 | 久久字幕网 | 在线看国产一区 | 狠狠操影视 | 这里只有精品视频在线 | 国产精品久久久久久久久久久免费 | 成人av在线资源 | 久久综合久久综合久久综合 | 亚州国产视频 | 亚洲精品在线一区二区 | 国产精品观看视频 | 婷婷丁香综合 | 91亚洲欧美激情 | 日韩 国产| 亚洲成人高清在线 | 亚洲精品美女久久久久 | 国产一区视频在线观看免费 | 伊人狠狠色丁香婷婷综合 | 少妇bbw揉bbb欧美 | 天天操天天干天天操天天干 | 欧美欧美| 精品亚洲欧美一区 | 亚洲电影av在线 | 网址你懂的在线观看 | 九九热在线视频免费观看 | 免费网站黄色 | 日韩在线观看高清 | 久久观看免费视频 | 国内精品小视频 | 国产69久久久欧美一级 | 天天天干夜夜夜操 | 亚洲一级二级 | 色94色欧美 | 亚洲人成人在线 | 欧美成人91 | 国产对白av | 色婷婷免费视频 | 六月久久婷婷 | 全久久久久久久久久久电影 | 爱射综合 | 九色福利视频 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 亚洲国产福利视频 | 91亚洲在线 | 亚洲一区免费在线 | 一区二区视频播放 | 天堂va欧美va亚洲va老司机 | 最新日韩视频在线观看 | 天天做日日做天天爽视频免费 | 美女免费视频网站 | 五月激情丁香婷婷 | 亚洲91网站 | 亚洲激情视频 | 成人影片在线播放 | 久久美女免费视频 | 欧美日韩一区二区三区在线免费观看 | 99久久日韩精品免费热麻豆美女 | av看片在线观看 | 中文字幕一区二区三区久久 | 成人影片在线免费观看 | 91亚洲在线 | 欧美日韩另类在线 | 精品一区二区三区四区在线 | 中文字幕xxxx | 精品亚洲va在线va天堂资源站 | 在线观看av国产 | 亚洲精品在线观看免费 | h网站免费在线观看 | 欧美福利在线播放 | 在线看日韩av | 国产成人三级一区二区在线观看一 | 国产精品色婷婷视频 | 久久看片 | 成人作爱视频 | 精品国内自产拍在线观看视频 | 国产美女在线免费观看 | 亚洲电影久久 | 高清有码中文字幕 | 国产精品日韩久久久久 | 久久综合狠狠综合久久狠狠色综合 | 久久久久久久国产精品 | 免费三级a | 欧美aaaxxxx做受视频 | 国产精品一区二区在线播放 | 91麻豆传媒 | 91九色蝌蚪视频 | 亚洲综合涩 | 成年人免费看片 | 天天操,夜夜操 | 在线观看国产www | 99色在线观看 | 99热官网 | 久久成人亚洲欧美电影 | 视频三区 | 91.精品高清在线观看 | 国产精品麻豆一区二区三区 | 69绿帽绿奴3pvideos | 国产亚洲精品成人 | 黄网站大全 | 主播av在线 | 国产日韩精品久久 | 亚洲永久精品视频 | 国产日韩精品在线 | 国产色黄网站 | 国产精品系列在线播放 | 午夜视频二区 | 伊人婷婷久久 | 日韩黄色av网站 | 精品一区二区影视 | 久久中文网 | 亚洲高清视频一区二区三区 | 日韩一区二区三区在线看 | 成 人 黄 色 视频免费播放 | 免费看色网站 | 久久99网 | 久久婷婷国产色一区二区三区 | 色网站在线免费观看 | 91精品视频网站 | 五月开心婷婷网 | 色天天 | 中国老女人日b | 久久精品伊人 | 97在线公开视频 | 亚洲综合欧美激情 | 亚洲视频资源在线 | 毛片永久新网址首页 | 午夜久久电影网 | 91亚洲成人 | 伊人国产视频 | 成人app在线免费观看 | 国产91精品看黄网站 | 少妇性bbb搡bbb爽爽爽欧美 | 99热在线国产 | 九色在线| 在线中文字幕网站 | 久久国产精品区 | 亚洲人人网 | 日韩电影精品一区 | 五月色综合 | 制服丝袜一区二区 | 69视频国产 | 久久精品99国产国产 | 超碰在线9 | 91在线视频导航 | 四虎成人精品永久免费av | 中文字幕国产亚洲 | 欧美a影视 | 在线色视频小说 | 免费观看黄色12片一级视频 | 色综合久久久久 | 久久 亚洲视频 | 久久福利| 超碰在线亚洲 | 久久久亚洲影院 | jizz999| 亚洲国内精品 | 国产片免费在线观看视频 | 国产精品久久久久久一区二区三区 | 五月婷婷av| 亚洲黄色免费在线看 | 亚洲一一在线 | 久久国产精品偷 | 日韩视频中文字幕 | 欧美极品少妇xbxb性爽爽视频 | 福利av影院 | 欧美一区中文字幕 | 69久久99精品久久久久婷婷 | 日本不卡123| 国产r级在线观看 | 91av看片 | 国产三级午夜理伦三级 | 久久精品日产第一区二区三区乱码 | 亚洲美女精品区人人人人 | 中文字幕视频观看 | 美女黄频网站 | 久久国产精品99久久久久久丝袜 | 国产精品久久精品国产 | 国产精品亚 | 色美女在线 | 欧美激情视频一二三区 | 中文字幕在线专区 | 国产精品第二十页 | 色综合夜色一区 | 成人免费视频a | 国产精品福利无圣光在线一区 | 日本h视频在线观看 | 一本一本久久a久久精品牛牛影视 | 日韩在线视频不卡 | 国产精品视频 | 在线观看亚洲 | 国产黄色在线看 | 亚洲国产高清视频 | 国产区欧美 | 国产视频精品免费播放 | 日韩精品一区二区三区不卡 | 日本不卡一区二区三区在线观看 | 日本黄色黄网站 | 欧美一级视频免费 | 亚洲视频在线观看 | 国产午夜小视频 | 国产网红在线 | 丁香婷婷综合激情五月色 | 精品一区二区在线免费观看 | 999电影免费在线观看 | 国产永久免费 | 福利一区二区 | 日韩中文字幕视频在线观看 | 久久久久国产精品视频 | 最近中文字幕mv | 久久精品国产亚洲 | 91完整版 | 91网站在线视频 | 日本电影黄色 | 2021国产精品 | 天天操网址 | 日本中文字幕网址 | 国产一区久久久 | 国产精品久久久久久久久久尿 | 91精品视频网站 | 久草精品视频 | 五月婷丁香网 | 91视频麻豆 | av在线影片 | www天天操| av线上看| 中文字幕一区二区三区四区在线视频 | 日本久久精 | 五月天免费网站 | 国产小视频在线看 | 精品毛片在线 | 精品久久久国产 | 国产剧情久久 | 91最新地址永久入口 | 91看片在线免费观看 | 成人全视频免费观看在线看 | 久久国产精品99久久久久 | 国产免费又黄又爽 | 日本动漫做毛片一区二区 | 日本在线观看一区 | 97成人精品 | av在线之家电影网站 | 18国产精品白浆在线观看免费 | 91在线欧美| 国产在线精品国自产拍影院 | 国产精彩在线视频 | 婷婷丁香六月 | 国产黄色精品在线观看 | 一区二区三区免费在线观看 | 久久久久国产一区二区三区 | 亚洲综合成人婷婷小说 | 国产精品一区二 | 日韩在线一区二区免费 | 久久天堂影院 | 97天堂| 综合色亚洲 | 美女天天操 | 亚洲一片黄 | 又黄又爽又刺激的视频 | 日韩欧美一区二区在线观看 | 天天干,天天草 | 国产日产在线观看 | 免费人成网ww44kk44 | av成人免费在线看 | 开心激情网五月天 | 国产精品一区二区久久 | 久久视奸| 在线三级av | 亚洲精品午夜久久久久久久 | 热久久99这里有精品 | 日本久久免费电影 | 久久久首页 | av中文天堂 | 欧美日韩国产一二 | 欧美日韩中文在线视频 | 免费看国产精品 | 色婷婷 亚洲 | 国产成人一区二区三区在线观看 | 久久视频一区二区 | 久久久久福利视频 | avove黑丝| 免费看精品久久片 | 亚洲国产资源 | 最新国产福利 | 国产一卡二卡四卡国 | 精品久久久久一区二区国产 | 国产一级做a爱片久久毛片a | 麻豆国产网站 | 日本精品一区二区在线观看 | 日日操网站 | 日本久久久影视 | av一级网站 | 国产麻豆精品95视频 | 久久免费视频在线观看30 | 国产一级免费在线观看 | 国产不卡免费av | 国产在线精品区 | 久久综合激情 | 久久欧美在线电影 | 免费亚洲视频在线观看 | 爱爱一区 | 探花视频在线观看 | 成人av久久| 日韩激情小视频 | 天天曰视频| 最近免费在线观看 | 成年人免费看 | 日韩日韩日韩日韩 | 激情视频在线观看网址 | 中文字幕在线播放日韩 | 可以免费看av | 国产精品久久99综合免费观看尤物 | 中文字幕视频在线播放 | 中文字幕999 | 天天做日日做天天爽视频免费 | 69中文字幕 | 五月在线视频 | 五月天天色 | 国产视频在线观看一区 | 久久久久国产一区二区三区 | 国产高清视频网 | 国产精品久久av | 99亚洲国产 | 久久精品永久免费 | 99这里精品| 久操中文字幕在线观看 | 中文字幕成人在线观看 | 国产精品久久久久久久久久白浆 | 丁香色婷 | 亚洲a色| 日韩av在线一区二区 | 免费看的黄网站 | 国产日韩精品在线观看 | 欧美一级电影 | 天天操天天干天天爱 | 久久精品91久久久久久再现 | 亚洲天天综合网 | 99国产成+人+综合+亚洲 欧美 | 欧美va天堂va视频va在线 | 国产伦理久久精品久久久久_ | 日韩在线观看中文字幕 | 91亚洲精品乱码久久久久久蜜桃 | 色国产在线 | 精品高清美女精品国产区 | 免费观看一级 | 免费高清在线一区 | 国产黄视频在线观看 | 国产专区免费 | 日韩精品一区二区免费 | 欧美少妇bbwhd | 啪啪激情网 | 在线观看日韩中文字幕 | 综合网伊人 | 成人av影院在线观看 | www在线观看国产 | 久久久久久福利 | 亚洲国产999 | 国产免费一区二区三区最新 | 亚洲精品国精品久久99热 | 成人在线观看资源 | 欧美韩日精品 | 五月天高清欧美mv | 黄色av一区二区三区 | 国产99久久久国产精品免费二区 | 久久久国产一区 | 久草视频网 | 在线观看aaa | 国产美女免费看 | 精品国产一区二区三区四 | 超碰人人国产 | 粉嫩av一区二区三区四区 | 国产精品1区2区3区 久久免费视频7 | 又黄又爽的视频在线观看网站 | 操夜夜操 | 久久国产精品免费 | 99re在线视频观看 | 国产在线精品一区二区三区 | 久久不色| 精品久久久久一区二区国产 | 婷色在线| av高清免费在线 | 亚洲精品视频第一页 | 国产日女人 | 欧美三级高清 | 国产香蕉视频在线观看 | 日韩专区av | 欧美一进一出抽搐大尺度视频 | 一区二区丝袜 | 久久久久久久久久久网 | 免费视频久久久久 | 在线免费看片 | 五月婷婷综合激情网 | 国产在线观看你懂的 | 久久免费精品 | 日韩av不卡在线观看 | 午夜91视频| 国产视频一级 | 91九色国产蝌蚪 | 99精品免费久久久久久久久日本 | 国产永久免费高清在线观看视频 | 国产精品久久久久aaaa | www.久久com| 欧美aa一级片 | 日本狠狠干| 天堂av官网| 精品国产免费人成在线观看 | 亚洲国产日韩欧美在线 | 91在线免费视频 | 一区二区三区在线观看免费视频 | 九九免费观看全部免费视频 | 久久综合九色综合久99 | 欧美电影黄色 | 免费av观看网站 | 亚洲永久免费av | 美女视频永久黄网站免费观看国产 | av在线短片| 超碰九九 | 欧美日韩二区在线 | 国产精品免费看久久久8精臀av | 91精品久久久久久久久久久久久 | 久久久久久久影院 | 久久久久久蜜桃一区二区 | 成人免费视频在线观看 | 亚洲国产成人久久综合 | 99热在线国产精品 | 欧美成人在线免费观看 | 亚洲精品视频在线 | 亚洲成年人av | 91国内在线视频 | 天天操天天色天天射 | 国产精品99久久久久久有的能看 | 热re99久久精品国产66热 | 五月婷婷激情综合 | 久久综合导航 | 天天做天天射 | 天天综合网~永久入口 | 日本精油按摩3 | 婷婷色视频 | 中文久草 | 久久久影院一区二区三区 | 亚洲区精品 | 国产成人精品久久二区二区 | 久久婷婷色综合 | 色吧久久 | 美腿丝袜av | 日韩美在线 | 免费视频久久久 | 视频二区 | 27xxoo无遮挡动态视频 | 91中文在线 | 天堂资源在线观看视频 | 国产精品专区在线观看 | 国产亚洲视频系列 | 国产不卡视频在线 | 久久综合九色综合网站 | 91av影视| 日韩免费在线视频观看 | 久久亚洲免费视频 | 国产免费中文字幕 | 天天综合网在线 | 91麻豆网| 91精品一区二区在线观看 |