C语言中的面向切面编程(AOP)

概念

首先给出一段由ChatGPT给出的简短的AOP概念:

AOP是一种编程方法,用来将在程序中多处重复出现的代码(比如日志、权限控制)从主要业务逻辑中抽取出来,提高代码的模块化和可维护性。

抽取后的代码会在原始的业务逻辑代码中特定的位置执行,这些位置由切点(Pointcut)定义。通常会在方法执行前、执行后、抛出异常时等特定点执行抽取出的代码,这些点被称为连接点(Join Point)。

概述

在C语言中,编译器所提供的编译期和执行期的能力相较于java或者其他语言来说会弱一些,这也许就是可能很少听到在C语言中搞面向切面编程的原因之一吧。

从上面的概念上来看,AOP一般是在一些函数(或类方法)执行前后做一些额外处理,例如调用前增加一些权限控制,调用后增加一些日志记录。从这些行为上来说,任何语言其实都可以做到。我们可以简单的在一个函数的开始加一段逻辑或调用某个函数来实现权限验证,在函数返回前调用某个函数添加日志等等。类似如下代码:

arduino 复制代码
void foo(void)
{
  if (!verify_identity())
    return;

  //...
  
  log("end");
}

但很显然,这么做会在程序的很多个函数中添加很多重复的代码(例如本例的verify_identitylog),以至于代码变得比较臃肿。

那么有没有什么办法来瘦身呢?

这就是AOP擅长的领域了。

写在示例之前

C语言编译器没有提供很完整的AOP支持,因此我们需要自行手动实现,或者使用一些现有的库来实现。

本文将使用开源C语言库Melon的函数模板来实现上面的效果。

在Melon提供的函数模板组件中,实现了若干宏函数,这些宏函数都是用来定义不同类型的函数的。这些用宏来定义的函数和我们原生C语言中的函数的区别,简单来说就是,在我们实际要执行的函数逻辑外,再封装一个函数,这个函数会在我们指定的函数逻辑开始前和结束后调用一个回调函数(即函数的入口回调函数出口回调函数)。

基于函数模板的这一特性,Melon中实现了一个span组件,用来度量使用函数模板定义的函数的时间开销。

但如果事情仅限于此,那么这种AOP很显然能做到的事情也基本仅限于此了。

因此,Melon支持了c99,并利用c99提供的宏特性,实现了将函数模板定义的函数的实参以可变参数的形式传递到入口和出口回调函数中。这就意味着,入口和出口回调函数可以访问函数的参数,并对参数的内容作出修改(主要针对指针指向的内存中的数据)。

这样,就给我们在回调函数中提供了更多的可操作空间。我们可以针对不同的函数,修改其参数值,从而来影响后续函数调用中的执行逻辑。例如前面的权限验证,我们可以将其大致简化为如下形式:

arduino 复制代码
void entry_callback(char *file, char *func, int line, ...)
{
  va_list args;
  va_start(args, line);
  int *a = (int *)va_arg(args, int *);
  va_end(args);
  if (!verify_identity())
    *a = 0;
}

void exit_callback(char *file, char *func, int line, ...)
{
  va_list args;
  va_start(args, line);
  int *a = (int *)va_arg(args, int *);
  va_end(args);
  log("%d\n", *a);
}

void foo(int *a)
{
  if (!*a)
    return;

  //...
}

int bar(int *d, int e)
{
  if (!*d)
    return -1;

  //...
  return 0;
}

这里的代码只是一个示意,后面会给出一个实际可用的示例。

我们可以随意增加函数,这些函数都会利用同一对入口和出口函数来实现身份验证。

示例

下面就给出一个可用的使用函数模板实现AOP的C语言代码。

arduino 复制代码
//a.c

#include "mln_func.h"
#include <stdio.h>
#include <string.h>
#include <stdarg.h>

MLN_FUNC_VOID(static, void, foo, (int *a, int b), (a, b), {
    printf("in %s: %d\n", __FUNCTION__, *a);
    *a += b;
})

MLN_FUNC(static, int, bar, (void), (), {
    printf("%s\n", __FUNCTION__);
    return 0;
})

static void my_entry(const char *file, const char *func, int line, ...)
{
    if (strcmp(func, "foo"))
        return;

    va_list args;
    va_start(args, line);
    int *a = (int *)va_arg(args, int *);
    va_end(args);

    printf("entry %s %s %d %d\n", file, func, line, *a);
    ++(*a);
}

static void my_exit(const char *file, const char *func, int line, ...)
{
    if (strcmp(func, "foo"))
        return;

    va_list args;
    va_start(args, line);
    int *a = (int *)va_arg(args, int *);
    va_end(args);

    printf("exit %s %s %d %d\n", file, func, line, *a);
}

int main(void)
{
    int a = 1;

    mln_func_entry_callback_set(my_entry);
    mln_func_exit_callback_set(my_exit);

    foo(&a, 2);
    return bar();
}

这段函数中,我们使用MLN_FUNCMLN_FUNC_VOID来定义了两个函数,即foobar。两个函数的逻辑很简单,就是printf输出当前函数名以及参数值(如果有参数的话)。同时,我们也使用了mln_func_entry_callback_setmln_func_exit_callback_set定义了两个全局回调函数,用来在函数调用开始和结束时调用。

我们可以看到,回调函数中使用strcmp对进入回调的函数做了过滤,仅对foo函数做额外处理。在入口回调中输出函数信息及第一个参数的值,随后修改参数指针指向的内存中的值。在出口回调中输出函数信息和参数值。

我们来编译一下(我们假定这个代码文件名为a.c):

css 复制代码
cc -o a a.c -I /usr/local/melon/include/ -L /usr/local/melon/lib/ -lmelon -std=c99 -DMLN_C99 -DMLN_FUNC_FLAG

这里:

  • /usr/local/melon是Melon库的默认安装路径。

  • -std=c99是启用c99。

  • -DMLN_C99是定义一个名为MLN_C99的宏,这个宏用来启用函数模板组件中C99下才有的特性。

  • -DMLN_FUNC_FLAG用来定义一个名为MLN_FUNC_FLAG的宏,这个宏用来启用函数模板功能。是的,如果没有这个宏,上面的那些使用MLN_FUNC定义的函数就是普通的C语言函数,也不会触发入口和出口回调函数的调用。

执行一下看看效果:

less 复制代码
entry a.c foo 6 1
in __mln_func_foo: 2
exit a.c foo 6 4
__mln_func_bar

可以看到:

  • 入口回调函数中,foo的第一个参数指向的内存中的值为1

  • 进入foo的实际函数逻辑中,printf输出当前的函数名为__mln_func_foo,以及此时看到的第一个参数的值为2,不再是1了,因为在入口回调函数中被修改了。__mln_func_foo这个函数执行的才是我们定义的逻辑,而foo是对__mln_func_foo的一个封装。

  • 出口回调函数中,我们看到第一个参数的值变为了4,因为它在我们给出的函数逻辑中做了修改。

  • 最后输出的是bar的实际执行逻辑所在的函数名,与foo的形式一致。

最后,我们去掉MLN_FUNC_FLAG这个宏再次编译一次:

css 复制代码
cc -o a a.c -I /usr/local/melon/include/ -L /usr/local/melon/lib/ -lmelon -std=c99 -DMLN_C99

然后执行一下看看输出结果:

yaml 复制代码
in foo: 1
bar

可以看得出,此时foobar不再是封装函数,而是我们定义的函数逻辑的函数名,即普通的C语言函数。

读到这里的都是真爱,感谢阅读!

相关推荐
AndyFrank29 分钟前
mac crontab 不能使用问题简记
linux·运维·macos
筱源源1 小时前
Kafka-linux环境部署
linux·kafka
EricWang13581 小时前
[OS] 项目三-2-proc.c: exit(int status)
服务器·c语言·前端
我是谁??1 小时前
C/C++使用AddressSanitizer检测内存错误
c语言·c++
算法与编程之美1 小时前
文件的写入与读取
linux·运维·服务器
希言JY2 小时前
C字符串 | 字符串处理函数 | 使用 | 原理 | 实现
c语言·开发语言
xianwu5432 小时前
反向代理模块
linux·开发语言·网络·git
午言若2 小时前
C语言比较两个字符串是否相同
c语言
Amelio_Ming2 小时前
Permissions 0755 for ‘/etc/ssh/ssh_host_rsa_key‘ are too open.问题解决
linux·运维·ssh
Ven%3 小时前
centos查看硬盘资源使用情况命令大全
linux·运维·centos