【第12节】C++设计模式(结构型模式)-Proxy(代理)模式

一、问题背景

使用 Proxy 模式优化对象访问

在某些情况下,直接访问对象可能会导致性能问题或安全性问题。Proxy 模式(代理模式)通过引入一个代理对象来控制对原始对象的访问,从而解决这些问题。以下是几种典型的应用场景:

(1)虚代理(Virtual Proxy): 当创建对象的开销较大时(例如加载一幅大图片),可以将创建过程交给代理对象来完成。代理对象在实际需要时才会创建真实对象,从而延迟对象的创建。

(2)远程代理(Remote Proxy): 当需要操作网络上的对象时(尤其是在网络性能较差的情况下),可以通过代理对象来操作远程对象。代理对象负责处理网络通信细节,客户端无需关心。

(3)保护代理(Protection Proxy): 当需要对对象的访问进行控制时(例如在论坛系统中,管理员和普通用户的操作权限不同),可以通过代理对象来实现权限控制。

**(4)智能指针(Smart Pointer):**智能指针是一种特殊的代理,用于管理动态内存的分配和释放。关于智能指针的更多内容,可以参考 Andrew Koenig 的《C++ 沉思录》第 5 章。

二、Proxy 模式简介

Proxy 模式是一种结构型设计模式,它通过引入一个代理对象来控制对原始对象的访问。代理对象与原始对象实现相同的接口,客户端通过代理对象间接访问原始对象。Proxy 模式的核心思想是将客户端与原始对象解耦,从而在不改变客户端代码的情况下增强或控制对原始对象的访问。

Proxy 模式的优点

(1)延迟加载:通过虚代理,可以延迟对象的创建,直到真正需要时才创建对象。

(2)远程访问:通过远程代理,可以简化客户端对远程对象的访问。

(3)访问控制:通过保护代理,可以实现对对象的访问控制。

(4)解耦:代理模式将客户端与原始对象解耦,使得客户端无需关心原始对象的具体实现。

三、代码实现

Proxy 模式的核心思想非常简单:通过引入一个代理对象来间接访问目标对象,从而在不修改目标对象的情况下,增加额外的功能或控制逻辑。以下是一个简单的 Proxy 模式实现示例,使用 C++ 编写。

代码结构

(1)Subject 类:定义了真实对象和代理对象的共同接口。

(2)ConcreteSubject 类:实现了 `Subject` 接口,是真实的对象。

(3)Proxy 类:实现了 `Subject` 接口,并持有一个 `ConcreteSubject` 对象的引用,控制对 `ConcreteSubject` 的访问。

完整代码

cpp 复制代码
// Proxy.h
#ifndef _PROXY_H_
#define _PROXY_H_

// Subject 类:定义真实对象和代理对象的共同接口
class Subject {
public:
    virtual ~Subject() {}
    virtual void Request() = 0; // 请求接口
protected:
    Subject() {}
};

// ConcreteSubject 类:实现 Subject 接口,是真实的对象
class ConcreteSubject : public Subject {
public:
    ConcreteSubject() {}
    ~ConcreteSubject() {}
    void Request() override; // 实现请求接口
};

// Proxy 类:实现 Subject 接口,并持有一个 ConcreteSubject 对象的引用
class Proxy : public Subject {
public:
    Proxy();
    Proxy(Subject* sub); // 构造函数,传入真实对象
    ~Proxy();
    void Request() override; // 实现请求接口
private:
    Subject* _sub; // 持有的真实对象
};

#endif //~_PROXY_H_
cpp 复制代码
// Proxy.cpp
#include "Proxy.h"
#include <iostream>
using namespace std;

// ConcreteSubject 的实现
void ConcreteSubject::Request() {
    cout << "ConcreteSubject::Request() called." << endl;
}

// Proxy 的实现
Proxy::Proxy() : _sub(nullptr) {}

Proxy::Proxy(Subject* sub) : _sub(sub) {}

Proxy::~Proxy() {
    delete _sub; // 释放真实对象
}

void Proxy::Request() {
    cout << "Proxy::Request() called." << endl;
    if (_sub) {
        _sub->Request(); // 调用真实对象的请求接口
    }
}
cpp 复制代码
// main.cpp
#include "Proxy.h"
#include <iostream>
using namespace std;

int main(int argc, char* argv[]) {
    Subject* sub = new ConcreteSubject(); // 创建真实对象
    Proxy* p = new Proxy(sub); // 创建代理对象,并传入真实对象
    p->Request(); // 通过代理对象调用请求接口
    delete p; // 释放代理对象
    return 0;
}

代码说明

(1)Subject 类:定义了真实对象和代理对象的共同接口 `Request`。

(2)ConcreteSubject 类:实现了 `Request` 接口,是真实的对象。

(3)Proxy 类:实现了 `Request` 接口,并持有一个 `ConcreteSubject` 对象的引用。在 `Request` 方法中,代理对象可以在调用真实对象的 `Request` 方法前后执行一些额外的操作(例如权限检查、日志记录等)。

(4)客户端:在 `main` 函数中,客户端通过代理对象 `p` 调用 `Request` 方法,而无需直接与真实对象 `sub` 交互。

四、总结讨论

Proxy 模式的最大优点在于它实现了逻辑和实现的彻底解耦。通过引入代理对象,可以在不改变客户端代码的情况下增强或控制对原始对象的访问。例如:

(1)延迟加载:代理对象可以在真正需要时才创建真实对象,从而节省资源。

(2)远程访问:代理对象可以隐藏网络通信的细节,使客户端像访问本地对象一样访问远程对象。

(3)访问控制:代理对象可以在调用真实对象的方法之前进行权限检查,从而保护真实对象。

Proxy 模式通过引入代理对象来控制对原始对象的访问,从而解决了直接访问对象可能导致的性能问题、安全性问题等。它适用于需要延迟加载、远程访问、访问控制等场景。通过合理使用 Proxy 模式,可以有效地提高系统的灵活性和可维护性。

相关推荐
?!7148 分钟前
算法打卡第18天
c++·算法
pengyu17 分钟前
【Java设计原则与模式之系统化精讲:壹】 | 编程世界的道与术(实战指导篇)
java·后端·设计模式
zh_xuan23 分钟前
c++ std::pair
开发语言·c++
CodeWithMe1 小时前
【C/C++】EBO空基类优化介绍
开发语言·c++
k要开心1 小时前
从C到C++语法过度1
开发语言·c++
小吕学编程1 小时前
策略模式实战:Spring中动态选择商品处理策略的实现
java·开发语言·设计模式
whoarethenext1 小时前
使用 C/C++的OpenCV 实时播放火柴人爱心舞蹈动画
c语言·c++·opencv
能工智人小辰2 小时前
Codeforces Round 509 (Div. 2) C. Coffee Break
c语言·c++·算法
梦星辰.2 小时前
VSCode CUDA C++进行Linux远程开发
linux·c++·vscode
pan_junbiao2 小时前
Spring框架的设计模式
java·spring·设计模式