设计模式 - 模板方法模式 笔记

我们实现一个功能的时候,很多时候,会先写好大致的框架的代码,然后再引入插件、类库来具体实现代码功能。

假设我正在设计一个媒体播放器模块。我先写好一个【是否正在播放】属性,一个【播放】方法,一个【停止播放】方法,【播放】方法中首先判定是否正在播放,没有播放再播放;【停止播放】方法逻辑趋同。

( C# 开发 )

一开始想用一个接口实现,先写好接口作为框架,但是接口中的方法貌似不建议填充内容,问问了问AI,它告诉我:

业界普遍的做法有以下几种:

方案一:抽象基类(最常用、最清晰)

这是最直接的"空框架"实现。先把通用逻辑写好,具体播放/停止逻辑留空让子类实现。

csharp

复制代码
public abstract class PlayerBase
{
    // 框架层:状态字段
    private bool _isPlaying;
    
    // 框架层:只读属性,子类可以获取状态,但最好由基类维护
    public bool IsPlaying => _isPlaying;
    
    // 播放方法:框架实现状态判断 + 状态变更
    public void Play()
    {
        if (IsPlaying)
        {
            Console.WriteLine("已在播放,忽略 Play 调用");
            return;
        }
        
        // 调用子类实现的具体播放逻辑
        OnPlay();
        
        // 播放后修改状态
        _isPlaying = true;
    }
    
    // 停止方法:类似逻辑
    public void Stop()
    {
        if (!IsPlaying)
        {
            Console.WriteLine("已停止,忽略 Stop 调用");
            return;
        }
        
        OnStop();
        _isPlaying = false;
    }
    
    // 子类需要实现的具体行为
    protected abstract void OnPlay();
    protected abstract void OnStop();
}

子类实现:

csharp

复制代码
public class Mp3Player : PlayerBase
{
    protected override void OnPlay()
    {
        Console.WriteLine("MP3 开始播放...");
        // 真正的 MP3 播放逻辑
    }
    
    protected override void OnStop()
    {
        Console.WriteLine("MP3 停止播放");
        // 真正的停止逻辑
    }
}

优点:

  • 完美封装"先判断状态再执行"的通用流程

  • 基类自行维护 IsPlaying 状态,子类无需操心

  • 子类只关心具体播放/停止的实现

  • 符合模板方法模式

方案二:接口 + 抽象基类(更灵活,适合多人协作)

如果你的框架需要定义多个不相关的播放器族,接口用于对外暴露契约,抽象基类用于代码复用。

csharp

复制代码
// 1. 定义公共接口(契约)
public interface IMediaPlayer
{
    bool IsPlaying { get; }
    void Play();
    void Stop();
}

// 2. 提供默认实现框架(可复用)
public abstract class PlayerBase : IMediaPlayer
{
    private bool _isPlaying;
    public bool IsPlaying => _isPlaying;
    
    public void Play()
    {
        if (_isPlaying) return;
        OnPlay();
        _isPlaying = true;
    }
    
    public void Stop()
    {
        if (!_isPlaying) return;
        OnStop();
        _isPlaying = false;
    }
    
    protected abstract void OnPlay();
    protected abstract void OnStop();
}

优点:

  • 接口定义了标准,其它团队可以自己实现接口(不用基类)

  • 基类提供了便捷的实现方式(推荐使用基类)

  • 扩展性更好


而当我问它,这种先搭框架代码,再考虑具体实现的做法,业界最多的称呼是什么:它说就是

模板方法模式, 是吗?就当是吧,在此记录一下(完)

相关推荐
xuhaoyu_cpp_java1 小时前
项目学习(三)分页查询
java·经验分享·笔记·学习
Cloud_Shy6185 小时前
解读《Effective Python 3rd Edition》:从练气到老魔(第五章 Item 33 - 35)
开发语言·人工智能·笔记·python·学习方法
做cv的小昊5 小时前
计算机图形学:【Games101】学习笔记08——光线追踪(辐射度量学、渲染方程与全局光照、蒙特卡洛积分与路径追踪)
图像处理·笔记·学习·计算机视觉·游戏引擎·图形渲染·概率论
星恒随风5 小时前
C++ 类和对象入门(五):初始化列表、explicit 和 static 成员详解
开发语言·c++·笔记·学习·状态模式
艾利克斯冰5 小时前
Java 设计模式-行为型模式(更新中)
java·开发语言·设计模式
伊布拉西莫9 小时前
【流畅的Python】第20章:并发执行器 — 学习笔记
笔记·python·学习
星心源七境10 小时前
七境体系全解析:从六韬兵法到AI锁颜,一套贯穿古典智慧与现代应用的成长操作系统
人工智能·设计模式·设计
AOwhisky10 小时前
学习自测与解析:MySQL第五、六、七期核心知识点详解
运维·数据库·笔记·学习·mysql·云计算
niuniuyi~11 小时前
QT学习笔记
笔记·qt·学习
咸甜适中11 小时前
rust语言学习笔记Trait(十六)Error(错误)
笔记·学习·rust