Android中所涉及的常用设计模式

设计模式是解决特定问题的一系列套路。设计模式的使用不是语法规定,不要为了套用设计模式而用设计模式,它的使用为了解决问题而采用的一套用来提高代码可复用性、可维护性、可读性、稳健性以及安全性的解决方案。

1、单例模式

概念:Ensure a class has only one instance, and provide a global point of access to it.
动态确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。
优点:

  • 由于单例模式在内存中只有一个实例,减少了内存开销。对于那些耗内存的类,只实例化一次,大大提高性能,尤其是移动开发中。
  • 单例模式可以避免对资源的多重占用,例如一个写文件时,由于只有一个实例存在内存中,避免对同一个资源文件的同时写操作。
  • 单例模式可以在系统设置全局的访问点,优化和共享资源访问。
public class Singleton {    
    private volatile static  Singleton instance = null;    
    
    private Singleton(){    
    }    
     
    public static Singleton getInstance() {    
        if (instance == null) {    
            synchronized (Singleton.class) {    
                if (instance == null) {    
                    instance = new Singleton();    
                }    
            }    
        }    
        return instance;    
    }    
}    

构造函数私有化,定义静态函数获得实例就不多说了,这里着重说一下volatile:
volatile本质是在告诉jvm当前变量在寄存器中的值是不确定的,需要从内存中读取,
synchronized则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住.
(首先我们要先意识到有这样的现象,编译器为了加快程序运行的速度,对一些变量的写操作会先在寄存器或者是CPU缓存上进行,最后才写入内存.而在这个过程,变量的新值对其他线程是不可见的.而volatile的作用就是使它修饰的变量的读写操作都必须在内存中进行!)
synchronized
同步块大家都比较熟悉,通过 synchronized关键字来实现,所有加上synchronized和 块语句,在多线程访问的时候,同一时刻只能有一个线程能够用synchronized修饰的方法 或者 代码块。不但保证了可见性还保证了原子性。
volatile
用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值。volatile很容易被误用,用来进行原子性操作。其实volatile只能保证变量的可见性,不能保证原子性。
再就是这个双重判断null :
这是因为如果线程A进入了该代码,线程B 在等待,这是A线程创建完一个实例出来后,线程B 获得锁进入同步代码,实例已经存在,木有必要再创建一个,所以双重判断有必要。

注意:volatile对singleton的创建过程的重要性

  • 禁止指令重排序(有序性)
    实例化一个对象其实可以分为三个步骤:
      (1)分配内存空间。
      (2)初始化对象。
      (3)将内存空间的地址赋值给对应的引用。
    但是由于操作系统可以对指令进行重排序,所以上面的过程也可能会变成如下过程:
      (1)分配内存空间。
      (2)将内存空间的地址赋值给对应的引用。
      (3)初始化对象
    如果是这个流程,多线程环境下就可能将一个未初始化的对象引用暴露出来,从而导致不可预料的结果(如题目的描述,这里就是因为 instance = new Singleton(); 不是原子操作,编译器存在指令重排,从而存在线程1 创建实例后(初始化未完成),线程2 判断对象不为空后对其操作,但实际对象仍为空,造成错误)。因此,为了防止这个过程的重排序,我们需要将变量设置为volatile类型的变量,volatile的禁止重排序保证了操作的有序性。
  • Singleton对象的内存可见性
    这里由于synchronized锁的是Singleton.class对象,而不是Singleton对象,所以synchronized只能保证Singleton.class对象的内存可见性,但并不能保证Singleton对象的内存可见性;这里用volatile声明Singleton,可以保证Singleton对象的内存可见性。这一点作用也是非常重要的(如题目的描述,避免因为线程1 创建实例后还只存在自己线程的工作内存,未更新到主存。线程 2 判断对象为空,创建实例,从而存在多实例错误)。

Android中 用到的地方很多,比如Android-Universal-Image-Loader中的单例,EventBus中的单例最后给出一个管理我们activity的类,可以作为一个简单工具类:

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());    
    }    
}    

单例模式在Android源码中的应用:
在Android源码中,使用到单例模式的例子很多,如:
InputMethodManager类:

public final class InputMethodManager {  
    static final boolean DEBUG = false;  
    static final String TAG = "InputMethodManager";  
  
    static final Object mInstanceSync = new Object();  
    static InputMethodManager mInstance;  
      
    final IInputMethodManager mService;  
    final Looper mMainLooper;  

创建唯一的实例static InputMethodManager mInstance;

/**  
     * Retrieve the global InputMethodManager instance, creating it if it  
     * doesn't already exist.  
     * @hide  
     */  
    static public InputMethodManager getInstance(Context context) {  
        return getInstance(context.getMainLooper());  
    }  
      
    /**  
     * Internally, the input method manager can't be context-dependent, so  
     * we have this here for the places that need it.  
     * @hide  
     */  
    static public InputMethodManager getInstance(Looper mainLooper) {  
        synchronized (mInstanceSync) {  
            if (mInstance != null) {  
                return mInstance;  
            }  
            IBinder b = ServiceManager.getService(Context.INPUT_METHOD_SERVICE);  
            IInputMethodManager service = IInputMethodManager.Stub.asInterface(b);  
            mInstance = new InputMethodManager(service, mainLooper);  
        }  
        return mInstance;  
    }  

防止多线程同时创建实例:

synchronized (mInstanceSync) {           
 if (mInstance != null) { 
           return mInstance;          
  }

当没有创建实例对象时,调用mInstance = new InputMethodManager(service, mainLooper); 其中类构造函数如下所示:

InputMethodManager(IInputMethodManager service, Looper looper) {  
        mService = service;  
        mMainLooper = looper;  
        mH = new H(looper);  
        mIInputContext = new ControlledInputConnectionWrapper(looper,  
                mDummyInputConnection);  
          
        if (mInstance == null) {  
            mInstance = this;  
        }  
    }  

2、建造者模式(Builder 模式)

定义:将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示
概念就是比较抽象的,让大家很难理解的,如果简单从这个一个概念就搞懂了这个模式的话,那就不用费力的去查资料整理后边的东西了。
这里我们通过一个例子来引出Build模式。假设有一个Person类,他的一些属性可以为null,可以通过这个类来构架一大批人

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;    
    }    
}    

然后为了方便,你可能会写这么一个构造函数来传属性

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

或者为了更方便还会写一个空的构造函数

public Person() {    
}  

有时候还会比较懒,只传入某些参数,又会来写这些构造函数

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;    
}    

于是就可以来创建各种需要的类

Person p1=new Person();    
Person p2=new Person("张三");    
Person p3=new Person("李四",18);    
Person p4=new Person("王二",21,180);    
Person p5=new Person("麻子",16,170,65.4);   

其实这种写法的坏处在你写的过程中想摔键盘的时候就该想到了,既然就是一个创建对象的过程,怎么这么繁琐,并且构造函数参数过多,其他人创建对象的时候怎么知道各个参数代表什么意思呢,这个时候我们为了代码的可读性,就可以用一下Builder模式了
给Person类添加一个静态Builder类,然后修改Person的构造函数,如下:

public class Person {    
    private String name;    
    private int age;    
    private double height;    
    private double weight;    
    
    private Person(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,最后提供了一个build函数来创建person对象,对应的在Person的构造函数中,传入了Builder对象,然后依次对自己的成员变量进行赋值。此外,Builder的成员函数返回的都是this的另一个作用就是让他支持链式调用,使代码可读性大大增强
于是我们就可以这样创建Person对象:

Person.Builder builder=new Person.Builder();    
Person person=builder    
        .name("张三")    
        .age(18)    
        .height(178.5)    
        .weight(67.4)    
        .build();  

Android中大量地方运用到了Builder模式,比如常见的对话框创建:

AlertDialog.Builder builder=new AlertDialog.Builder(this);    
AlertDialog dialog=builder.setTitle("对话框")    
        .setIcon(android.R.drawable.ic_dialog)    
        .setView(R.layout.custom_view)    
        .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中StringBuilder 和StringBuffer都用到了Builder模式,只不过是稍微简单一点了Gson中的GsonBuilder:

GsonBuilder builder=new GsonBuilder();    
Gson gson=builder.setPrettyPrinting()    
        .disableHtmlEscaping()    
        .generateNonExecutableJson()    
        .serializeNulls()    
        .create(); 

网络框架OKHttp:

Request.Builder builder=new Request.Builder();    
Request request=builder.addHeader("","")    
    .url("")    
    .post(body)    
    .build(); 

可见大量框架运用了Builder 设计模式,总结一下吧:
定义一个静态内部类Builder,内部成员变量跟外部一样
Builder通过一系列方法给成员变量赋值,并返回当前对象(this)
Builder类内部提供一个build方法方法或者create方法用于创建对应的外部类,该方法内部调用了外部类的一个私有化构造方法,该构造方法的参数就是内部类Builder
外部类提供一个私有化的构造方法供内部类调用,在该构造函数中完成成员变量的赋值

3、 观察者模式

定义:Define a one-to-many dependency between objects so that when one object changes state, all its dependents are notified and updated automatically.
定义对象间一种一对多的依赖关系,使得当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新。
主要包括四个部分:

  • Subject被观察者。是一个接口或者是抽象类,定义被观察者必须实现的职责,它必须能动态地增加、取消观察者,管理观察者并通知观察者。
  • Observer观察者。观察者接收到消息后,即进行update更新操作,对接收到的信息进行处理。
  • ConcreteSubject具体的被观察者。定义被观察者自己的业务逻辑,同时定义对哪些事件进行通知。
  • ConcreteObserver具体观察者。每个观察者在接收到信息后处理的方式不同,各个观察者有自己的处理逻辑。

这个好像还好理解那么一点点,不过还是先来讲个情景,
天气预报的短信服务,一旦付费订阅,每次天气更新都会向你及时发送
其实就是我们无需每时每刻关注我们感兴趣的东西,我们只需要订阅它即可,一旦我们订阅的事务有变化了,被订阅的事务就会即时的通知我们
我们来看一下观察者模式的组成:

  • 观察者,我们称它为Observer,有时候我们也称它为订阅者,即Subscriber;
  • 被观察者,我们称它为Observable,即可以被观察的东西,有时候还会称之为主题,即Subject.

至于观察者模式的具体实现,java里为我们提供了Observable类和Observer接口供我们快速实现该模式,但是这里为了加深印象,不用这个两个类
我们来模拟上边的场景,先定义一个Weather的类:

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 + '\'' +    
                '}';    
    }    
}    

然后定义我们的被观察者,我们希望它能够通用,所以定义成泛型,内部应该暴露出registerunRegister供观察者订阅和取消订阅,至于观察者的保存,我们用ArrayList即可,另外,当主题发生变化的时候,需要通知观察者来做出响应,还需要一个notifyObservers方法,具体实现如下:

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,该接口也是泛型的

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

一旦订阅的主题发生了变化,就会调用该接口
用一下,我们定义一个天气变化的主题,也就是被观察者,再定义两个观察者来观察天气的变化,一旦变化了就打印出天气的情况,注意,一定要用register方法来注册,否则观察者收不到变化的信息,而一旦不感兴趣,就可以调用unregister方法:

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的点击事件:

Button btn=new Button(this);    
btn.setOnClickListener(new View.OnClickListener() {    
    @Override    
    public void onClick(View v) {    
        Log.e("TAG","click");    
    }    
});    

另外广播机制,本质也是观察者模式

调用registerReceiver方法注册广播,调用unregisterReceiver方法取消注册,之后使用sendBroadcast发送广播,之后注册的广播会受到对应的广播信息,这就是典型的观察者模式
开源框架EventBus也是基于观察者模式的,
观察者模式的注册,取消,发送事件三个典型方法都有:

EventBus.getDefault().register(Object subscriber);    
EventBus.getDefault().unregister(Object subscriber);    
EventBus.getDefault().post(Object event);    

4 、策略模式

定义:策略模式定义了一系列算法,并将每一个算法封装起来,而且使他们可以相互替换,策略模式让算法独立于使用的客户而独立改变

最常见的就是关于出行旅游的策略模式,出行方式有很多种,自行车,汽车,飞机,火车等,如果不使用任何模式,代码是这样子的:

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语句,这违反了面向对象的原则之一,对修改封装(开放封闭原则)
题外话:面向对象的三大特征:封装,继承和多态
五大基本原则:单一职责原则(接口隔离原则),开放封闭原则,Liskov替换原则,依赖倒置原则,良性依赖原则
好,回归主题,如何用策略模式来解决这个问题
首先,定义一个策略的接口:

public interface Strategy {    
    void travel();    
}   

然后根据不同的出行方法来实现该接口:

public class WalkStrategy implements Strategy{    
    
    @Override    
    public void travel() {    
        System.out.println("walk");    
    }    
    
}   
public class PlaneStrategy implements Strategy{    
    
    @Override    
    public void travel() {    
        System.out.println("plane");    
    }    
    
}    
public class SubwayStrategy implements Strategy{    
    
    @Override    
    public void travel() {    
        System.out.println("subway");    
    }    
    
}    

此外还需要一个包装策略的类,来调用策略中的接口:

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();    
        }    
    }    
}    

测试一下代码:

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();    
    }    
}  

以后如果再增加什么别的出行方式,就再继承策略接口即可,完全不需要修改现有的类。
策略模式优缺点:

  • 定义一系列算法:策略模式的功能就是定义一系列算法,实现让这些算法可以相互替换。所以会为这一系列算法定义公共的接口,以约束一系列算法要实现的功能。如果这一系列算法具有公共功能,可以把策略接口实现成为抽象类,把这些公共功能实现到父类里面,对于这个问题,前面讲了三种处理方法,这里就不罗嗦了。
  • 避免多重条件语句:根据前面的示例会发现,策略模式的一系列策略算法是平等的,可以互换的,写在一起就是通过if-else结构来组织,如果此时具体的算法实现里面又有条件语句,就构成了多重条件语句,使用策略模式能避免这样的多重条件语句。
  • 更好的扩展性:在策略模式中扩展新的策略实现非常容易,只要增加新的策略实现类,然后在选择使用策略的地方选择使用这个新的策略实现就好了。
  • 客户必须了解每种策略的不同:策略模式也有缺点,比如让客户端来选择具体使用哪一个策略,这就可能会让客户需要了解所有的策略,还要了解各种策略的功能和不同,这样才能做出正确的选择,而且这样也暴露了策略的具体实现。
  • 增加了对象数目:由于策略模式把每个具体的策略实现都单独封装成为类,如果备选的策略很多的话,那么对象的数目就会很可观。
  • 只适合扁平的算法结构:策略模式的一系列算法地位是平等的,是可以相互替换的,事实上构成了一个扁平的算法结构,也就是在一个策略接口下,有多个平等的策略算法,就相当于兄弟算法。而且在运行时刻只有一个算法被使用,这就限制了算法使用的层级,使用的时候不能嵌套使用。

Android中的应用
下面说说在Android里面的应用。在Android里面策略模式的其中一个典型应用就是Adapter,在我们平时使用的时候,一般情况下我们可能继承BaseAdapter,然后实现不同的View返回,GetView里面实现不同的算法。外部使用的时候也可以根据不同的数据源,切换不同的Adapter。

5、原型模式

定义:用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象。

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 +    
                '}';    
    }    
}    

要实现原型模式,按照以下步骤来:
1,实现一个Cloneable接口:

public class Person implements Cloneable{    
    
}    

重写Object的clone方法,在此方法中实现拷贝逻辑:

@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;    
} 

测试一下:

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集合,就酱紫:

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

在进行拷贝的时候就要注意了,如果还是跟之前的一样操作,就会发现其实两个不同的人的兴趣集合的是指向同一个引用,我们对其中一个人的这个集合属性进行操作 ,另一个人的这个属性也会相应的变化,其实导致这个问题的本质原因是我们只进行了浅拷贝,也就是指拷贝了引用,最终两个对象指向的引用是同一个,一个发生变化,另一个也会发生变化。显然解决方法就是使用深拷贝:

@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;    
}  

不再是直接引用,而是拷贝了一份,
其实有的时候我们看到的原型模式更多的是另一种写法:在clone函数里调用构造函数,构造函数里传入的参数是该类对象,然后在函数中完成逻辑拷贝:

@Override    
public Object clone(){    
    return new Person(this);    
}    
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类,

public Object clone() {    
    return new Bundle(this);    
}     
public Bundle(Bundle b) {    
    super(b);    
    
    mHasFds = b.mHasFds;    
    mFdsKnown = b.mFdsKnown;    
}  

然后是Intent类:

@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,再修改不同的地方,便可以直接使用

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接口:

/** 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;    
}    

2021期待与你一起共事,点击查看岗位

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容

  • 面向对象的六大原则 单一职责原则 所谓职责是指类变化的原因。如果一个类有多于一个的动机被改变,那么这个类就具有多于...
    JxMY阅读 935评论 1 3
  • 设计模式汇总 一、基础知识 1. 设计模式概述 定义:设计模式(Design Pattern)是一套被反复使用、多...
    MinoyJet阅读 3,929评论 1 15
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,914评论 25 707
  • 勿忧,杜康酒满旁; 勿忧,萱馥郁芬芳; 勿忧,解聆人在旁; 勿忧,西岭人安康。
    萌萌宝贝公主阅读 547评论 0 0