如何编写跨平台的C语言代码?

嵌入式大杂烩 2023-06-09 22:03

弱符号

弱符号是指在定义或者声明一个对象(变量、结构体成员、函数)时,在对象的前面添加 __attribute__((weak)) 标志所得到的对象符号。如下所示函数即为一个弱对象符号 void test_weak_attr(void),或者称该函数是弱函数属性的、虚函数。

__attribute__((weak)) void test_weak_attr(void)// 或者使用如下样式的定义,两者等效void __attribute__((weak)) test_weak_attr(void){    printf("Weak Func!\r\n");}

弱符号的作用与示例

弱符号是相对于强符号而言的,在定义或者声明变量、函数时,未添加 __attribute__((weak)) 标识的就默认为强符号。如下,最普通的函数定义,就是定义了一个强符号 void test_strong_ref(void):

void test_weak_attr(void){    printf("this is a strong func\r\n");}


驱动程序往往需要考虑兼容性,因为要兼任很多厂商的不同型号的设备。若驱动程序中使用强符号定义一些与适配的设备的特性相关的功能,则下次适配其他设备时,该强符号函数可能需要被修改,以兼容新的设备。当适配的设备很多时,频繁地更改驱动代码将破坏驱动的可维护性。

弱符号的出现可以很好地解决该问题。弱符号的对象具有可以被重定义的功能(即可以被重载)。下面通过测试说明弱符号这种可被重载的特性。

在 test_weak_attr.c 程序中定义如下弱函数:

// test_weak_attr.c#include  __attribute__((weak)) void test_weak_attr(void){    printf("this is a weak func\r\n");}


在 main.c 中定义如下程序:

// main.cvoid test_weak_attr(void){    printf("this is a strong func\r\n");} void app_main(void){    printf("init done\r\n");         test_weak_attr();}


编译运行该 main.c 程序,得到的结果是什么样子的呢?

1

this is a strong func

将 main.c 中的 void test_weak_attr(void) 函数注释掉,再重新编译运行程序得到的结果是:

1

this is a weak func

小结:在使用弱符号函数时,我们可以重新定义一个同名的强符号函数来替代它;若没有重新定义一个强函数来替换它,就使用弱函数的实现。弱函数就好像是一个可以被替换的“默认函数”。

值得一提的是,旧版本的编译器还可以使用如下方式的定义(仅声明无效)将一个对象定义为一个弱对象:

1

2

3

4

__weak void f(void)

{

//code

}

在 linux 的一些代码中,__weak 其实就是通过 __attribute__((weak))的重命名,两者等效。

弱引用

弱引用是在声明一个对象时,通过__attribute__ ((weakref()) 定义一个符号的引用关系。如下所示即定义 test_weakref() 函数弱引用 test_weak_ref() 函数。

1

static void test_weakref(void) __attribute__ ((weakref("test_weak_ref")));

弱引用是相对于强引用而言的。未通过 __attribute__ ((weakref()) 的符号和实现代码之间的关系是强引用。如下即为一个强引用函数。它直接给出了 函数 test_strong_ref(void) 的实现。

1

2

3

4

static void test_strong_ref(void)

{

    printf("this is a strong ref\r\n");

}

在编译程序的时候,我们可以直接使用 test_strong_ref(void) 而不必担心编译不通过。如果,我没有时间去实现 test_strong_ref(void) ,还想在程序里先使用该函数那该如何呢?(是的,就是想白嫖,不想实现,还想先在程序里使用这个函数)。

这个时候弱引用就派上用场了。可以先将该函数定义为弱引用插入到代码中,待后期有时间再慢慢优化代码实现这个函数完整的功能。下面结合测试进行说明。

测试代码1:


static void test_weakref(void) __attribute__ ((weakref("test_weak_ref")));void app_main(void){    printf("init done\r\n");    if (test_weakref) {        test_weakref();    } else {        printf("There is no weakref\r\n");    }}


测试结果:

There is no weakref

测试代码2:

void test_weak_ref(void){    printf("this is a weak ref\n");}static void test_weakref(void) __attribute__ ((weakref("test_weak_ref")));void app_main(void){    printf("init done\r\n");    if (test_weakref) {        test_weakref();    } else {        printf("There is no weakref\r\n");    }}


测试结果:

this is a weak ref

小结:强引用,在未定义该强引用的实现时,编译会报错误:未定义的引用。弱引用允许定义一个未实现(未实例化)的对象,这在编译的时候会将该对象处理成 NULL,编译器并不会报错。通过使用弱引用可以实现后期优化代码的功能。而避免改动使用该函数的地方。使用弱函数可以实现类似“钩子(hook)"函数的功能。

实际上,包括C、python、go 编程语言在内的很多语言 都有类似用法,本篇文章叙述的方法同样适用于这些语言的相关开发。

注意:弱引用仅在静态编译中有效,动态链接中可能无效。

总结

弱符号、弱引用都是增强程序的可维护性的方法。弱符号通过可以被重定义的特性,实现可以被替换实现。弱引用通过可以暂时使用一个未定义的函数的功能,实现允许后期再实现该函数具体功能,而不必担心编译不通过。

为了方便理解,我们先预设一个应用场景:

我们编写了一个模拟IIC的驱动,希望它能够在不同的的平台运行,目标的平台就设为 stm32 标准库,stm32 HAL 库,stm32 LL 库,和 RT-Thread Driver 驱动库。

或许读者有疑惑,为什么同样是 stm32 ,却分成三个平台呢?这是因为从跨平台软件编写者的角度看,只要调用的库的 API 不一致,就和换一个不同的平台没有什么本质的差别,如果在代码中写死了 API 的调用,即使是同一个平台,仍然像多平台一样不能运行。

由此可以看出,跨平台的困难所在,也不是由硬件平台所导致的,而是由代码所依赖的 API 的不同导致的。同一个平台,如果依赖的 API 不同,代码就不能跨平台,同样地,不同的平台,如果依赖的 API 相同,也可以跨平台。

所以归根结底,是代码所依赖的 API 出现了不同,所以下文中所说的“平台”,实际上对应的是一套 API 。

我们继续说这个模拟 IIC 的驱动,模拟 IIC 驱动是使用 GPIO 的反转来模拟 IIC 协议,所以依赖了平台的 GPIO API,如果把调用 GPIO 的部分写死,那么换一个平台,就肯定不能在多个平台上运行。

下面我们开始讨论在多平台运行的解决方案。

我们先从最朴素简单的解决方案开始,然后逐步迭代到弱函数的方案,这样有两方面好处:

一是从简单的方案开始,循序渐进地介绍,可以降低阅读门槛。
二是可以带读者亲历一遍方案的演进过程,以及演进动因。

和给直接给结果相比,注重过程和动因更能够还原技术决策的真实过程。因为任何技术都是从简单朴素逐步演进而来的,如果直接给出最后的结果,会产生理解的断层,即使记住了几种技术的优劣,在新的场景中,面对更加多样化的实际问题也会难免乏力。

先从最朴素的方案讲起。

方案一、手动控制添加编译的 .c 文件

朴素的方案有很多,比如就是多搞几个版本的 .c 文件,比如SIMU_IIC_STM32_HAL.c ,SIMU_IIC_STM32_LL.c, SIMU_IIC_RTT.c 需要哪个就添加哪个进去编译不就完了嘛!

这种朴素的方案虽然看起来简单,但是,这几个文件中包含有共用的逻辑,例如模拟 IIC 的协议的实现,如何将 8bit 的数据依次发送,等等。

这些共用的逻辑,相当于在每个文件中都复制了一份,一旦修改到共用的逻辑,就要手动同步每个文件,这会导致代码冗余和维护难度的急剧增加,很容易出现人为失误。如果需要添加更多的平台支持,就需要再次复制修改代码。

另一个问题是,使用不同的编译工具链,添加编译文件的方式并不一样,例如,Visual Studio 和 MDK keil 通常是手动添加,而 CMake 通常直接添加目录或者通过文件后缀进行搜索。用户在使用不同的编译工具时,需要针对编译工具来分别处理,这也增加了维护的成本。

因此,我们需要寻找更加优雅的解决方案。

方案一中最核心的问题,是没有分离共用的逻辑,和各个平台的适配接口。
在通常的命名惯例中,共用的逻辑称为 Common,而各个平台的适配接口称为 Port。

在接下来的方案中,我们就会引入 Common 和 Port 分离的设计思想,Common 和 Port 的分离,使得对共用逻辑的修改和增强直接 “分发” 到了各个的 Port 中,而增添新的平台,不需要对 Common 做任何修改。

方案二、条件编译

一种更加优雅的解决方案是使用条件编译。条件编译是一种编译时根据条件选择编译代码的技术,可以通过编译器提供的宏定义和预处理指令来实现。

在我们的模拟 IIC 驱动中,可以直接编写 Common 部分,然后 Common 部分通过条件编译,可以根据平台的不同选择不同的 GPIO Port API。

例如,在 STM32 标准库中,可以使用 GPIO_SetPinMode 和 GPIO_WritePin 接口来模拟 IIC 协议,而在 STM32 HAL 库中,可以使用 HAL_GPIO_WritePin 和 HAL_GPIO_ReadPin 接口来模拟 IIC 协议。因此,在代码中可以使用如下的条件编译方式:

#if defined (USE_STM32_STD_LIB)    /* STM32 Standard Peripheral Library */    GPIO_SetPinMode(SDA_PORT, SDA_PIN, GPIO_MODE_OUTPUT_PP);    GPIO_SetPinMode(SCL_PORT, SCL_PIN, GPIO_MODE_OUTPUT_PP);    ...#elif defined (USE_STM32_HAL_LIB)    /* STM32 HAL Library */    HAL_GPIO_WritePin(SDA_PORT, SDA_PIN, GPIO_PIN_SET);    HAL_GPIO_WritePin(SCL_PORT, SCL_PIN, GPIO_PIN_SET);    ...#elif defined (USE_STM32_LL_LIB)    /* STM32 LL Library */    LL_GPIO_SetOutputPin(SDA_PORT, SDA_PIN);    LL_GPIO_SetOutputPin(SCL_PORT, SCL_PIN);    ...#elif defined (USE_RTT_DRIVER)    /* RT-Thread Driver */    rt_pin_mode(SDA_PIN, PIN_MODE_OUTPUT);    rt_pin_mode(SCL_PIN, PIN_MODE_OUTPUT);    ...#endif


这样,在编译代码时,我们可以通过宏定义来选择编译使用哪个平台的代码,

从而实现跨平台运行。

然而,这种条件编译方式还是有一些问题,如果我们需要添加新的平台支持,就需要添加新的宏定义和条件编译,而且需要修改模块的源码。

方案三 函数指针

我们可以进一步分离 Common 和 Port,将其放在不同的 .c 文件中,Common 通过函数调用的方式来访问 Port 提供的接口,这样可以更加灵活和方便地添加新的平台支持。

具体实现可以通过在 Common 中定义一些函数指针类型来实现。

例如,我们可以定义一个名为 IICOps 的结构体,其中包含了一些指向函数的指针,这些函数实现了具体的 IIC 操作。

在 Port 中,我们实现这些函数,并将其指针传递给 Common 中的 IICOps 结构体。

这样,在 Common 中就可以通过调用这些函数指针来访问 Port 提供的接口了。

这种方案的好处是,添加新的平台支持时,只需要实现相应的 Port 函数,并将其指针传递给 Common 中的结构体即可,不需要修改 Common 的源码。

同时,由于 Common 和 Port 分离,不同平台的适配代码可以相互独立,修改一方不会影响到另一方,从而减少了代码冗余和维护难度。

但是,使用函数指针有两个主要的缺点,一是函数指针本身需要占据内存,增加了内存的开销,二是函数指针需要在初始化时进行加载。

具体来说,函数指针的内存开销相对于代码本身并不大,通常可以忽略不计,但在嵌入式系统中,资源有限,内存开销需要更加注意。

而函数指针的初始化需要在程序启动时进行,这也会对程序的启动时间产生一定的影响。此外,函数指针的使用可能会导致一定的运行时开销,需要在程序性能和资源利用方面做出权衡。

另外,函数指针的使用需要程序员有一定的技术水平和经验,需要合理地进行函数指针的定义、传递和调用等操作,避免出现潜在的错误和安全问题。

总的来说,函数指针是一种比较灵活和方便的方式,可以帮助我们实现代码的跨平台支持,但需要在实际应用中仔细考虑其使用场景和影响,做出合理的决策。

方案四、Common 中声明,Prot 中实现

在这种方案中,我们仍然将 Common 和 Port 分离,但是我们不再使用函数指针来访问 Port 中的接口,而是将其定义为 extern 声明,由 Port 来实现具体的函数。

具体实现可以通过在 Common 中定义一些 extern 声明的函数,这些函数实现了具体的 IIC 操作,但是并不在 Common 中实现具体的代码逻辑,而是在 Port 中实现。

在 Port 中,我们实现这些函数,并将其声明为 extern,然后在编译时链接到 Common 中。

这样,在 Common 中就可以通过调用这些函数来访问 Port 提供的接口了。

这种方案的好处是,添加新的平台支持时,只需要实现相应的 Port 函数,并在编译时链接到 Common 中即可,不需要修改 Common 的源码。

这样 Port 的实现函数的挂载就提前到了编译阶段,避免了运行时挂载函数指针的复杂性和容易出错问题。以及避免了函数指针的内存占用。

但是,和 IIC 的例子不同的是,在一些更实际的项目中,随着软件复杂度的提升, Common 中使用的 Port 函数数量会快速膨胀,这时,每个 Port 函数都必须要实现,即使这个功能非常的冷门,这样 Common 中每增加一个 Port 的依赖,都要求所有的 Port 进行及时的跟进,否则整个项目都无法编译通过。

接下来,就要有请 weak (弱函数)机制出马了

但方案四的缺点在于,在一些更实际的项目中,随着软件复杂度的提升,Common 中使用的 Port 函数数量会快速膨胀,这时,每个 Port 函数都必须要实现,即使这个功能非常的冷门,这样 Common 中每增加一个 Port 的依赖,都要求所有的 Port 进行及时的跟进,否则整个项目都无法编译通过。

方案五 弱函数

为了解决这个问题,可以使用 weak (弱函数)机制,将所有的 Port 函数都定义为 weak 函数。

weak 函数是一种特殊的函数类型,带 weak 的函数和不带 weak 的函数可以同时存在,如果有不带 weak 的函数,就会优先链接不带 weak 的实现,如果没有找到不带 weak 的实现函数,就会使用 weak 函数作为默认的实现。

即,在使用弱函数时,如果找到多个实现,链接器会选择优先级最高的实现。

在 C 语言中,可以使用 attribute((weak)) 来声明一个函数为弱函数。例如:

attribute((weak)) void port_func(){    // 默认实现}attribute((weak)) void port_func()


而在 Port 中,只需要实现需要的函数即可,如果某些函数不需要实现,可以不用管它,因为在链接时会使用 Common 中的默认实现。

这样在编译时如果没有找到相应的实现函数,就会使用默认的实现,而不会导致编译错误。

而在 Port 中,只需要实现需要的函数即可,如果某些函数不需要实现,可以不用管它,因为在编译时会使用 Common 中的默认实现。

这样,在添加新的平台支持时,只需要实现需要的函数,而不用实现所有的函数,大大简化了开发的难度和工作量。同时,也避免了函数指针的内存占用和运行时挂载函数指针的复杂性和容易出错问题。

弱函数的多编译器支持

在不同的平台上, weak 的声明方法也会有所不同,因此需要自己定义一个 weak 声明,例如 MY_WEAK,来支持不同的平台:

/* Compiler */#if defined(__ARMCC_VERSION) && (__ARMCC_VERSION >= 5000000) /* ARM Compiler \                                                              */#define MY_WEAK __attribute__((weak))#elif defined(__IAR_SYSTEMS_ICC__) /* for IAR Compiler */#define MY_WEAK __weak#elif defined(__MINGW32__) /* MINGW32 Compiler */#define MY_WEAK#elif defined(__GNUC__) /* GNU GCC Compiler */#define MY_WEAK __attribute__((weak))#endif/* default MY_WEAK */#ifndef MY_WEAK#define MY_WEAK#endif


可以看到,在不同的编译器下,weak 有不同的写法,上面的这些定义包含了对 armcc5、 armclang、IAR、GCC 的支持。

然而,弱函数的方案在一些平台有一些明显的缺陷,例如,MSVC编译器是微软公司开发的C/C++编译器,在Windows操作系统下被广泛使用。与GCC和Clang等主流编译器相比,MSVC对于弱函数的支持不太完善。

MSVC 中,可以通过使用#pragma weak来声明弱函数,但是这个特性只能在 x86 和 x64 平台下使用,而在 ARM 平台下是不支持的。

此外,在一些版本的MSVC编译器中,#pragma weak 的功能也存在一些限制和bug,所以一般在 MSVC 中直接取消 weak 还会更实际一些。

用弱函数对 Port 进行分类

weak 的引入使得我们的 Port 可以根据实际的需求进行划分,而不是一股脑地必须实现所有的 Port 函数。

引入了 weak 之后,我们就有条件将 Port 划分为以下的几种:

1.核心且无默认实现的 Port

这属于必须实现的 Port,缺乏这个 Port,模块的核心功能就运行不起来。例如模拟 IIC中对 IO 的操作 Port 函数,这种 Port 用不用 weak 的区别不大,属于最硬的骨头,在设计软件时应当注意尽可能地减少这种 Port。

在设计软件时,可以直接取消这类 Port 的弱定义,让编译器在编译时就抛出错误。既然缺少了这类 Port 系统的核心功能就无法工作,那么编译通过了也没有什么意义。

2. 核心且有默认实现的 Port

这类 Port 可以直接定义一个默认实现,在大多数情况下,用户就可以不用管这个 Port 了,而在有定制需求的场合下,又可以灵活地定制。

例如弱定义一个 port_printf 用来支持跨平台软件的打印输出,默认是直接使用平台的 vprintf,对于大多数的用户来说,只需要打印到平台自带的 printf 即可,因此对于大多数用户来说,这个 Port 不用实现,就能正常使用系统了。

attribute((weak)) void port_printf(char* fmt, ...) {  va_list args;  va_start(args, fmt);  vprintf(fmt, args);  va_end(args);}


而有定制需求的用户可以通过自己重写 port_printf() 来打印到其他的地方(比如输出到 log,或者输出到其他串口)。

在 printf 的例子中,我们默认了 printf 是所有的平台都提供了的,对 printf 进行这种假设是合理的,因为它是 libc 的标准函数。

然而,有些 Port 虽然有默认实现,却不能支持所有的平台,例如线程操作的 Port,在常见的平台中,如 linux、RT-Thread、FreeRTOS ,我们知道如何写默认实现,但是这些实现又不通用,这时我们可以在默认实现中结合条件编译,为常见的平台提供默认实现,例如:

attribute((weak)) void port_thread_start(port_thread_t* thread) {#ifdef __linux    pthread_mutex_lock(&(thread->mutex));    pthread_cond_signal(&(thread->cond));    pthread_mutex_unlock(&(thread->mutex));#elif USE_FREERTOS    vTaskResume(thread->thread);#else    #error "port_thread_start() 需要用户实现"#endif}


这个例子中为 linux 和 FreeRTOS 提供了默认的线程启动 Port 的实现,使用 linux 或者 FreeRTOS 的用户可以通过条件编译来直接使用默认实现。

而既不用 linux 也不用 FreeRTOS 的用户,则会在编译时遇到 #error,这提示他们要自己实现 port_thread_start()。

这种写法还有一种好处,就是在不支持 weak 的平台,例如 MSVC,就可以通过 _WIN32 条件编译来进行跨平台支持。

3. 边缘但无默认实现的 Port

还有一类 Port,它们比较冷门,只有部分用户会使用到,但是又难以提供默认的实现。例如用 port_reboot() 来重启硬件,每个硬件平台重启硬件的 API 都是不同的,我们无法提供一个默认的实现。

但是,没有这个 Port,也不影响系统的核心功能,只是在某些时候(例如开启了超时自动重启功能),又有这个 Port才行,这样的 Port 就属于是边缘但无默认实现的 Port。

这时,就可以选择将 Port 缺失的错误延后到运行时,具体在操作时,就可以编写一个 weak 的实现,而这个实现中抛出一个运行时错误,例如:

attribute((weak)) void port_reboot(void){    printf("Error: port_reboot() 需要用户实现\r\n");    while(1);}


这样,只要不用到这个 Port,都可以编译通过且顺利运行,只有实际用到时,才会在运行时报错。

weak 在 GCC 链接静态库时的问题

这里我想特别提示一种常见的 weak 失效的问题,这种问题目前我只发现在 gcc 链接静态库时包含 weak 会出现。

gcc 在链接静态库时,默认的行为是只要找到第一个(不管是不是弱符号),就会将其链接,然后停止继续寻找,这样一来,如果你的 weak 是被第一个找到的,那么强定义的函数就失效了。

这个问题有多种解决方案,我这里只提示一种,有更好的方案可以进qq交流群:577623681 大家一起讨论。

解决方案:使用 "-Wl,--whole-archive" 选项来解决。当使用这个选项时,链接器将整个库文件都包含在链接输出文件中,而不考虑这些库文件是否实际上被使用了。这样就可以保证弱符号在整个库中得到了正确的链接,并且在可执行文件或其他库中保持有效。

需要注意的是,当使用 "-Wl,--whole-archive" 选项时,可能会将一些不必要的库文件链接到最终的可执行文件或库中,这可能会增加最终文件的大小。因此,应该仅在必要时使用这个选项。

本文来源网络,免费传达知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除。

注意

由于微信公众号近期改变了推送规则,为了防止找不到,可以星标置顶,这样每次推送的文章才会出现在您的订阅列表里。


猜你喜欢:

谈谈嵌入式软件的兼容性!

实用 | 分享几个非常实用的开源项目

多进程编程知识汇总,附代码例子!

柔性数组在实际项目中的应用?

干货 | protobuf-c之嵌入式平台使用

C语言、嵌入式重点知识:回调函数

实用 | 10分钟教你搭建一个嵌入式web服务器

在公众号聊天界面回复1024,可获取嵌入式资源


嵌入式大杂烩 专注于嵌入式技术,包括但不限于C/C++、嵌入式、物联网、Linux等编程学习笔记,同时,内包含大量的学习资源。欢迎关注,一同交流学习,共同进步!
评论
  • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
    hai.qin_651820742 2025-01-07 14:52 102浏览
  • 故障现象一辆2017款东风风神AX7车,搭载DFMA14T发动机,累计行驶里程约为13.7万km。该车冷起动后怠速运转正常,热机后怠速运转不稳,组合仪表上的发动机转速表指针上下轻微抖动。 故障诊断 用故障检测仪检测,发动机控制单元中无故障代码存储;读取发动机数据流,发现进气歧管绝对压力波动明显,有时能达到69 kPa,明显偏高,推断可能的原因有:进气系统漏气;进气歧管绝对压力传感器信号失真;发动机机械故障。首先从节气门处打烟雾,没有发现进气管周围有漏气的地方;接着拔下进气管上的两个真空
    虹科Pico汽车示波器 2025-01-08 16:51 56浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 155浏览
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 214浏览
  •  在全球能源结构加速向清洁、可再生方向转型的今天,风力发电作为一种绿色能源,已成为各国新能源发展的重要组成部分。然而,风力发电系统在复杂的环境中长时间运行,对系统的安全性、稳定性和抗干扰能力提出了极高要求。光耦(光电耦合器)作为一种电气隔离与信号传输器件,凭借其优秀的隔离保护性能和信号传输能力,已成为风力发电系统中不可或缺的关键组件。 风力发电系统对隔离与控制的需求风力发电系统中,包括发电机、变流器、变压器和控制系统等多个部分,通常工作在高压、大功率的环境中。光耦在这里扮演了
    晶台光耦 2025-01-08 16:03 48浏览
  • 「他明明跟我同梯进来,为什么就是升得比我快?」许多人都有这样的疑问:明明就战绩也不比隔壁同事差,升迁之路却比别人苦。其实,之间的差异就在于「领导力」。並非必须当管理者才需要「领导力」,而是散发领导力特质的人,才更容易被晓明。许多领导力和特质,都可以通过努力和学习获得,因此就算不是天生的领导者,也能成为一个具备领导魅力的人,进而被老板看见,向你伸出升迁的橘子枝。领导力是什么?领导力是一种能力或特质,甚至可以说是一种「影响力」。好的领导者通常具备影响和鼓励他人的能力,并导引他们朝着共同的目标和愿景前
    优思学院 2025-01-08 14:54 55浏览
  • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
    丙丁先生 2025-01-07 09:25 111浏览
  • 本文介绍编译Android13 ROOT权限固件的方法,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。关闭selinux修改此文件("+"号为修改内容)device/rockchip/common/BoardConfig.mkBOARD_BOOT_HEADER_VERSION ?= 2BOARD_MKBOOTIMG_ARGS :=BOARD_PREBUILT_DTB
    Industio_触觉智能 2025-01-08 00:06 87浏览
  • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
    知白 2025-01-07 15:02 140浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
    GIRtina 2025-01-07 11:02 115浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 195浏览
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 160浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦