当前位置: 首页 > news >正文

DLL中的inline static成员变量:Windows开发中的常见陷阱

在Windows平台进行C++开发时,DLL(动态链接库)是一个非常重要的概念。它让我们能够实现代码的模块化和动态加载,提高了程序的灵活性和维护性。然而,当我们在DLL中使用C++17引入的inline static成员变量时,可能会遇到一些意想不到的问题。今天我们就来深入探讨这个话题。

在正式开始前,我们先回顾一下C++17引入inline static成员变量的初衷。在C++17之前,类的静态成员变量必须在类外单独定义,这常常导致代码分散,不够优雅。比如:

// header.h
class MyClass {static int value;
};// source.cpp
int MyClass::value = 42;

C++17的inline关键字解决了这个问题,允许我们直接在类定义中初始化静态成员变量:

class MyClass {inline static int value = 42;
};

这看起来很美好,但当我们在DLL环境中使用这个特性时,问题就来了。让我们通过一个具体的例子来说明:

假设我们有一个计数器类,用于在整个程序中统计某个事件的发生次数:

// counter.h
class Counter {
public:inline static int count = 0;static void increment() {count++;}static int get_count() {return count;}
};

现在我们创建两个DLL,都使用这个Counter类:

// dll1.cpp
#include "counter.h"extern "C" __declspec(dllexport) void dll1_count() {Counter::increment();
}// dll2.cpp
#include "counter.h"extern "C" __declspec(dllexport) void dll2_count() {Counter::increment();
}

在主程序中调用这两个DLL的函数:

// main.cpp
int main() {dll1_count();  // 期望count变为1dll2_count();  // 期望count变为2int final_count = Counter::get_count();// 实际上final_count可能仍然是1
}

问题出在哪里?事实上,每个DLL都会获得inline static成员变量的一份独立副本。这就像一个建筑物里每个房间都安装了独立的温度计,而不是共用一个中央温控系统。这显然违背了我们想要一个全局计数器的初衷。

要解决这个问题,我们需要使用DLL导出导入机制:

// counter.h
#ifdef BUILDING_DLL
#define DLL_SPEC __declspec(dllexport)
#else
#define DLL_SPEC __declspec(dllimport)
#endifclass DLL_SPEC Counter {
public:static int count;  // 注意:不能使用inline了static void increment();static int get_count();
};// counter.cpp
int Counter::count = 0;void Counter::increment() {count++;
}int Counter::get_count() {return count;
}

这样改造后,所有DLL和主程序都会共享同一个计数器实例。但代价是我们失去了inline带来的便利,必须在源文件中定义静态成员变量。

这个问题还衍生出了一些相关的注意事项。例如,如果我们在模板类中使用inline static成员变量:

template<typename T>
class TemplateCounter {inline static int count = 0;
};

每个模板实例化都会获得自己的static变量副本,这在DLL环境中会更加复杂。如果不同的DLL实例化了相同的模板参数,它们各自又会得到独立的副本。

在实际开发中,我们需要根据具体场景做出选择:

  1. 如果静态成员变量确实需要在多个DLL间共享,就应该使用导出导入机制,放弃inline。
  2. 如果静态成员变量只在单个DLL内使用,使用inline是安全的。
  3. 对于模板类,需要特别注意实例化的位置和导出导入声明的使用。
40326b99667545a9a424aec4bf7243b9.png

除了技术层面的考虑,这个问题也提醒我们在设计API时要充分考虑DLL边界的影响。有时候,使用其他方式来共享数据可能是更好的选择,比如:

  • 使用进程间通信机制
  • 通过显式的接口传递共享数据
  • 使用集中式的数据管理器

这些替代方案虽然可能需要更多的代码,但能提供更清晰的数据流动和更好的可维护性。

总而言之,inline static成员变量是C++17的一个很好的特性,但在Windows DLL开发中需要谨慎使用。理解其在DLL环境下的行为特点,选择合适的使用方式,对于开发可靠的Windows应用程序至关重要。当我们在享受现代C++带来的便利性的同时,也要时刻注意平台特定的限制和陷阱。

相关文章:

DLL中的inline static成员变量:Windows开发中的常见陷阱

在Windows平台进行C开发时&#xff0c;DLL&#xff08;动态链接库&#xff09;是一个非常重要的概念。它让我们能够实现代码的模块化和动态加载&#xff0c;提高了程序的灵活性和维护性。然而&#xff0c;当我们在DLL中使用C17引入的inline static成员变量时&#xff0c;可能会…...

pandas 读写excel

在Python中&#xff0c;使用Pandas库读写Excel文件是一个常见的操作。Pandas提供了read_excel和to_excel方法来分别实现读取和写入Excel文件的功能。以下是一些基本的示例&#xff1a; ### 读取Excel文件 python import pandas as pd # 读取Excel文件 df pd.read_excel(pat…...

记录Threadlocal使用

编写ThreadLocal工具类 package com.jjking.jplan.context;public class BaseContext<T> {public static final ThreadLocal threadLocal new ThreadLocal();//存储用户public static void set(Object t) {threadLocal.set(t);}//获取用户public static <T> T ge…...

2024 ccpc 辽宁省赛 E(构造 思维?)L(二分+一点点数论知识?)

E 题意&#xff1a; 可以注意到&#xff1a; 我的两种方格都四个方格的大小。 所以 如果存在一种摆放方式 那么 4|nm。 再考虑一种特殊的情况 22 &#xff0c;此时虽然我的积是4 但是无法摆放的。 1>对于 4 | n,或者 4 | m.我直接摆放第二种方格就可以了。 如果我n 是4 的…...

【iOS】设计模式的六大原则

【iOS】设计模式的六大原则 文章目录 【iOS】设计模式的六大原则前言开闭原则——OCP单一职能原则——SRP里氏替换原则——LSP依赖倒置原则——DLP接口隔离原则——ISP迪米特法则——LoD小结 前言 笔者这段时间看了一下有关于设计模式的七大原则&#xff0c;下面代码示例均为OC…...

网络安全:攻防技术-Google Hacking的实现及应用

前言 google hacking其实并算不上什么新东西&#xff0c;在早几年我在一些国外站点上就看见过相关的介绍&#xff0c;但是由于当时并没有重视这种技术&#xff0c;认为最多就只是用来找找未改名的mdb或者别人留下的webshell什么的&#xff0c;并无太大实际用途。但是前段时间仔…...

输入一行字符,分别统计出其中英文字母、空格、数字和其它字符的个数。-多语言

目录 C 语言实现 Python 实现 Java 实现 Js 实现 Ts 实现 题目&#xff1a;输入一行字符&#xff0c;分别统计出其中英文字母、空格、数字和其它字符的个数。 程序分析&#xff1a;利用while语句,条件为输入的字符不为\n。 C 语言实现 #include <stdio.h>int mai…...

2-2-18-9 QNX系统架构之文件系统(三)

阅读前言 本文以QNX系统官方的文档英文原版资料为参考&#xff0c;翻译和逐句校对后&#xff0c;对QNX操作系统的相关概念进行了深度整理&#xff0c;旨在帮助想要了解QNX的读者及开发者可以快速阅读&#xff0c;而不必查看晦涩难懂的英文原文&#xff0c;这些文章将会作为一个…...

各大浏览器(如Chrome、Firefox、Edge、Safari)的对比

浏览器如Chrome、Firefox、Edge等在功能、性能、隐私保护等方面各有特点。以下是对这些浏览器的详细对比&#xff0c;帮助你选择合适的浏览器。 1. Google Chrome 市场份额&#xff1a;Chrome是目前市场上最流行的浏览器&#xff0c;约占全球浏览器市场的65%以上。 性能&#…...

nginx搭建直播推流服务

文章目录 学习链接步骤使用nginx搭建直播推流服务安装依赖库下载nginx-http-flv-module模块下载nginx解压nginx&#xff0c;进入nginx目录设置nginx编译配置编译并安装配置nginx rtmp服务启动nginx 准备另外一台电脑下载OBS下载OBS windows | linux 安装vlc观看直播flv协议hls协…...

单片机-- 松瀚sonix学习过程

硬件&#xff1a;松瀚sn8f5701sg、SN-LINK 3 Adapter模拟器、sn-link转接板 软件&#xff1a; keil-c51&#xff08;v9.60&#xff09;&#xff1a;建立工程&#xff0c;编辑&#xff0c;烧录程序 SN-Link_Driver for Keil C51_V3.00.005&#xff1a;安装sonix设备包和snlin…...

循环神经网络:从基础到应用的深度解析

&#x1f35b;循环神经网络&#xff08;RNN&#xff09;概述 循环神经网络&#xff08;Recurrent Neural Network, RNN&#xff09;是一种能够处理时序数据或序列数据的深度学习模型。不同于传统的前馈神经网络&#xff0c;RNN具有内存单元&#xff0c;能够捕捉序列中前后信息…...

从扩散模型开始的生成模型范式演变--SDE

SDE是在分数生成模型的基础上&#xff0c;将加噪过程扩展时连续、无限状态&#xff0c;使得扩散模型的正向、逆向过程通过SDE表示。在前文讲解DDPM后&#xff0c;本文主要讲解SDE扩散模型原理。本文内容主要来自B站Up主deep_thoughts分享视频Score Diffusion Model分数扩散模型…...

【python使用kazoo连ZooKeeper基础使用】

from kazoo.client import KazooClient, KazooState from kazoo.exceptions import NoNodeError,NodeExistsError,NotEmptyError import json# 创建 KazooClient 实例&#xff0c;连接到 ZooKeeper 服务器 zk KazooClient(hosts127.0.0.1:2181) zk.start()# 定义节点路径 path…...

【设计模式系列】解释器模式(十七)

一、什么是解释器模式 解释器模式&#xff08;Interpreter Pattern&#xff09;是一种行为型设计模式&#xff0c;它的核心思想是分离实现与解释执行。它用于定义语言的文法规则&#xff0c;并解释执行语言中的表达式。这种模式通常是将每个表达式抽象成一个类&#xff0c;并通…...

只出现一次的数字

只出现一次的数字 给你一个 非空 整数数组 nums &#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 你必须设计并实现线性时间复杂度的算法来解决此问题&#xff0c;且该算法只使用常量额外空间。 示例 1 &#xff…...

SpringMVC-08-json

8. Json 8.1. 什么是Json JSON(JavaScript Object Notation, JS 对象标记) 是一种轻量级的数据交换格式&#xff0c;目前使用特别广泛。采用完全独立于编程语言的文本格式来存储和表示数据。简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。易于人阅读和编写&#xf…...

技术文档的语言表达

技术文档的语言表达 在这个瞬息万变的技术世界中&#xff0c;了解如何撰写有效的技术文档显得尤为重要。无论是开发团队还是最终用户&#xff0c;清晰、简洁且有条理的文档都是连接各方的桥梁。本文将深入探讨技术文档的语言表达&#xff0c;从其重要性、写作原则到各种类型&a…...

UEFI 事件

UEFI 不再支持中断&#xff08;准确地说&#xff0c;UEFI 不再为开发者提供中断支持&#xff0c;但在UEFI内部还是使用了时钟中断&#xff09;&#xff0c;所有的异步操作都要通过事件&#xff08;Event&#xff09;来完成。 启动服务为开发者提供了用于操作事件、定时器及TPL…...

大师开讲-图形学领域顶级专家王锐开讲Vulkan、VSG开源引擎

王锐&#xff0c;毕业于清华大学&#xff0c;图形学领域顶级专家&#xff0c;开源技术社区的贡献者与推广者。三维引擎OpenSceneGraph的核心基石开发者与维护者&#xff0c;倾斜摄影数据格式osgb的发明人。著有《OpenSceneGraph 3 Cookbook》,《OpenSceneGraph 3 Beginers Guid…...

小F的矩阵值调整

问题描述 小F得到了一个矩阵。如果矩阵中某一个格子的值是偶数&#xff0c;则该值变为它的三倍&#xff1b;如果是奇数&#xff0c;则保持不变。小F想知道调整后的矩阵是什么样子的。 测试样例 样例1&#xff1a; 输入&#xff1a;a [[1, 2, 3], [4, 5, 6]] 输出&#xff1a…...

ORB-SLAM2 ----- LocalMapping::SearchInNeighbors()

文章目录 一、函数意义二、函数讲解三、函数代码四、本函数使用的匹配方法ORBmatcher::Fuse()1. 函数讲解2. 函数代码 四、总结 一、函数意义 本函数是用于地图点融合的函数&#xff0c;前面的函数生成了新的地图点&#xff0c;但这些地图点可能在前面的关键帧中已经生成过了&a…...

给UE5优化一丢丢编辑器性能

背后的原理 先看FActorIterator的定义 /*** Actor iterator* Note that when Playing In Editor, this will find actors only in CurrentWorld*/ class FActorIterator : public TActorIteratorBase<FActorIterator> {//..... }找到基类TActorIteratorBase /*** Temp…...

【Docker】常用命令汇总

Docker 是1个开源的应用容器引擎&#xff0c;基于Go 语言并遵从 Apache2.0 协议开源。 可以让开发者打包他们的应用以及依赖包到一个轻量级、可移植的容器中&#xff0c;然后发布到任何流行的 Linux 机器上&#xff0c;也可以实现虚拟化。 容器是完全使用沙箱机制&#xff0c;相…...

Mybatis:CRUD数据操作之多条件查询及动态SQL

Mybatis基础环境准备请看&#xff1a;Mybatis基础环境准备 本篇讲解Mybati数据CRUD数据操作之多条件查询 1&#xff0c;编写接口方法 在 com.itheima.mapper 包写创建名为 BrandMapper 的接口。在 BrandMapper 接口中定义多条件查询的方法。 而该功能有三个参数&#xff0c;…...

【笔记】轻型民用无人驾驶航空器安全操控

《轻型民用无人驾驶航空器安全操控》 理论考试培训材料 法规部分 【民用无人驾驶航空器的分类】 1、如何定义微型、轻型无人驾驶航空器&#xff1f; 微型无人驾驶航空器&#xff0c;是指空机重量小于0.25千克&#xff0c;最大平飞速度不超过40千米/小时&#xff0c;无线电发…...

TouchGFX设计模式代码实例说明

一)Model - View - Presenter (MVP) 模式在 TouchGFX 中的应用 1)Model&#xff08;模型&#xff09;&#xff1a; 模型代表应用程序的数据和业务逻辑。例如&#xff0c;在一个简单的计数器应用中&#xff0c;模型可以是一个包含计数器当前值的类。 class CounterModel { pri…...

flink学习(7)——window

概述 窗口的长度(大小): 决定了要计算最近多长时间的数据 窗口的间隔: 决定了每隔多久计算一次 举例&#xff1a;每隔10min,计算最近24h的热搜词&#xff0c;24小时是长度&#xff0c;每隔10分钟是间隔。 窗口的分类 1、根据window前是否调用keyBy分为键控窗口和非键控窗口…...

restTemplate get请求

报错解释&#xff1a; 这个报错信息表明在使用RestTemplate进行GET请求时&#xff0c;需要提供一个请求类型&#xff08;reqType&#xff09;&#xff0c;但是传入的值为空。这通常意味着在构建请求或者调用方法时&#xff0c;没有正确设置请求的Content-Type头部&#xff0c;…...

ffmpeg 预设的值 加速

centos 安装ffmpeg 编译安装 官网获取最新的linux ffmpeg 代码 https://ffmpeg.org//releases/ mkdir -p /data/app/ffmpeg cd /data/app/ffmpeg wget http://www.ffmpeg.org/releases/ffmpeg-7.1.tar.gz tar -zxvf ffmpeg-7.1.tar.gz#安装所需的编译环境 yum install -y \…...

做网站之前要备案是什么意思/全国教育培训机构平台

Spring框架已经成为Java开发人员的必备知识&#xff0c;而且Spring 3引入了强大的新特性&#xff0c;例如SpEL、Spring 表达式语言、loC 容器的新注解以及用户急需的对REST的支持。无论你是刚刚接触Spring还是被 Spring 3.0的新特性所吸引&#xff0c;这份笔记都是掌握Spring的…...

阜宁做网站的公司/小红书怎么做关键词排名优化

当你打开任何一个朋友的微信朋友圈&#xff0c;首先映入眼帘的是什么呢?你可能会说是他的头像&#xff0c;但其实最吸睛的除了头像之外还有朋友圈的背景图。这块地方比头像的位置更为宽大&#xff0c;我们在这里可以充分发挥自己的想象力设计出适合自己的非常具有吸引力的朋友…...

怎么做网站seo优化/深圳网络品牌推广公司

演讲达人成长记——用身体提升演讲感染力 你是否想自信满满地站在讲台上&#xff1f;你是否想滔滔不绝地发表自己的观点&#xff1f;《演讲达人成长记》&#xff0c;助你早日在讲台上叱咤风云&#xff01; 讲座主题&#xff1a; 时间&#xff1a;2013年1月26日上午9:45—11:30 …...

苏州网站设计公司/企业网站建设的步骤

为什么80%的码农都做不了架构师&#xff1f;>>> 在我把博客的标语修改了以后&#xff0c;当然只是一个某方面的测试。生活是一个有趣的循环&#xff0c;当我们试着往围城外走的时候&#xff0c;我们又被拉到围城里。 ##什么是全栈工程师 在现在这一个时代来说&…...

电子商务网站建设设计报告/武汉百度推广开户

计算机应用类专业建设和革新探索(共3022字)计算机应用类专业建设和革新探索(共3022字)1.高职高专计算机应用类专业现状剖析1.1专业定位欠准确&#xff0c;职业方向不明确计算机应用类专业相关的职业岗位多&#xff0c;每个职业岗位在知识水平和能力结构上各有侧重&#xff0c;计…...

江西学校网站建设/蚌埠seo外包

前言 在国内编译Spark项目需要从Maven源下载很多依赖包&#xff0c;官方源在国内大环境下的下载速度大家都懂得&#xff0c;那个煎熬啊&#xff0c;简直是浪费生命。 如果你的下载速度很快&#xff0c;你现在就可以无视这篇文章了。 阿里云给国内开发者提供了一个非常的良心的服…...