設計模式 ~ 享元模式

来源:https://www.cnblogs.com/liuxh-cn/archive/2020/05/23/she-ji-mo-shi--xiang-yuan-mo-shi.html
-Advertisement-
Play Games

概念內部狀態、外部狀態、享元池角色 & UMLDemo: 編輯器圖片重用 - JavaReference概念享元模式(Flyweight Pattern),是以 共用 的方式,對 大量細粒度對象 重用,來減少記憶體的使用(避免大量重覆地創建、銷毀對象)。名稱中的Flyweight,是搏擊比賽中體重級別... ...


概念

享元模式(Flyweight Pattern),是以 共用 的方式,對 大量細粒度對象 重用,來減少記憶體的使用(避免大量重覆地創建、銷毀對象)。

名稱中的Flyweight,是搏擊比賽中體重級別之一,中文稱為 蠅量級次最輕量級 。把這個單詞移植到軟體工程中,也是用來表示特別小的對象,即細粒度對象。由此可見,享元模式的特點不僅是 共用,更是強調 細粒度的共用

內部狀態、外部狀態、享元池

享元類中可以共用的相同的內容稱為 內部狀態(Intrinsic State),需要外部環境設置的特異內容稱為 外部狀態(Extrinsic State),二者相互獨立,彼此解耦,組合在一起共同構成了享元類。不同但是有關係的(比如同一類的)享元對象聚合在一起構成了一個享元池,使用一個享元工廠來維護這個享元池
這樣在使用的時候,根據需求,調用享元工廠(在其維護的享元池中)獲取需要的享元對象,然後對其配置外部狀態,來使用。

由於區分了 內部狀態外部狀態,因此可以通過設置不同的外部狀態使得相同的對象具有一些不同的特征,而相同的內部狀態使可以共用的。所以,享元模式的本質是 將完整的對象分解出更細的粒度,來解耦變與不變,並共用不變,實現減少對象數量並節約記憶體的目的。

享元模式借鑒單例模式的共用,借鑒工廠模式的統一供應,其特點是對細粒度對象重用
與單例模式的區別在於,細粒度重用,而不是把完整的對象重用。

角色 & UML

抽象享元類
具體享元類
享元工廠類
客戶端

Demo: 編輯器圖片重用 - Java

需求:
享元模式在編輯器軟體中大量使用,如在一個文檔中多次出現相同的圖片,則只需要創建一個圖片對象,通過在應用程式中設置該圖片出現的位置,來實現該圖片在不同地方多次重覆顯示。

UML:

代碼結構:

./
├── Client.java
├── ImageNodeFactory.java
├── flyweight
│   ├── BirdImage.java
│   ├── ImageNode.java
│   ├── TreeImage.java
│   ├── bird.jpeg
│   └── tree.jpg
└── result.pdf

在這裡,PDF編輯使用itextpdf包,pom.xml中配置

<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itextpdf</artifactId>
    <version>5.5.13</version>
</dependency>

ImageNode <Interface>: 抽象享元類

package homework.Flyweight.flyweight;

import com.itextpdf.text.BadElementException;

import com.itextpdf.text.Image;
import java.io.IOException;

public interface ImageNode {


//    返回包裝了坐標(外部狀態 << 參數)的圖片(內部狀態)對象
    public Image getImageInPdf(int x, int y) throws IOException, BadElementException;
}

BirdImage <Class>: 具體享元類

package homework.Flyweight.flyweight;

import com.itextpdf.text.BadElementException;
import com.itextpdf.text.Image;

import javax.imageio.ImageIO;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;

public class BirdImage implements ImageNode {

//    享元維護的內部狀態(圖片,共用)
    private byte[] image;

//    載入內部狀態(圖片)
    public BirdImage() throws IOException {
        ByteArrayOutputStream baos = new ByteArrayOutputStream();

        ImageIO.write(
                ImageIO.read(new File("src/main/java/homework/Flyweight/flyweight/bird.jpeg")),
                "jpeg",
                baos);

        this.image = baos.toByteArray();
    }

    @Override
    public Image getImageInPdf(int x, int y) throws IOException, BadElementException {
        Image imageInPdf = Image.getInstance(this.image);

//        內部狀態(圖片對象) + 外部狀態(圖片位置)
        imageInPdf.scaleAbsolute(30, 30);
        imageInPdf.setAbsolutePosition(x, y);

//        列印圖片在記憶體中的地址
        System.out.println(this.image);
        return imageInPdf;
    }
}

TreeImage <Class>: 具體享元類

package homework.Flyweight.flyweight;

import com.itextpdf.text.BadElementException;
import com.itextpdf.text.Image;

import javax.imageio.ImageIO;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;

public class TreeImage implements ImageNode {

//    享元維護的內部狀態(圖片,共用)
    private byte[] image;

//    載入內部狀態(圖片)
    public TreeImage() throws IOException {
        ByteArrayOutputStream baos = new ByteArrayOutputStream();

        ImageIO.write(
                ImageIO.read(new File("src/main/java/homework/Flyweight/flyweight/tree.jpg")),
                "jpg",
                baos);

        this.image = baos.toByteArray();
    }

    @Override
    public Image getImageInPdf(int x, int y) throws IOException, BadElementException {
        Image imageInPdf = Image.getInstance(image);

//        內部狀態(圖片對象) + 外部狀態(圖片位置)
        imageInPdf.scaleAbsolute(30, 30);
        imageInPdf.setAbsolutePosition(x, y);

//        列印圖片在記憶體中的地址
        System.out.println(this.image);
        return imageInPdf;
    }
}

ImageNodeFactory <Class>: 享元工廠

package homework.Flyweight;

import homework.Flyweight.flyweight.BirdImage;
import homework.Flyweight.flyweight.ImageNode;
import homework.Flyweight.flyweight.TreeImage;

import java.io.IOException;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

public class ImageNodeFactory {

//    享元工廠維護的享元池
    private static Map<String, ImageNode> imagePool = new ConcurrentHashMap<>();

//    獲取享元
    public static ImageNode getImageNode(String type) throws IOException {
        if(!imagePool.containsKey(type)) {
            switch (type) {
                case "bird":
                    imagePool.put(type, new BirdImage());
                    break;
                case "tree":
                    imagePool.put(type, new TreeImage());
                    break;
            }
        }

        return imagePool.get(type);
    }

}

Client <Class>: 客戶端

package homework.Flyweight;

import com.itextpdf.text.*;
import com.itextpdf.text.pdf.PdfWriter;
import homework.Flyweight.flyweight.ImageNode;

import java.io.*;

public class Client {

    public static void main(String[] args) throws IOException, DocumentException {

//        1. 初始化PDF
        Rectangle rectangle = new Rectangle(PageSize.A4);       // 設置 PDF 紙張矩形,大小採用 A4
        rectangle.setBackgroundColor(BaseColor.WHITE);          // 設置背景色
        // 創建一個文檔對象,設置初始化大小和頁邊距
        Document document = new Document(rectangle, 10, 10, 10, 10);     // 上、下、左、右頁間距

        String pdfPath = "src/main/java/homework/Flyweight/result.pdf";   // PDF 的輸出位置
        PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream(pdfPath));


        document.open();    // 打開文檔對象

//        2. PDF中插入圖片
        for(int i=0; i<10; i++){
            ImageNode imageNode = ImageNodeFactory.getImageNode("tree");
//            圖片坐標(外部狀態)由客戶端維護
            document.add(imageNode.getImageInPdf(10*i,100*i));
            
            System.out.println("Object ImageNode: "+imageNode);

        }

        document.close();   // 關閉文檔
    }
}

控制台列印:

[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f
[B@16b4a017
Object ImageNode: homework.Flyweight.flyweight.TreeImage@5ecddf8f

Process finished with exit code 0

生成的PDF:

Reference

  1. https://www.jianshu.com/p/f9997c0c2a55
  2. 享元模式-連連看的圖片共用 https://blog.csdn.net/ABAP_Brave/article/details/78528416
  3. 結合JDK源碼看設計模式——享元模式

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 實操才能去感受自己的缺陷所在,雖然在演算法這一塊很弱勢,但不斷堅持,改變自己。 ...
  • 狀態模式: 根據狀態決定動作 當動作一定但是狀態可擴展適合使用狀態模式 當動作變化不適合 當狀態不會擴展也沒有必要使用使用傳統的switch即可。 先看一個沒有使用狀態模式的例子: package com.srr.dp.state; /** * 我的女朋友有很多種狀態 * 當給我的女朋友添加新的狀態 ...
  • Spring Boot 教程 swagger ui 1. 什麼是Swagger? Swagger™的目標是為REST APIs 定義一個標準的,與語言無關的介面,使人和電腦在看不到源碼或者看不到文檔或者不能通過網路流量檢測的情況下能發現和理解各種服務的功能。當服務通過Swagger定義,消費者就能 ...
  • Spring Boot 教程 RESTful 1. RESTful風格 1.1 簡介與特點 RESTful是一種網路應用程式的設計風格和開發方式,基於 "HTTP" ,可以使用 "XML" 格式定義或 "JSON" 格式定義。RESTful適用於移動互聯網廠商作為業務使能介面的場景,實現第三方 "O ...
  • 眾所周知,升級某個庫(假設為 xxx),可以用 命令,或者簡寫成 。 如果有多個庫,可以依次寫在 xxx 後面,以空格間隔。那麼,如何簡單優雅地批量更新系統中全部已安裝的庫呢? 接下來我們直奔主題,帶大家學習幾種方法/騷操作吧! 方法一:pip list 結合 Linux 命令 命令可以查詢已安裝的 ...
  • Redis持久化過程一直是影響redis性能的常見因素,如何監控持久化以及如何優化持久化過程呢?下麵我們就一起來看看吧。 fork的監控及優化 不管是使用哪種持久化,RDB持久化或AOF重寫,主進程都會fork出一個子進程,在子進程里完成rdb文件的生成或aof的重寫。fork操作對於操作系統來說屬 ...
  • Spring MVC的執行流程 一、名詞解釋 1、前端控制器(DispatcherServlet) 接收請求,響應結果,相當於轉發器,中央處理器 2、處理器映射器(HandlerMapping) 根據請求的url查找Handler(處理器/Controller) 可以通過XML和註解方式實現映射。 ...
  • 這是一個群友分享出來的一次阿裡巴巴面試題 ,介紹一下,渣渣雙非本,想要隨緣求offer,他說他知道自己的菜,沒想到還面過了,所以我就找他要了這次的面試經歷,來告訴大家,不要覺得自己菜,就不敢去大廠面試,可能就是因為你的菜而收到offer了。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...