重修设计模式-创建型-原型模式

发布于:2024-08-14 ⋅ 阅读:(45) ⋅ 点赞:(0)

重修设计模式-创建型-原型模式

原型模式就是利用已有对象(原型)通过拷贝方式来创建对象的模式,达到节省对象创建时间的目的。适用于对象创建成本较大,且同一类的不同对象之间差别不大的场景。

比如一个对象中的数据需要经过复杂计算才能得到(如排序),或者对象是从网络、文件系统等通过IO读取的,这种情况下就可以用原型模式快速拷贝出一个新对象来使用,而不是再经过复杂计算或读取 IO 来创建对象。

原型模式的核心就是对象的拷贝,且有浅拷贝和深拷贝的区别。

  • 浅拷贝:只会复制基本类型的数据和引用对象的内存地址,不会递归的拷贝引用对象本身,比如 Java 中 Object 的 clonse() 方法。
  • 深拷贝:不仅复制基本类型的数据,也会拷贝引用类型的对象(会开辟新的内存空间,并将新开辟的内存地址引用给新对象),从而得到一份完全独立的对象。

Kotlin 中 data class 的 copy() 方法,或 Java 中 Object 的 clone() 方法都是浅拷贝的实现,下面验证一下:

data class User(var name: String, var age: Int, val address: Address): Cloneable {
    public override fun clone(): Any {
        return super.clone()
    }
}

data class Address(var street: String, var city: String) : Cloneable {
    override fun clone(): Any {
        return super.clone()
    }
}

测试 copy() 和 clone() 方法:

fun main(args: Array<String>) {
    val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))
    val userCopy = user.copy()
    val userCopy1 = user.clone() as User

    println("user1:${user}")
    println("user2:${userCopy}")
    println("user address:${user.address == userCopy.address}")     //判断值,相当于equal
    println("user address:${user.address === userCopy.address}")    //判断内存地址

    println("---")

    println("user1:${user}")
    println("user2:${userCopy1}")
    println("user address:${user.address == userCopy1.address}")     //判断值,相当于equal
    println("user address:${user.address === userCopy1.address}")    //判断内存地址
}

代码执行结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:true
---
user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:true

可见引用类型对象指向的内存地址还是同一个,并没有真正地进行拷贝,这一点通过源码也可以得到验证。将 User 编译成字节码后,再反编译成 Java 语言:

public final class User implements Cloneable {
   @NotNull
   private String name;
   private int age;
   @NotNull
   private final Address address;

   //调用的是Object的clone()方法
   @NotNull
   public Object clone() {
      return super.clone();
   }

   //componentX()方法,用于解构赋值语法

   @NotNull
   public final User copy(@NotNull String name, int age, @NotNull Address address) {
      Intrinsics.checkNotNullParameter(name, "name");
      Intrinsics.checkNotNullParameter(address, "address");
      return new User(name, age, address);
   }
}
public final class Address implements Cloneable {
   @NotNull
   private String street;
   @NotNull
   private String city;

   @NotNull
   public Object clone() {
      return super.clone();
   }

   @NotNull
   public final Address copy(@NotNull String street, @NotNull String city) {
      Intrinsics.checkNotNullParameter(street, "street");
      Intrinsics.checkNotNullParameter(city, "city");
      return new Address(street, city);
   }
}

可以看到,copy() 方法并不会对引用类型对象进行拷贝工作,而是直接传入。而 clone() 方法则是由底层实现,执行逻辑相同。

@HotSpotIntrinsicCandidate
protected native Object clone() throws CloneNotSupportedException;

下面介绍深拷贝的几种实现方式:
1.深拷贝实现—自己实现Cloneable的clone方法:

data class User(var name: String, var age: Int, val address: Address): Cloneable {
    public override fun clone(): Any {
        return User(name, age, address.clone() as Address)
    }
}

data class Address(var street: String, var city: String) : Cloneable {
    public override fun clone(): Any {
        return Address(street, city)
    }
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:false

引用地址不相同了,说明是两个独立的对象,这种方式比较麻烦,增减字段都需要对 clone() 方法进行维护,如果引用对象嵌套较深还容易出错。

2.深拷贝实现—序列化:

fun main(args: Array<String>) {
    val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))
    val userCopy = deepCopy(user) as User

    println("user1:${user}")
    println("user2:${userCopy}")
    println("user address:${user.address == userCopy.address}")     //判断值,相当于equal
    println("user address:${user.address === userCopy.address}")    //判断内存地址
}

fun deepCopy(obj: Any?): Any {
    val bo = ByteArrayOutputStream()
    val oo = ObjectOutputStream(bo)
    oo.writeObject(obj)

    val bi = ByteArrayInputStream(bo.toByteArray())
    val oi = ObjectInputStream(bi)

    return oi.readObject()
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:false

这种实现方式需要类中所有引用类型(包括嵌套的)都实现 Serializable 接口,否则会抛出 NotSerializableException 异常。

3.深拷贝实现—Json

fun main(args: Array<String>) {
    val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))
    val userCopy = deepCopy(user)

    println("user1:${user}")
    println("user2:${userCopy}")
    println("user address:${user.address == userCopy.address}")     //判断值,相当于equal
    println("user address:${user.address === userCopy.address}")    //判断内存地址
}

inline fun <reified T> deepCopy(data: T): T {
    val gson = Gson()
    val json = gson.toJson(data)
    return gson.fromJson<T>(json, T::class.java)
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28, city=上海))
user address:true
user address:false

这里使用 Kotlin 的内联函数配合范型实化封装了 deepCopy 方法,其内部是通过Google 的 Json 解析库 Gson 进行实现的,先将对象转为 Json 字符串,然后再解析 Json 来创建新对象。

4.深拷贝实现—写时复制思想

以上深拷贝方式由于在拷贝时创建了大量对象,都会伴随性能损耗。其实可以借助 Copy On Write 思想,先通过浅拷贝将对象复制出来,再使用中再进行对象的创建,从而将对象创建的损耗平摊到后续业务中。

当然这种方式需要看具体业务场景再决定如何实现,如果应用一个复杂的模式,只得到一点点的性能提升,这就是所谓的过度设计,得不偿失。

经典场景

理论说完了,再结合实际场景来尝试使用。比如 App 的个人中心展示了用户的所有信息,如用户名,生日,地址,个性签名等,同时还有保存和重置按钮,其中保存按钮需要实时更新状态:只有真正的修改了用户信息才可以点击,否则置灰无法点击。重置按钮点击后会恢复页面原信息。

这种需求场景可以利用原型模式的思想,通过拷贝得到一个新对象,后续的信息修改都基于新对象,并且每次改动后都和原对象进行对比,判断字段值是否真正修改来更新保存按钮状态。在点击重置按钮时,重新基于原对象拷贝新对象,并根据该对象刷新页面实现重置功能。

总结

利用已有对象来创建新对象,以达到节省创建时间的目的,就叫做原型模式。

原型模式的核心是对象的拷贝,对象拷贝又分为浅拷贝和深拷贝,其中浅拷贝会共用引用类型对象,而深拷贝会创建新的引用对象。