Android 设计模式:(四)原型模式
前言
本文是对《Adroid 源码设计模式解析与实战》 何红辉、关爱民 著 人民邮电出版社所做的读书笔记。文章是对本书的一些列学习笔记,如若有侵犯到作者权益,还望作者能联系我,我会及时下架。
这本书不错,有兴趣的同学可以买原书看看。
感兴趣的朋友欢迎加入学习小组QQ群: 193765960。
版权归作者所有,如有转发,请注明文章出处:https://xiaodanchen.github.io/archives/
相关文章:
Android 设计模式:(一)面向对象的六大原则
Android 设计模式:(二)单例模式
Android 设计模式:(三)Builder模式
Android 设计模式:(四)原型模式
Android 设计模式:(五)工厂方法模式
Android 设计模式:(六)抽象工厂模式
Android 设计模式:(七)策略模式
1. 原型模式的定义
原型模式:对一个对象,通过克隆生成其副本,而不是通过new 的方式重新生成。
使用场景:
- 类初始化需要消耗非常多的资源,包括数据、硬件资源等。通过克隆的方式,可以避免这些消耗。
- 通过new 产生一个对象需要非常繁琐的数据准备或者访问权限,这时可以使用原型模式。
- 一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其数据时,可以用原型模式拷贝多个对象供调用者使用,即保护性拷贝。
- 一个对象,如果要求在某些对象中不允许对其修改,则可以使用原型模式,对其进行保护性拷贝,这样,无论对备份怎么修改都不会影响原型数据。
2. 原型模式的实现
在开发中,我们有时会满足一些需求,就是有的对象中的数据只允许客户端读取,而不允许修改。例如,用户登录信息,只允许在用户登录模块修改,在其他模块比如登录校验、个人信息显示等模块,用户数信息不允许修改。让我们看看该如何实现:
2.1 屌丝程序员小明
源码
/**
* 用户实体类
*/
public class User{
public int age;
public String name;
public String phone;
public Address address;
}
/**
* 用户地址类
*/
public class Address{
public String city;//城市
public String district;//区
public String street;//街道
public Address(String city,String district,String street){
this.city = city;
this.district = district;
this.street = street;
}
}
/**
* 登录接口
*/
public interface Login{
void login();
}
/**
* 登录实现
*/
public class LoginImpl implements Login{
@Override
public void login(){
User user = new User();
//登录服务器获取用户信息,将信息赋值给user
user.age = 22;
user.name = "xiaoming"
user.address = new Address("北京市","海淀区","花园东路");
...
//用户信息获取到后,将用户信息设置到Session中(单例模式)
LoginSession.getLoginSession().setLoginedUser(user);
}
}
/**
* 登录Session:单例模式
*/
public class LoginSession{
private static LoginSession instance = null;
//已登录用户
private User sUser;
private LoginSession(){}
//懒汉模式
public static LoginSession getLoginSession(){
if(null == instance){
instance = new LoginSession();
}
renturn instance;
}
//设置已登录的用户信息:包级私有函数
//public:所有类可见。
//pirvate:只有同一类内部的方法可见,在有就是内部类也可以访问到。
//默认(friendly):包内可见。
//protected:继承可见。
void setLoginedUser(User user){
sUser = user;
}
public User getLoginedUser(){
return sUser;
}
@Override
public void login(){
User user = new User();
//登录服务器获取用户信息,将信息赋值给user
user.age = 22;
user.name = "xiaoming"
...
//用户信息获取到后,将用户信息设置到Session中(单例模式)
LoginSession.getLoginSession().setLoginedUser(user);
}
}
解析
用户登录时从服务器获取用户信息,通过setLoginedUser方法设置给LoginSession。由于setLoginedUser的访问权限是包级别的,因此外部模块无法访问,在一定程度上满足了不允许其他模块修改的要求,小明很高兴。
可是,小明有一个比他还菜的同事大力协同开发,大力果然出奇迹啊:
//在用户信息功能模块,
//获取登录用户信息
User user = LoginSession.getLoginSession().getLoginedUser();
//测试:更新用户地址信息
user.address = new Address("北京市","朝阳区","大望路");
联调时发现,用户显示的信息和服务器获取的信息不一致,小民很郁闷,他本来打的好算盘是只允许LoginSession包下才能通过setLoginedUser设定修改用户信息,然而并没有其他人调用setLoginedUser方法,追查了好久终于发现大力这个猪队友干的好事。如何才能保证服务器获取到的数据在其他模块下不会被更改呢?但是屌丝到爆的小明对这个问题束手无策了。
无奈之下,小明找到了小民,小民嘴角一勾,“小CASE!使用clone获取副本”
源码2
/**
* 用户实体类
*/
public class User implements Cloneable{
...
@Override
public User clone(){
User user = null;
try{
user = (User)super.clone();
}catch(CloneNotSupportedException e){
e.printStackTrace();
}
return user;
}
}
//修改LoginSession的getLoginedUser方法
public User getLoginedUser(){
return sUser.clone();
}
解析2
小明采用了原型模式,让User实现了克隆功能,在不允许修个user的地方,使用其副本,即保护性拷贝。
经测试发现,浅拷贝并没有彻底的解决问题。比如:
//在用户信息功能模块,
//获取登录用户信息
User user = LoginSession.getLoginSession().getLoginedUser();
//测试:更新用户地址信息
user.address.city = ("北京市");
user.address.district = ("朝阳区");
user.address.street = ("大望路");
小明这次彻底懵逼了,不得已又找到了小民,小民看过小明的代码,笑了笑:“方向没错,只不过你使用了浅拷贝,要想解决你的问题,你应该使用深拷贝,”“
3. 浅拷贝和深拷贝
深拷贝(深复制)和浅拷贝(浅复制)是两个比较通用的概念,尤其在C++语言中,若不弄懂,则会在delete的时候出问题,但是我们在这幸好用的是Java。虽然java自动管理对象的回收,但对于深拷贝(深复制)和浅拷贝(浅复制),我们还是要给予足够的重视,因为有时这两个概念往往会给我们带来不小的困惑。
浅拷贝是指拷贝对象时仅仅拷贝对象本身(包括对象中的基本变量),而不拷贝对象包含的引用指向的对象。深拷贝不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象。举例来说更加清楚:对象A1中包含对B1的引用,B1中包含对C1的引用。浅拷贝A1得到A2,A2 中依然包含对B1的引用,B1中依然包含对C1的引用。深拷贝则是对浅拷贝的递归,深拷贝A1得到A2,A2中包含对B2(B1的copy)的引用,B2 中包含对C2(C1的copy)的引用。
3.1 装逼程序员小民
源码
/**
* 用户实体类
*/
public class User implements Cloneable{
...
@Override
public User clone(){
User user = null;
try{
user = (User)super.clone();
}catch(CloneNotSupportedException e){
e.printStackTrace();
}
user.address = (Address)address.clone();
return user;
}
}
/**
* 用户实体类
*/
public class Address implements Cloneable{
...
@Override
public Address clone(){
Address address = null;
try{
address = (Address)super.clone();
}catch(CloneNotSupportedException e){
e.printStackTrace();
}
return address;
}
}
//修改LoginSession的getLoginedUser方法
public User getLoginedUser(){
return sUser.clone();
}
解析
深拷贝不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象。