成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

五分鐘 掌握 原型模式

開發 前端
大家好,我是老田,今天我給大家分享設計模式中的原型模式。用貼切的生活故事,以及真實項目場景來講設計模式,最后用一句話來總結這個設計模式。

 [[404102]]

大家好,我是老田,今天我給大家分享設計模式中的原型模式。用貼切的生活故事,以及真實項目場景來講設計模式,最后用一句話來總結這個設計模式。

故事

還記得大四那年找工作,無意中我得從網上找到一份相對漂亮的程序員簡歷模板,然后全班同學開啟瘋狂的簡歷拷貝(U盤)。同時也鬧出了一個笑話,有幾位同學,拷貝過去的簡歷,內容完全沒改,名字都沒有改,截止投給面試官(校招面試官)。后來,結果大家也應該能猜出來,大家都去實習了,部分人還在找工作。后面公司面試官和同伴的其他同學反饋:收到一毛一樣的簡歷,好幾份,回來大家一聊就知道問題出哪里了,承認了自己拷貝過去完全沒改就拿出去投了,害,尷尬的一匹。

把簡歷拷貝分為為兩種:

  • 一種是拷貝簡歷,然后把信息修改成自己的
  • 另外一種是,拷貝簡歷,內容什么都不改。

原型模式定義

Specify the kinds of objects to create using a prototype instance ,and create new objects by coping this prototype

大致意思:用原型實例指定創建對象的種類,并且通過復制這些原型創建新的對象。

原型模式:Prototype Pattern,屬于創建型模式。

調用者不需要知道任何創建細節,也不用調用構造方法來創建對象。

使用場景

原型模式有如下使用場景:

  • 類初始化消耗資源較多
  • new產生的一個對象需要非常繁瑣的過程(數據準備、訪問權限等)
  • 構造函數比較復雜
  • 循環體內生成大量對象時
  • 在Spring中,原型模式應用的非常廣泛,例如:scope='prototype'

我們可以將一些getter和setter之類封裝成一個工廠方法,然后對于使用的人來說,調用方法就可以了,不需要知道里面的getter和setter是怎么處理的。我們也可以使用JDK提供的實現Cloneable接口,實現快速復制。

創建對象的四種方式:

new、反射、克隆、序列化

實際案例

大家是否有遇到過這種常見,就是項目中規定,不能把與數據庫表映射的entity類返回給前端,所以通常返回給前端的有各種O,比如:XxxVO、XxxBO、XxxDTO...

這時候就會出現下面的場景,大家也想已經猜到了。

下面是與數據庫表映射的UserEntity實體類。

  1. public class UserEntity { 
  2.     private Long id; 
  3.     private String name
  4.     private Integer age; 
  5.     //....可能還有很多屬性 
  6.     //省略getter setter 

返回給前端或者調用方的UserVO實體類。

  1. public class UserVO { 
  2.     private Long id; 
  3.     private String name
  4.     private Integer age; 
  5.     //....可能還有很多屬性 
  6.     //省略getter setter 

此時,從數據庫里查出來的UserEntity需要轉換成UserVO,然后再返回給前端(或者調用方)。

  1. public class ObjectConvertUtil { 
  2.  
  3.     public static UserVo convertUserEntityToUserVO(UserEntity userEntity) { 
  4.         if (userEntity == null) { 
  5.             return null
  6.         } 
  7.         UserVo userVo = new UserVo(); 
  8.  
  9.         userVo.setId(userEntity.getId()); 
  10.         userVo.setName(userEntity.getName()); 
  11.         userVo.setAge(userEntity.getAge()); 
  12.          //如果還有更多屬性呢? 
  13.         return userVo; 
  14.     } 

從這個util類中,我們可以看出,如果一個類的屬性有幾十個,上百個的,這代碼量是不是有點恐怖?

于是,我們通常都會使用一些工具類來處理,比如常見有以下:

  1. BeanUtils.copy(); 
  2. JSON.parseObject() 
  3. Guava工具類 
  4. ..... 

這些工具類就用到了原型模式。

通過一個對象,創建一個新的對象。

也把原型模式稱之為對象的拷貝、克隆。

其實對象的克隆分淺克隆和深克隆,下面我們就來聊聊淺克隆和深克隆。

  • 淺克隆:創建一個新對象,新對象的屬性和原來對象完全相同,對于非基本類型屬性,仍指向原來對象的屬性所指向的對象的內存地址。
  • 深克?。簞摻ㄒ粋€新對象,屬性中引用的其他對象也會被克隆,不再指向原有對象地址。

我們先來聊聊淺克隆,都喜歡由淺入深。

淺克隆

比如,我現在相對用戶信息User進行克隆,但是User中有用戶地址信息UserAddress屬性。

以下是代碼的實現:

  1. //用戶地址信息 
  2. public class UserAddress  implements Serializable
  3.     private String province; 
  4.     private String cityCode; 
  5.  
  6.     public UserAddress(String province, String cityCode) { 
  7.         this.province = province; 
  8.         this.cityCode = cityCode; 
  9.     } 
  10. //用戶信息 
  11. public class User implements Cloneable { 
  12.     private int age; 
  13.     private String name
  14.     //用戶地址信息 
  15.     private UserAddress userAddress; 
  16.  
  17.     //getter setter 省略 
  18.  
  19.     @Override 
  20.     protected Object clone() throws CloneNotSupportedException {  
  21.         return super.clone(); 
  22.     } 
  23. //測試 
  24. public class UserTest { 
  25.     public static void main(String[] args) throws Exception { 
  26.         User user = new User(); 
  27.         user.setAge(20); 
  28.         user.setName("田維常"); 
  29.         UserAddress userAddress = new UserAddress("貴州""梵凈山"); 
  30.         user.setUserAddress(userAddress); 
  31.  
  32.         User clone = (Useruser.clone(); 
  33.  
  34.         System.out.println("克隆前后UserAddress比較:" + (user.getUserAddress() == clone.getUserAddress())); 
  35.     } 

輸出結果

  1. 克隆前后 UserAddress 比較:true 

兩個對象屬性 UserAddress 指向的是同一個地址。

這就是所謂的淺克隆,只是克隆了對象,對于該對象的非基本類型屬性,仍指向原來對象的屬性所指向的對象的內存地址。

關系如下:

深克隆

關于深克隆,我們來用一個很經典的案例,西游記里的孫悟空。一個孫悟空能變成n多個孫悟空,手里都會拿著一個金箍棒。

按照前面的淺克隆,結果就是:孫悟空倒是變成很多孫悟空,但是金箍棒用的是同一根。

深克隆的結果是:孫悟空變成了很多個,金箍棒也變成很多個根。

下面我們用代碼來實現:

  1. //猴子,有身高體重和生日 
  2. public class Monkey { 
  3.     public int height; 
  4.     public int weight; 
  5.     public Date birthday; 

孫悟空也是猴子,兵器 孫悟空有個金箍棒:

  1. import java.io.Serializable
  2. //孫悟空的金箍棒 
  3. public class JinGuBang implements Serializable
  4.     public float  h=100; 
  5.     public float  d=10; 
  6.     //金箍棒變大 
  7.     public void big(){ 
  8.         this.h *=10; 
  9.         this.d *=10; 
  10.     } 
  11.     //金箍棒變小 
  12.     public void small(){ 
  13.         this.h /=10; 
  14.         this.d /=10; 
  15.     } 

齊天大圣孫悟空:

  1. import java.io.*; 
  2. import java.util.Date
  3.  
  4. //孫悟空有七十二變,拔猴毛生成一個金箍棒 
  5. //使用JDK的克隆機制, 
  6. //實現Cloneable并重寫clone方法 
  7. public class QiTianDaSheng extends Monkey implements Cloneable, Serializable { 
  8.  
  9.     public JinGuBang jinGuBang; 
  10.  
  11.     public QiTianDaSheng() { 
  12.         this.birthday = new Date(); 
  13.         this.jinGuBang = new JinGuBang(); 
  14.     } 
  15.  
  16.     @Override 
  17.     protected Object clone() throws CloneNotSupportedException { 
  18.         return this.deepClone(); 
  19.     } 
  20.  
  21.     //深克隆 
  22.     public QiTianDaSheng deepClone() { 
  23.         try { 
  24.             //內存中操作完成、對象讀寫,是通過字節碼直接操作 
  25.             //與序列化操作類似 
  26.             ByteArrayOutputStream bos = new ByteArrayOutputStream(); 
  27.             ObjectOutputStream oos = new ObjectOutputStream(bos); 
  28.             oos.writeObject(this); 
  29.  
  30.             ByteArrayInputStream bais = new ByteArrayInputStream(bos.toByteArray()); 
  31.             ObjectInputStream bis = new ObjectInputStream(bais); 
  32.  
  33.             //完成一個新的對象,底層是使用new創建的一個對象 
  34.             //詳情可以了解readObject方法 
  35.             QiTianDaSheng qiTianDaSheng = (QiTianDaSheng) bis.readObject(); 
  36.             //每個猴子的生日不一樣,所以每次拷貝的時候,把生日改一下 
  37.             qiTianDaSheng.birthday = new Date(); 
  38.             return qiTianDaSheng; 
  39.         } catch (Exception ex) { 
  40.             ex.printStackTrace(); 
  41.             return null
  42.         } 
  43.     } 
  44.  
  45.     //淺克隆,就是簡單的賦值 
  46.     public QiTianDaSheng shalllowClone(QiTianDaSheng target) { 
  47.         QiTianDaSheng qiTianDaSheng = new QiTianDaSheng(); 
  48.         qiTianDaSheng.height = target.height; 
  49.         qiTianDaSheng.weight = target.weight; 
  50.  
  51.         qiTianDaSheng.jinGuBang = target.jinGuBang; 
  52.         qiTianDaSheng.birthday = new Date(); 
  53.         return qiTianDaSheng; 
  54.  
  55.     } 

接著我們就來測試一下:

  1. public class DeepCloneTest { 
  2.     public static void main(String[] args) { 
  3.         QiTianDaSheng qiTianDaSheng = new QiTianDaSheng(); 
  4.         try { 
  5.             QiTianDaSheng newObject = (QiTianDaSheng) qiTianDaSheng.clone(); 
  6.             System.out.print("深克隆后 "); 
  7.             System.out.println("金箍棒是否一直:" + (qiTianDaSheng.jinGuBang == newObject.jinGuBang)); 
  8.              
  9.         } catch (Exception ex) { 
  10.             ex.printStackTrace(); 
  11.         } 
  12.          
  13.         QiTianDaSheng newObject=qiTianDaSheng.shalllowClone(qiTianDaSheng); 
  14.         System.out.print("淺克隆后 "); 
  15.         System.out.println("金箍棒是否一直:" + (qiTianDaSheng.jinGuBang == newObject.jinGuBang)); 
  16.     } 

輸出結果為:

  1. 深克隆后 金箍棒是否一直:false 
  2.  
  3. 淺克隆后 金箍棒是否一直:true 

結論

深克隆后每個孫悟空都有自己的金箍棒,而淺克隆后每個孫悟空用的金箍棒實質上還是同一根。

總結

切記:深和淺,指的是克隆對象里的屬性(引用類型)是否指向同一個內存地址。

為了更深刻的理解深克隆和淺克隆,我們回答文中的簡歷拷貝的故事。

  • 深拷貝:拷貝一份簡歷,然后對簡歷中的信息進行修改成自己的
  • 淺拷貝:拷貝一份簡歷,簡歷內容完全不變

優點:

  • Java 原型模式基于內存二進制流復制,比直接 new 的性能會更好一些。
  • 可以利用深克隆保存對象狀態,存一份舊的(克隆出來),在對其修改,可以充當一個撤銷功能。

缺點:

  • 需要配置 clone 方法,改造時需要對已有類進行修改,違背 “開閉原則”。
  • 如果對象間存在多重嵌套引用時,每一層都需要實現克隆。

我們從原型模式的定義,使用場景,真實案例、淺克隆、深克隆、優缺點等方面,對原型模式進行了一個全面的講解。

一句話總結:

一份簡歷,全班同學用

本文轉載自微信公眾號「Java后端技術全棧」,可以通過以下二維碼關注。轉載本文請聯系Java后端技術全棧公眾號。

 

責任編輯:武曉燕 來源: Java后端技術全棧
相關推薦

2009-11-17 14:50:50

Oracle調優

2025-01-24 08:38:47

2021-01-11 09:33:37

Maven數目項目

2009-11-05 10:55:22

Visual Stud

2017-01-10 09:07:53

tcpdumpGET請求

2018-01-08 16:19:04

微信程序輪播圖

2021-01-13 09:23:23

優先隊列React二叉堆

2024-09-23 17:05:44

2020-10-21 14:29:15

原型模式

2021-05-18 08:52:31

Prototype 原型模式設計模式

2021-10-28 19:09:09

模式原型Java

2020-06-16 08:47:53

磁盤

2024-12-11 07:00:00

面向對象代碼

2025-03-13 06:22:59

2009-11-16 10:53:30

Oracle Hint

2017-04-25 12:07:51

AndroidWebViewjs

2024-03-21 09:51:22

Python爬蟲瀏覽網站

2024-06-07 08:19:05

2021-10-20 06:58:10

工具低代碼無代碼

2022-08-04 13:27:35

Pythonopenpyxl
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 男女性毛片 | 中文日本在线 | 国产亚洲精品久久yy50 | 国产黄色小视频 | 日韩一区二区在线观看视频 | 91九色视频 | 国产精品欧美一区二区三区不卡 | 亚洲欧美成人影院 | 国产精品1区2区3区 中文字幕一区二区三区四区 | 9porny九色视频自拍 | 一本岛道一二三不卡区 | 免费精品久久久久久中文字幕 | 国产精品久久久久久久久动漫 | 精品国产亚洲一区二区三区大结局 | 人人擦人人| 日韩欧美亚洲 | 羞羞视频在线免费 | av网站在线播放 | 狠狠色狠狠色综合系列 | 亚洲一一在线 | 国产精品久久久久久久免费观看 | 久久专区| 日本成人中文字幕 | 国产高清在线观看 | 91色综合| 不用播放器看的av | 亚洲午夜精品一区二区三区他趣 | 瑟瑟视频在线看 | 综合久久久久久久 | 成人在线视 | 一区二区亚洲 | 人人看人人射 | 久久精品久久久久久 | 色综合美女 | 久久精品免费观看 | 午夜视频一区 | 国产日韩欧美在线 | 中文字幕加勒比 | av在线免费观看网站 | 在线日韩av电影 | 黄网站涩免费蜜桃网站 |