喵哥讲了pimpl模式之后的再补充

简介: 喵哥讲了pimpl模式之后的再补充

前言:

之前看到喵哥在知乎上回答了一个设计模式的问题,其中介绍了pimpl模式(Private Implementation,顾名思义,将实现私有化,力图使得头文件对改变不透明)。”这个和qt里面的d-pointer用法应该是一致的“,我也给喵哥做了个小回复。

这是喵哥问题的截图:

839886cd3c004deb9abdba851aa48a4c.png下面我也来分享一下自己平时使用的d-pointer用法。

作者:良知犹存

转载授权以及围观:欢迎关注微信公众号羽林君

或者添加作者个人微信:become_me


喵哥的回答介绍

// MyClass.h
class MyClass {
public:
    void func1();
    void func2();
private:
    class impl;
    impl* pimpl;
};
// MyClass.cc
class MyClass::impl {
public:
    void func1();
    void func2();
private:
    void func3();
    void func4();
    int a;
    int b;
};
MyClass::MyClass() {
    pimpl = new impl;
}
void MyClass::func1() {
    pimpl->func1();
}

将类的private属性隐藏进一个内部类,然后通过一个指针访问(提前声明)它的接口。在头文件中只

暴露出应该暴露的功能,然后持有一个Impl的指针,而Impl则具体在http://MyClass.cc中定义,

用户什么都看不到。然后所有的功能都通过Impl完成。头文件里的Impl的指针也可以通过智能指针(u

nique_ptr)来代替,但这不是本文的重点。

再总结一下pimpl模式的优点:

非常适合隐藏private实现:如果想要在头文件中暴露public接口,但又不想暴露private实现的细节,则可以使用pimpl模式来隐藏细节。

pimpl模式也被称为编译防火墙,是一种用来减少编译时间的方法。通常来讲,如果头文件里的某些内容变更了,意味着所有引用该头文件的代码都要被重新编译,即使变更的是无法被用户类访问的私有成员。将这部分代码从被引用多次的头文件里移除到只被引用编译一次的源文件中,更改此文件就不会付出太长的编译时间。

作者:程序喵大人

链接:https://www.zhihu.com/question/340301316/answer/2264148507

个人的一些使用补充:

喵哥介绍的已经很详细了,写了一个demo,还把pimpl模式优点给大家介绍了,我就不多做赘述了。

但是我想给大家分享一种我自己实际使用的 d-pointer方法,有些区别,但是这种模式下的变种,分享给大家,希望可以对大家有些帮助。

Pimpl或者d-pointer机制其实这是桥接模式的一种组合使用。通过在新的类定义使用其他类,我们可以对实现某个功能类随意的进行增删和修改,

不过Pimpl也有些缺点:

例如,每次函数调用都涉及到指针操作,程序运行速度可能变慢;此外也需要在堆上开辟空间,记得使用完之后 delete,一般也建议使用智能指针定义。

d-pointer和Pimpl使用组建上都是差不多的,我们会定义一个类或者结构体,然后再在使用的类中进行 定义该类的指针变量,在使用时候 new。

按理说都差不多情况下,选择一种就好了,为什么还要介绍 d-pointer呢,因为Pimpl有些场景没法一起共存使用,例如我定义的impl类功能十分庞杂,我想把它定义为一个单独的文件,这个时候pimpl就不太好使用了。因为pimpl模式下,被桥接使用的类属于最外层类的一个类成员,我们对该被桥接类的定义都被括在该文件下。而d-pointer使用中,被桥接类只要在文件最前面申明一下就好。其他的使用方式是和pimpl没有区别的。

下面是我使用的示例demo:

test_pimpl.hpp

#pragma once
#include <iostream>
class impl{
  public:
    impl(){}
    ~impl(){}
    void func1(){
      std::cout<<__FUNCTION__<<" target "<<std::endl;
    }
    void func2(){}
  private:
    void func3(){}
    void func4(){}
    int a;
    int b;
};

pimpl_bridge.h

#pragma once
class impl;//pimpl模式在类内部定义
class MyClass {
  public:
    MyClass();
    void func1();
    void func2();
  private:
    impl* pimpl;
};

main.cpp

#include <iostream>
#include "test_pimpl.hpp"
#include "pimpl_bridge.h"
MyClass::MyClass() {
  pimpl = new impl();
}
void MyClass::func1() {
  pimpl->func1();
}
int main()
{ 
  MyClass test;
  test.func1();
}

结语

这就是我自己的一些设计模式的使用分享。如果大家有更好的想法和需求,也欢迎大家加我好友交流分享哈。


作者:良知犹存,白天努力工作,晚上原创公号号主。公众号内容除了技术还有些人生感悟,一个认真输出内容的职场老司机,也是一个技术之外丰富生活的人,摄影、音乐 and 篮球。关注我,与我一起同行。



目录
相关文章
|
C++ Windows
vs2017下dump文件自动生成及dump文件的调试使用
vs2017下dump文件自动生成及dump文件的调试使用
812 0
vs2017下dump文件自动生成及dump文件的调试使用
|
机器学习/深度学习 缓存 前端开发
快速上手python的简单web框架flask
python可以做很多事情,虽然它的强项在于进行向量运算和机器学习、深度学习等方面。但是在某些时候,我们仍然需要使用python对外提供web服务。
快速上手python的简单web框架flask
|
机器学习/深度学习 传感器 算法
基于线性二次调节器(LQR)法实现机器人路径规划附matlab代码
基于线性二次调节器(LQR)法实现机器人路径规划附matlab代码
|
小程序 IDE 开发工具
【社区每周】修复开启样式隔离后, picker-view 组件部分属性不生效(11月第二期)
【社区每周】修复开启样式隔离后, picker-view 组件部分属性不生效(11月第二期)
173 10
|
C++
C++11日志之spdlog
C++11日志之spdlog
324 1
|
编译器 程序员 C语言
【C++ 类型系统】了解C++ 中 标量、复合、标准布局、平凡和聚合类型
【C++ 类型系统】了解C++ 中 标量、复合、标准布局、平凡和聚合类型
602 0
|
网络协议 Linux Shell
一文打通gentooLinux(Uefi/SystemD/gnome/国内源flatpak/windowsmanager/自裁kernel)
## 注意 本人使用 ``` install-amd64-minimal-20220123T170538Z.iso stage3-amd64-desktop-systemd-20220116T170534Z.tar.xz ``` 配置信息 ``` CPU: Intel i5-8300H (8) @ 4.000GHz GPU: NVIDIA GeForce GTX 1060 Mobile 6GB Host: OMEN by HP Laptop 15-dc0xxx ``` 安装目标 amd64/UEFI/systemd/gnome/gentoo 达到日常使用目的
|
设计模式 前端开发 Java
MVC到底是设计模式还是软件架构?
MVC到底是设计模式还是软件架构?
524 0
|
前端开发 JavaScript
案例挑战——MVVM框架理解和实践
案例挑战——MVVM框架理解和实践
526 0
|
Python
Python通过http.server构建web服务器
Python通过http.server构建web服务器
486 0