Java觀察者模式詳解

觀察者模式(Observer Pattern)是一種行為型設計模式,它定義了對象之間的一對多依賴關系,當一個對象(主題)的狀態發生改變時,所有依賴于它的對象(觀察者)都會自動收到通知并更新。

核心概念

觀察者模式包含以下核心角色:

  1. ?Subject(主題/被觀察者)?

    • 維護觀察者列表
    • 提供注冊、刪除和通知觀察者的接口
  2. ?ConcreteSubject(具體主題)?

    • 實現Subject接口
    • 管理自身狀態,并在狀態變化時通知所有觀察者
  3. ?Observer(觀察者)?

    • 定義更新接口,接收主題通知
  4. ?ConcreteObserver(具體觀察者)?

    • 實現Observer接口
    • 根據主題的狀態變化執行具體操作

模式特點

  • ?松耦合?:主題和觀察者之間松耦合
  • ?自動通知?:狀態變化時自動通知觀察者
  • ?一對多?:一個主題可對應多個觀察者
  • ?動態訂閱?:可動態添加/刪除觀察者

應用場景

觀察者模式適用于以下場景:

  1. ?消息發布/訂閱系統?:如聊天室、新聞推送
  2. ?用戶界面組件?:GUI事件處理
  3. ?實時數據監控?:如股票價格變動、溫度監控
  4. ?事件驅動系統?:如游戲引擎、企業監控系統
  5. ?分布式系統?:節點狀態變更通知

代碼實現示例

以下是觀察者模式的基本實現(推模型):

// 觀察者接口
public interface Observer {void update(String message);
}// 主題接口
public interface Subject {void registerObserver(Observer o);void removeObserver(Observer o);void notifyObservers();
}// 具體主題 - 新聞發布中心
public class NewsAgency implements Subject {private List<Observer> observers = new ArrayList<>();private String news;public void setNews(String news) {this.news = news;notifyObservers();}public void registerObserver(Observer o) {observers.add(o);}public void removeObserver(Observer o) {observers.remove(o);}public void notifyObservers() {for (Observer o : observers) {o.update(news);}}
}// 具體觀察者 - 新聞訂閱者
public class NewsSubscriber implements Observer {private String name;public NewsSubscriber(String name) {this.name = name;}@Overridepublic void update(String message) {System.out.println(name + " 收到新聞: " + message);}
}

通知模型對比

觀察者模式有兩種主要的通知模型:

模型類型數據傳遞方式優點缺點
推模型主題主動推送詳細數據給觀察者實時性強,響應快可能傳遞冗余數據
拉模型觀察者收到通知后主動拉取數據按需獲取,靈活性高增加主題的訪問壓力

Java內置支持

Java在java.util包中提供了觀察者模式的實現:

  • Observable類(主題)
  • Observer接口

但自Java 9起已被標記為@Deprecated,推薦使用java.beans包中的PropertyChangeListenerPropertyChangeSupport

優缺點分析

?優點?:

  1. 主題和觀察者之間松耦合
  2. 支持廣播通信
  3. 符合開放-封閉原則
  4. 可動態添加/刪除觀察者

?缺點?:

  1. 通知順序不可控
  2. 觀察者過多時可能影響性能
  3. 可能導致循環依賴
  4. 觀察者不知道彼此存在,可能導致更新沖突

實際應用案例

  1. ?聊天系統?:聊天室作為主題,用戶作為觀察者
  2. ?溫度監控系統?:溫度傳感器作為主題,顯示設備和報警系統作為觀察者
  3. ?彩票系統?:彩票中心作為主題,購彩者作為觀察者
  4. ?企業監控系統?:員工電腦作為主題,安全模塊作為觀察者

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/88006.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/88006.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/88006.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

創世新布控球 國標

目錄 結論&#xff1a; UDP模式 對講 平臺頁面設置 設備tcp被動 舊的創世版本&#xff08;平臺選的設備tcp被動&#xff1b;設備側無法設置&#xff09; 新創世從2.8改到180上&#xff0c;先UDP&#xff0c;全報文 參考文檔 結論&#xff1a; 對講的tcp主被動&#xff0…

【Dify精講】第18章:企業級功能定制

在企業級AI應用的實際部署中&#xff0c;你很快就會發現開源版本的標準功能往往無法滿足復雜的業務需求。作為一個在多家企業實施AI系統的老兵&#xff0c;我深知企業級定制的痛點和需求。今天&#xff0c;讓我們一起深入Dify的企業級功能定制&#xff0c;看看如何在現有架構基…

PHP $_GET 變量詳解

PHP $_GET 變量詳解 引言 在PHP編程中,$_GET變量是處理HTTP GET請求參數的一種非常便捷的方式。本文將詳細介紹PHP $_GET變量的使用方法、特點以及在實際開發中的應用。 一、什么是$_GET變量? $_GET是一個預定義的PHP超級全局變量,用于存儲HTTP GET請求中的數據。當用戶…

Kafka動態配置深度解析

在分布式消息隊列領域&#xff0c;Kafka憑借其高吞吐量、低延遲和可擴展性成為眾多企業的首選。隨著業務場景的日益復雜和數據流量的動態變化&#xff0c;靜態配置已難以滿足需求&#xff0c;Kafka的動態配置功能應運而生。通過動態配置&#xff0c;用戶無需重啟集群或中斷服務…

為WIN10微軟輸入法的全角切換Bug禁用Shift+Space組合鍵

20250621 By wdhuag 目錄 前言&#xff1a; 參考&#xff1a; 使用AutoHotkey屏蔽快捷鍵&#xff08;推薦&#xff09;&#xff1a; 使用PowerToys的鍵盤管理器屏蔽快捷鍵&#xff08;不推薦&#xff09;&#xff1a; 網上其它的方法&#xff1a; 前言&#xff1a; 是的…

Shell腳本調試與錯誤處理詳解

在 Shell 腳本中&#xff0c;set 命令用于控制腳本的執行行為和調試選項。以下是詳細解釋&#xff1a; 1. set -e 和 set e set -e&#xff08;嚴格錯誤檢查&#xff09;&#xff1a; 當命令返回非零退出狀態&#xff08;失敗&#xff09;時&#xff0c;立即退出腳本。 示例&a…

鯤鵬服務器創建Zookeeper鏡像實例

配置Kafka過程中&#xff0c;少不了要使用Zookeeer&#xff0c;這里記錄一下配置Zookeeper鏡像實例的過程。 創建目錄 mkdir -p /data/docker/zookeeper/data mkdir -p /data/docker/zookeeper/conf mkdir -p /data/docker/zookeeper/logs說明&#xff1a;data目錄為數據掛載…

GitHub Actions 自動 CI 測試 WorkFlow工作流搭建

大家好&#xff0c;我是此林。 代碼托管平臺 Github 我們應該比較熟悉。每次我們提交代碼到 GitHub 倉庫時&#xff0c;特別是開源項目&#xff0c;一般都會自動觸發測試腳本運行&#xff0c;幫你驗證代碼沒有引入新的錯誤。 這個其實就是 GitHub Actions&#xff0c;一般我們…

0-機器學習簡介

有監督學習 目標&#xff1a;建立一個模型(函數)&#xff0c;來描述輸入(x)和輸出(y)之間的映射關系。 價值&#xff1a;模型訓練完成后&#xff0c;新的輸入&#xff0c;模型會給出預測值輸出。 注意點&#xff1a; 1.要有足夠的訓練樣本 2.輸入和輸出之間有關聯關系 3.輸入…

前端跨域解決方案(6):Nginx

1 Nginx 核心 Nginx 是一個開源的高性能 HTTP 和反向代理服務器&#xff0c;以輕量級、高并發處理能力和低資源消耗著稱。除作為 Web 服務器外&#xff0c;還可充當郵件代理服務器和通用的 TCP/UDP 代理服務器&#xff0c;廣泛應用于現代 Web 架構中。 在 Windows 系統中使用…

C++智能指針編程實例

智能指針是C11引入的重要特性&#xff0c;用于自動管理動態分配的內存&#xff0c;防止內存泄漏。下面介紹幾種高級智能指針編程實例。 1. 共享所有權模式 (shared_ptr) 循環引用問題及解決方案 #include <memory> #include <iostream>class B; // 前向聲明clas…

單元測試總結

一、測試方案: 單元測試方案應包括以下步驟: 1.理解代碼結構:仔細閱讀代碼,理解程序的結構、邏輯和算法。 2.制定測試目標:明確你想要測試的功能和輸出結果; 3.撰寫測試用例:編寫涵蓋所有測試目標的測試用例; 4.執行測試:運行測試用例以驗證功能的正確性; 5.編寫報告:根據測試…

Spring面向切面編程AOP(2)

前置通知&#xff08;Before Advice&#xff09; 前置通知在目標方法執行之前被調用&#xff0c;常用于執行一些預處理邏輯&#xff0c;例如權限驗證、參數校驗等。在 Spring 配置文件中&#xff0c;前置通知通過<aop:before>標簽進行配置&#xff0c;以下是一個典型的示…

設備故障預測與健康管理技術:從數據到決策的工業智能進化之路?

在工業 4.0 與智能制造浪潮的推動下&#xff0c;設備故障預測與健康管理&#xff08;Prognostics and Health Management, PHM&#xff09;技術已成為企業實現數字化轉型的核心驅動力。據統計&#xff0c;制造業中設備非計劃停機 1 小時的平均損失高達 25 萬美元&#xff0c;而…

RabbitMQ從入門到實踐:消息隊列核心原理與典型應用場景

在現代應用開發中&#xff0c;系統各部分之間的通信至關重要。這就是像RabbitMQ這樣的消息代理發揮作用的地方。無論您是在構建微服務架構、實現任務隊列&#xff0c;還是開發實時聊天應用程序&#xff0c;RabbitMQ都可能成為改變游戲規則的工具。本文將深入探討RabbitMQ是什么…

基于Spring Boot和Vue的網上軍事論壇設計與實現

目錄 一.&#x1f981;前言二.&#x1f981;開源代碼與組件使用情況說明三.&#x1f981;核心功能1. ?算法設計2. ?Java開發語言3. ?Redis數據庫4. ?部署項目 四.&#x1f981;演示效果1. 管理員模塊1.1 用戶管理1.2 內容審核1.3 權限分配1.4 菜單管理1.5 字典管理 2. 用戶…

LLMs基礎學習(八)強化學習專題(6)

LLMs基礎學習&#xff08;八&#xff09;強化學習專題&#xff08;6&#xff09; 文章目錄 LLMs基礎學習&#xff08;八&#xff09;強化學習專題&#xff08;6&#xff09;深度強化學習&#xff08;DQN&#xff09;DQN 起源&#xff1a;《Playing Atari with Deep Reinforceme…

JVM(10)——詳解Parallel垃圾回收器

Parallel 垃圾回收器&#xff08;也稱為 吞吐量優先收集器&#xff09;。它是 Java 早期&#xff08;特別是 JDK 8 及之前&#xff09;在多核處理器上的默認垃圾回收器&#xff0c;其核心設計目標是最大化應用程序的吞吐量。 一、Parallel 回收器的定位與設計目標 核心目標&am…

MySQL(91)什么是分布式數據庫?

分布式數據庫是一種將數據存儲在多個物理位置的數據庫系統。這些位置可能分布在不同的服務器、數據中心甚至地理位置。分布式數據庫系統允許數據的存儲、處理和訪問分布在多個節點上&#xff0c;以提高數據的可用性、可靠性、可擴展性和性能。 1. 分布式數據庫的特點 1.1 數據…

Java事務失效(面試題)的常見場景

1. 方法非public修飾 原理&#xff1a; Spring AOP代理&#xff08;CGLIB或JDK動態代理&#xff09;默認無法攔截非public方法。 示例&#xff1a; Service public class UserService {Transactionalvoid updateUser() { // 非public方法// 事務不會生效&#xff01;} } 修…