c++ - UML 类图的 <> vs std::bind()
问题描述
到目前为止,在两个特定的存储库之间,我使用了一个接口类(具有继承),而我最近用一个使用 std::function() 和 std::bind() 的回调函数替换了它。
使用旧的类似接口的方法,我最终得到了这个:
//a.hpp
#include "b.hpp"
class A{
public:
A(InterfaceB* pb) : m_pb(pb) {};
void bar(){m_pb->foo();};
private:
InterfaceB* m_pb;
};
--
//b.hpp
#include <iostream>
class InterfaceB{
public:
virtual void foo() = 0;
};
class B : public InterfaceB {
public:
void foo(){ std::cout << "hi"<< std::endl; };
};
--
//main.cpp
#include "a.hpp"
#include <memory>
int main(){
InterfaceB* pb = new B;
A a(pb);
a.bar();
delete pb;
}
--
在 UML 中,我会像这样绘制上面的小例子:
为了减少 repos(这里是 A 和 B 类)之间的依赖关系,我删除了接口并使用了函数包装器。
//a_callback.hpp
#include <functional>
class A{
public:
void setBcallback(std::function<void(void)> callback){m_callback = callback;};
void bar(){m_callback();};
private:
std::function<void(void)> m_callback;
}
--
//b_callback.hpp
#include <iostream>
class B {
public:
void foo(){ std::cout << "hi"<< std::endl; };
}
--
//main.cpp
#include "a_callback.hpp"
#include <functional>
int main(){
A a;
B b;
a.setBcallback(std::bind(&B::foo, &b));
a.bar();
}
--
这对我来说是个棘手的问题,我没能在 Google 上找到 C++ 的 std::bind()/std::function() 和 UML 的 << bind >> 是如何相互转换的。所以我的问题是,如何在类图中显示函数包装器的使用?根据我的发现,我可能会这样做:
但它只是感觉松散和不足。任何帮助将非常感激!
这个问题之前已被标记为重复:如何在 UML 类图中表示回调。但是我的问题是 C++ 特定的,并且说“原始”被标记为 Java,不幸的是我没有得到任何帮助。我的问题不是我认为它解释的“如何在 UML 中显示回调”,而是更多的“如何在 UML 中显示 std::bind()”,我认为这更棘手。这里发生了两件事,一是使用 bind() 设置函数包装器,二是通过包装器调用。我只是看不到上面的那个线程是如何解决这个特定问题的。谢谢!
解决方案
UML 绑定
依赖是<<bind>>
指UML 模板绑定:
TemplateBinding 是 TemplateableElement 和模板之间的关系,它指定实际 ParameterableElements 对模板的正式 TemplateParameters 的替换。
模板绑定是一种特殊的实现依赖,它表明一个类是一个模板的特化,并规定了模板替换。
一个典型的例子是:
using void_function = std::function<void(void)> ;
C++ 绑定
在 C++std::bind()
中,动态地将参数绑定到可调用对象。 这是完全不同的语义。
如果您想在 UML 中显示这一点,那么它不会有太大帮助。你可以 :
表明 bind 返回的匿名类型是一个可调用对象的模板实例化,其中一个参数替换为 B(与上图非常相似)。
如果有用,说明这个匿名类依赖于 B。
如果有用,请显示从 A 到此匿名类的可选关系(0..1),理解其他匿名类可能存在替代关系(如果您想在图表上说明几个并制作清楚它们是相互排斥的)。
不幸的是,无论你画什么,它都不会像你的 C++ 设计那样通用和强大,而且对理解也没有多大帮助。
推荐:
UML 图的目标不是以图形方式编程,而是提供对内部结构的洞察。所以我强烈建议保持简单:
唯一真正的关系是 A 和回调的抽象可调用类之间。那必须在图表上。
您还可以表明此抽象回调可以依赖于图表中的其他相关类。对这些依赖关系的注释可以用简单的语言解释依赖关系表达了对成员函数的绑定。
推荐阅读
- python - 正则表达式:在时间戳之间获取文本
- python - 当我在 Python 中关闭命令提示符时,如何防止命令提示符关闭?
- mongodb - 如何在mongoDB中推送除重复对象之外的对象
- rust - cargo 是否具有等效的 npm install --save 选项来自动将依赖项添加到 Cargo.toml?
- javascript - 如何使用 Chart JS 在条形图中设置实心标签
- java - 如何使用 akexorcist/Android-Localization 库版本 1.0.8 从 Fragment 更改我的应用程序的语言
- php - 如何使用 PHP 将 USD 更改为 INR
- javascript - 图表 js 加载屏幕
- airflow - 在同一个集群中使用两个不同的端口配置气流?
- c# - 从 JSON 获取子节点