0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何给一个变量设置一个别名?

strongerHuang 来源:IOT物联网小镇 作者:IOT物联网小镇 2022-06-06 09:33 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

别名是啥玩意?

stackoverflow上看到一个有趣的话题:如何给一个变量设置一个别名?(How to assign to a variable an alias?

4eaada1e-e530-11ec-ba43-dac502259ad0.png

所谓的变量别名,就是通过通过不同的标识符,来表示同一个变量。

我们知道,变量名称是给程序员使用的。

编译器的眼中,所有的变量都变成了地址

请注意:这里所讨论的别名,仅仅是通过不同的标识符来引用同一个变量。

与强符号、弱符号的概念没有任何关系,那是另一个话题。

在上面这个帖子中,作者首先想到的是通过宏定义,对变量进行重新命名。

这样的做法,将会在编译之前的预处理环节,把宏标识符替换为变量标识符。

在网友回复的答案中,大部分都是通过指针来实现:让不同的标识符指向同一个变量。

不管怎么说,这也算是一种别名了。

但是,这些答案有一个局限:这些代码必须一起进行编译才可以,否则就可能出现无法找到符号的错误信息。

现在非常流行插件编程,如果开发者想在插件中通过一个变量别名来引用主程序中的变量,这该如何处理呢?

本文提供两个方法来实现这个目的,并通过两个简单的示例代码来进行演示。

文末有示例代码的下载地址。

方法1:反向注册

之前我接触过一些CodeSys的代码,里面的代码质量真的是非常的高,特别是软件架构设计部分。

传说:CodySys 是工控界的 Android

其中有个反向注册的想法,正好可以用在变量别名上面。

示例代码中一共有 2 个文件:main.cplugin.c

main.c中定义了一个全局变量数组,编译成可执行程序main

plugin.c中通过一个别名来使用main.c中的全局变量。

plugin.c被编译成一个动态链接库,被可执行程序main动态加载(dlopen)。

plugin.c中,提供一个函数func_init,当动态库被main dlopen之后,这个函数就被调用,并且把真正的全局变量的地址通过参数传入

这样的话,在插件中就可以通过一个别名来使用真正的变量了(比如:修改变量的值)。

本质上,这仍然是通过指针来进行引用。

只不过利用动态注册的思想,把指针与变量的绑定关系在时间和空间上进行隔离。

plugin.c 源文件

#include 

int *alias_data = NULL;

void func_init(int *data)
{
printf("libplugin.so: func_init is called. 
");
alias_data = data;
}

void func_stage1(void)
{
printf("libplugin.so: func_stage1 is called. 
");
if (alias_data)
{
alias_data[0] = 100;
alias_data[1] = 200;
}
}

main.c 源文件

#include 
#include 
#include 

// defined in libplugin.so
typedef void (*pfunc_init)(int *);
typedef void (*pfunc_stage1)(void);

int data[100] = { 0 };


void main(void)
{
data[0] = 10;
data[1] = 20;

printf("data[0] = %d 
", data[0]);
printf("data[1] = %d 
", data[1]);

// open libplugin.so
void *handle = dlopen("./libplugin.so", RTLD_NOW);
if (!handle)
{
printf("dlopen failed. 
");
return;
}

// get and call init function in libplugin.so
pfunc_init func_init =  (pfunc_init) dlsym(handle, "func_init");
if (!func_init)
{
printf("get func_init failed. 
");
return;
}
func_init(data);

// get and call routine function in libplugin.so
pfunc_stage1 func_stage1 =  (pfunc_stage1) dlsym(handle, "func_stage1");
if (!func_stage1)
{
printf("get func_stage1 failed. 
");
return;
}
func_stage1();

printf("data[0] = %d 
", data[0]);
printf("data[1] = %d 
", data[1]);

return;
}

编译指令如下:

gcc -m32 -fPIC --shared plugin.c -o libplugin.so
gcc -m32 -o main main.c -ldl

执行结果:

data[0] = 10 
data[1] = 20 
libplugin.so: func_init is called. 
libplugin.so: func_stage1 is called. 
data[0] = 100 
data[1] = 200

可以看一下动态链接库的符号表:

readelf -s libplugin.so | grep data
4ee38580-e530-11ec-ba43-dac502259ad0.png

可以看到alias_data标识符,并且是在本文件中定义的全局变量。

方法2:嵌入汇编代码

在动态加载的插件中使用变量别名,除了上面演示的动态注册的方式,还可以通过嵌入汇编代码来: 设置一个全局标号来实现。

直接上示例代码:

plugin.c源文件

#include 

asm(".Global alias_data");
asm("alias_data = data");

extern int alias_data[];

void func_stage1(void)
{
printf("libplugin.so: func_stage1 is called. 
");

*(alias_data + 0) = 100;
*(alias_data + 1) = 200;
}

main.c源文件

#include 
#include 
#include 

// defined in libplugin.so
typedef void (*pfunc_init)(int *);
typedef void (*pfunc_stage1)(void);

int data[100] = { 0 };


void main(void)
{
data[0] = 10;
data[1] = 20;

printf("data[0] = %d 
", data[0]);
printf("data[1] = %d 
", data[1]);

// open libplugin.so
void *handle = dlopen("./libplugin.so", RTLD_NOW);
if (!handle)
{
printf("dlopen failed. 
");
return;
}

// get and call routine function in libplugin.so
pfunc_stage1 func_stage1 =  (pfunc_stage1) dlsym(handle, "func_stage1");
if (!func_stage1)
{
printf("get func_stage1 failed. 
");
return;
}
func_stage1();

printf("data[0] = %d 
", data[0]);
printf("data[1] = %d 
", data[1]);

return;
}

编译指令:

gcc -m32 -fPIC --shared plugin.c -o libplugin.so
gcc -m32 -rdynamic -o main main.c -ldl

执行结果:

data[0] = 10 
data[1] = 20 
libplugin.so: func_stage1 is called. 
data[0] = 100 
data[1] = 200

也来看一下libplugin.so中的符号信息:

readelf -s libplugin.so | grep data
4f4798ea-e530-11ec-ba43-dac502259ad0.png

小结

这篇文档通过两个示例代码,讨论了如何在插件中(动态链接库),通过别名来访问真正的变量。

不知道您会不会有这样的疑问:直接使用extern来声明一下外部定义的变量不就可以了,何必这么麻烦?

道理是没错!

但是,在一些比较特殊的领域或场景中(比如一些二次开发中),这样的需求是的确存在的,而且是强需求。

审核编辑 :李倩


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    4423

    浏览量

    68042
  • 代码
    +关注

    关注

    30

    文章

    4983

    浏览量

    74537
  • 变量
    +关注

    关注

    0

    文章

    616

    浏览量

    29619

原文标题:如何给全局变量起一个别名?

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    如何变频器恢复出厂设置

    变频器恢复出厂设置(也叫参数初始化)没有通用的固定步骤,因为不同品牌、不同型号的变频器,操作方法和参数代码都不同。但核心逻辑是相通的,都是通过修改特定的功能码来完成。 你可以根据
    的头像 发表于 03-21 15:41 1651次阅读
    如何<b class='flag-5'>给</b>变频器恢复出厂<b class='flag-5'>设置</b>

    小艺开放平台鸿蒙智能体开发智能体配置-基础信息

    描述,描述会直接显示在智能体的详情页内展示用户查看。 智能体别名:用于智能体分发,提高分发准确率,支持配置多个别名。 本文参考鸿蒙官方文档
    发表于 02-02 16:47

    传递无符号的 long 型指针函数

    C语言允许传递指针函数,只需要简单地声明函数参数为指针类型即可。 下面的实例中,我们传递无符号的 long 型指针函数,并在函数内改变这个值: 实例 #include #i
    发表于 01-27 11:51

    使用博图(TIA Portal)监控PROFINET从站通讯状态的两方法

    。在数据块中定义array of bool类型的变量(如state),长度根据从站数量确定(通常为1024位)。 调用DeviceStates指令 在OB1或其他循环组织块中,调用
    的头像 发表于 01-17 11:21 1657次阅读
    使用博图(TIA Portal)监控PROFINET从站通讯状态的两<b class='flag-5'>个</b>方法

    智能显示模块怎么在显示工程中寄存器设置初始值?我想变量上电的默认值该如何设置

    智能显示模块怎么在显示工程中寄存器设置初始值?我想变量上电的默认值该如何
    发表于 12-11 09:54

    typedef结构体使用

    ) { DATE* date; ... } 甚至还可以顺便给它的指针也定义个别名: typedef struct date { int year; int month; int day; } DATE, *PDATE;
    发表于 12-08 07:04

    智能显示模块怎么在显示工程中寄存器设置初始值?我想变量上电的默认值该如何设置

    智能显示模块怎么在显示工程中寄存器设置初始值?我想变量上电的默认值该如何
    发表于 12-06 10:20

    常用变量的介绍

    寄存器变量般较短的变量类型适合定义为寄存器变量,如short、char等; auto:c语言变量缺省存储类型就是auto typedef
    发表于 11-21 07:05

    Java同仁单点的AI&quot;开胃菜&quot;--搭建自己的本地问答系统

    这是我参与创作者计划的第1篇文章 大家好,因为对AI大模型很感兴趣,相信很多兄弟们跟我样,所以最近花时间了解了些,有些总结 分享大家,希望对各位有所帮助; 本文主要是目标是 讲
    的头像 发表于 11-03 17:16 801次阅读
    <b class='flag-5'>给</b>Java同仁单点的AI&quot;开胃菜&quot;--搭建<b class='flag-5'>一</b><b class='flag-5'>个</b>自己的本地问答系统

    PLC中Static和Temp变量的区别

    大家好,收到粉丝投稿,让博主讲下Static变量和Temp变量的区别,新入行的兄弟可能会对这两概念不太能理解。
    的头像 发表于 09-24 14:51 1742次阅读
    PLC中Static和Temp<b class='flag-5'>变量</b>的区别

    rt-thread studio怎么设置才能让char变量是真正的有符号类型?

    最近被bug折磨了3天,路debug进来,发现最终的原因是,定义了char变量,但实际
    发表于 09-16 08:23

    【原创】TDMS设置一个写入位置函数的摸索

    开时,必须启用缓冲 3、必须先调用“tdms设置一个写入位置”再调用高级tdms写入函数 4、写入函数完成后,必须使用高级tdms关闭,利用刷新是无用的 5、高级tdms打开函数,生成的tdms引用放入全局变量,然后在其他vi
    发表于 08-11 20:54

    AI玩具或成为下一个万亿新赛道

    如果你将拥有家庭新成员,你首先会想到什么?是孩子还是宠物?如果我说你下一个家庭成员,或许是会“察言观色”的AI玩具,这件事是不是听上
    的头像 发表于 07-29 10:15 1325次阅读

    如何打造属于自己的手势识别应用

    期小编大家介绍了和MediaPipe的相遇之路,本期小编将带着大家起来动手,如何打造属于自己的手势识别应用。
    的头像 发表于 07-29 10:12 1344次阅读
    如何打造<b class='flag-5'>一</b><b class='flag-5'>个</b>属于自己的手势识别应用

    Keysight是德示波器的5触发设置与波形分析方法

    Keysight是德示波器在电子测量领域应用广泛,精准的触发设置与高效的波形分析方法对获取准确信号信息至关重要。以下为您介绍5常用的触发设置及波形分析方法。   边沿触发设置  边沿
    的头像 发表于 06-27 16:00 1802次阅读
    Keysight是德示波器的5<b class='flag-5'>个</b>触发<b class='flag-5'>设置</b>与波形分析方法