框架一种回调函数框架

Posted 李柱明

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了框架一种回调函数框架相关的知识,希望对你有一定的参考价值。


前言

主要记录回调函数的部分用途。

李柱明博客:https://www.cnblogs.com/lizhuming/p/15486882.html

概念

嵌入式编程,一定要理解指针。

本人认为,指针是基础,用指针的思维去进行细节编程,甚至架构搭建。

回调函数就是让 CPU 跳到该函数地址去执行,而该地址由你设置。

回调函数

目的:

  1. 解耦。
  2. 分离、分层。
  3. 精简代码,高效开发。

理念:

可分为创建者使用者

  • 创建者:

    • 创建、实现回调函数。
    • 回调函数的内容是创建者的业务。
    • 内容建议:尽量做消息转发,告知创建者业务主线程去执行实际业务。
  • 使用者:

    • 调用回调函数。
    • 该回调可以拥有多个使用者。
    • 使用者业务需要用到创建者这个业务时,调用回调函数即可。

高级使用 1

建议结合结构体链表去使用。

把回调函数及其它功能数据绑定到结构体中;

若有多个类似的功能就可以把这些功能插入到链表(数组、队列、树都可以)中管理;

例子 1:

  • 若线程 B 发生了事件 b,若需要通知到线程 A,则找出线程 A 的通知回调函数执行即可。若需要通知到某类功能的所有业务,这时就可以轮询链表,执行其绑定的通知回调函数。
  • 参考:((20210727212744-26qed62))

例子 2:

  • 设备与驱动的绑定。在 linux 驱动开发时很常见。
  • 比如我一个设备支持多种通信,包括 i2c、spi。
  • 用户 A 想使用 i2c 通信,那就把封装好的 i2c API 绑定到该设备,再使用该设备提供的统一 API。
  • 用户 A 想使用 spi 通信,那就把封装好的 spi API 绑定到该设备,再使用该设备提供的统一 API。
  • 上面说的统一 API 是该设备暴露到外界共给用户使用的,用户对其无须关系。
  • 比如用户都使用 send(); 发送数据即可,配置时配置好后,底层通信无需关心,直接使用即可。

高级使用 2

独立业务或分层业务之间 IPC 设计。

业务 A:

  • 其业务管理结构体包含两个回调函数:

    • 回调函数 1:(其它业务到本地业务的单向通道)

      • 在业务 A 实现,唯一的。
      • 其作用主要是供给外部业务使用。所以把其调用封装成一个 API。
      • 其内容主要是消息传递,如消息队列、信号量等等,把消息事件转发到业务 A 的业务线程中。
      • 该回调函数可以不写入业务管理结构体,只要提供 API 调用执行该回调即可。
    • 回调函数 2:(本地业务到其它业务的单向通道)

      • 在其它需要调用业务 A 的业务里实现,如业务 B。不唯一,多种多样。
      • 主要供给本地业务使用。
      • 其作用主要是把业务 A 的事件转发到外部业务。
      • 其主要内容是把业务 A 的事件或数据转发给注册该回调函数的业务。

如果按照分层思维,业务 A 可以为底层,供给各个上层使用,各个上层只需要注册各自的回调函数到业务 A 即可(即是建立业务 A 到上层业务的单向通道)。

NS3-对象框架之聚合

NS-3中的回调其实就是C 语言基本的函数指针的封装类。

  • 函数行参
  • 类的成员变量

回调函数是当特定事件或者满足某种条件时(时间超时)被调用,用于对该事件或者条件进行响应,是一种可扩展编程的常用手段。回调的最大好处在于扩展性强。不需要和具体的函数进行绑定,而可以在创建的时候动态决定到底调用那个函数。例如,此时我们不想再调用加法,而想调用乘法,那么可以给A对象实例绑定乘法操作,而无需改变A类的定义。

NS3中使用回调思想来处理各种协议调用或者追踪系统。

NS3中的回调(callback.cc)

NS3中对普通函数、对象的方法、类的静态方法等进行了的封装(Callback对象,其中使用C++的模板来表示对应函数的返回值和参数。其中第一个模板参数表示函数的返回值,其后的模板参数表示函数的参数),在src/core/module/callback.h中实现了回调最多可以添加9个参数。

NS3中的回调
  普通函数的回调|(FunctorCallbackImpl)
    BoundFunctorCallbackImpl
  方法的回调|(MemPtrCallbackImpl)
   BoundMemPtrCallbackImpl
  具体的实现
        MakeCallback()
        MakeBoundCallback()

普通函数的回调(FunctorCallbackImpl)

  • 回调的类型:Callback<对应函数的返回值类型,....>

技术图片

  • MakeCallback()函数来创建一个回调对象
Callback<int, int, int> callback = MakeCallback(&add);
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

#include "ns3/core-module.h"

using namespace ns3;

NS_LOG_COMPONENT_DEFINE ("TryCallback");

class A {
public:
    A(Callback<int, int, int> calculator);

    void calculate();
private:
    Callback<int, int, int> m_calculator;
};

A::A (Callback<int, int, int> calculator)
{
    m_calculator = calculator;
}

void
A::calculate ()
{
    int result = m_calculator(5, 6);
    NS_LOG_UNCOND(result);
}

int add(int a, int b)
{
    return a + b;
}

int
main (int argc, char *argv[])
{
    A one(MakeCallback(&add));
    one.calculate();
}

方法的回调(MemPtrCallbackImpl)

在C++当中,方法被封装在类或者对象当中,因此其使用方法和普通函数是不同的。因此,NS-3为方法回调专门创建了一个类型MemPtrCallbackImpl。这个类型也可以直接通过MakeCallback()函数来创建,和普通函数回调不同的是,创建方法的回调必须传入具体的对象.

  • 定义回调:
Callback<Complex, Complex &> m_complexCalculator;
  • 创建回调对象:
Complex first(1, 3);
Callback<Complex, Complex &> callback = MakeCallback(&Complex::add, &first);

完整程序

/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

#include <iostream>

#include "ns3/core-module.h"

using namespace ns3;

NS_LOG_COMPONENT_DEFINE ("TryCallback");

class Complex {
public:
    Complex(int a, int b);
    Complex add(Complex & another);
    friend std::ostream & operator <<(std::ostream & out, Complex & c);
private:
    int m_a;
    int m_b;
};

Complex::Complex (int a, int b)
{
    m_a = a;
    m_b = b;
}

Complex Complex::add (Complex & another)
{
    int a = m_a + another.m_a;
    int b = m_b + another.m_b;
    return Complex(a, b);
}

std::ostream & operator <<(std::ostream & out, Complex & c) {
    out << c.m_a << " + " << c.m_b << "i";
    return out;
}

class A {
public:
    A(Callback<int, int, int> calculator);
    A(Callback<Complex, Complex &> complexCalculator);

    void calculate();
    void complexCalculate(Complex b);
private:
    Callback<int, int, int> m_calculator;
    Callback<Complex, Complex &> m_complexCalculator;
};

A::A (Callback<int, int, int> calculator)
{
    m_calculator = calculator;
}

A::A (Callback<Complex, Complex &> complexCalculator)
{
    m_complexCalculator = complexCalculator;
}

void
A::complexCalculate (Complex b)
{
    Complex result = m_complexCalculator(b);
    std::cout << result << std::endl;
}

void
A::calculate ()
{
    int result = m_calculator(5, 6);
    NS_LOG_UNCOND(result);
}

int add(int a, int b)
{
    return a + b;
}

int multiply(int a, int b)
{
    return a * b;
}

int
main (int argc, char *argv[])
{
    Complex first(1, 3);
    Complex second(2, 5);

    A one(MakeCallback(&Complex::add, &first));
    one.complexCalculate(second);//3+8i
}

绑定参数的回调(MakeBoundCallback)

NS-3提供了一种绑定参数的回调,可以在创建回调的时候就给回调绑定好部分参数。绑定的参数,只能从第一个参数开始顺序绑定,不能跳过任何参数

对于对象的方法,由于对象本身就具有存储数据的能力,就无需绑定参数的回调这个概念。因此,NS-3当中并未提供绑定参数的对象方法的回调。

/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

#include <iostream>

#include "ns3/core-module.h"

using namespace ns3;

NS_LOG_COMPONENT_DEFINE ("TryCallback");

class A {
public:
    A(Callback<int, int> calculator);

    void calculate();
private:
    Callback<int, int> m_calculator;
};

A::A (Callback<int, int> calculator)
{
    m_calculator = calculator;
}

void
A::calculate ()
{
    int result = m_calculator(6);
    NS_LOG_UNCOND(result);
}

int add(int a, int b)
{
    return a + b;
}

int multiply(int a, int b)
{
    return a * b;
}

int
main (int argc, char *argv[])
{
    A one(MakeBoundCallback(&add, 8));
    one.calculate();
}

判断回调是否绑定函数(Callback::IsNull())

如果不给回调绑定函数就直接调用,程序会发生异常。

  • 为保证程序的健壮性,NS3中提供了方法Callback::IsNull()来判断回调是否已经绑定函数。
  • 已经给回调绑定函数,再使用完成后,想取消绑定,则可以使用NS-3提供的MakeNullCallback()方法,将回调置空。
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

#include <iostream>

#include "ns3/core-module.h"

using namespace ns3;

NS_LOG_COMPONENT_DEFINE ("TryCallback");

class A {
public:
    A();
    A(Callback<int, int, int> calculator);

    void calculate();

    void
    setCalculator (const Callback<int, int, int>& calculator)
    {
        m_calculator = calculator;
    }

private:
    Callback<int, int, int> m_calculator;
};

A::A(){}

A::A (Callback<int, int, int> calculator)
{
    m_calculator = calculator;
}

void
A::calculate ()
{
    if(m_calculator.IsNull()) {
        NS_LOG_UNCOND("callback is not bind to any function.");
    } else {
        int result = m_calculator(6, 8);
        NS_LOG_UNCOND(result);
    }
}

int add(int a, int b)
{
    return a + b;
}

int multiply(int a, int b)
{
    return a * b;
}

int
main (int argc, char *argv[])
{
    A one(MakeCallback(&add));
    one.calculate();
    one.setCalculator(MakeNullCallback<int, int, int>()); //构造空回调
    one.calculate();
}

回调作为属性(CallbackValue)

NS-3专门给回调构造了一个属性值类型CallbackValue,并且实现了其访问器和检查器。

程序演示:

/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */

#include <iostream>

#include "ns3/core-module.h"

using namespace ns3;

NS_LOG_COMPONENT_DEFINE ("TryCallback");

namespace ns3 {

  class A : public Object{
  public:
    static TypeId GetTypeId (void);

    A();

    void calculate();

    void
    setCalculator (const Callback<int, int, int>& calculator)
    {
      m_calculator = calculator;
    }

  private:
    Callback<int, int, int> m_calculator;
  };

  A::A(){}

    TypeId
    A::GetTypeId (void)
    {
    static TypeId tid = TypeId ("ns3::A")
      .AddConstructor<A> ()
      .SetParent<Object> ()
      .AddAttribute ("calculator", "help text",
                     CallbackValue (),
                     MakeCallbackAccessor (&A::m_calculator),
                     MakeCallbackChecker ())
            ;
    return tid;
    }

  void
  A::calculate ()
  {
    if(m_calculator.IsNull()) {
      NS_LOG_UNCOND("callback is bind to any function.");
    } else {
      int result = m_calculator(6, 8);
      NS_LOG_UNCOND(result);
    }
  }

  int add(int a, int b)
  {
    return a + b;
  }

  int multiply(int a, int b)
  {
    return a * b;
  }
}

int
main (int argc, char *argv[])
{
    Ptr<A> a = CreateObject<A>();//使用智能指针创建一个A类的实例
    a->SetAttribute("calculator", CallbackValue(MakeCallback(&add)));//为属性calculator指定一个回调函数add
    a->calculate();
}

例子中将之前A当中的有参构造函数删除,此时向A当中指定回调的任务现在由属性框架来完成。因此A必须继承Object类,并且实现GetTypeId()方法来使用属性框架。在GetTypeId()方法中,创建了一个属性叫做”calculator”,其绑定到了成员变量m_calculator上,属性值类型为CallbackValue。该属性默认回调值为空,使用了变量访问器和默认检查器。

以上是关于框架一种回调函数框架的主要内容,如果未能解决你的问题,请参考以下文章

ajax回调函数无法获取后台传过来的值(SSM框架),大神请指教!

NS3-对象框架之聚合

爬虫日记(99):Twisted的Deferred重新审视

爬虫日记(96):Twisted的通过对象进行回调

如何使用 C++ 成员函数作为 C 框架的回调函数

Scrapy框架: Request回调函数