std::ratio<1,1000> 是什么意思?
author: hjjdebug
date: 2025年 05月 14日 星期三 09:45:24 CST
description: std::ratio<1,1000> 是什么意思?
文章目录
- 1. 它是一种数值吗?
- 2. 它是一种类型吗?
- 3. std:ratio 是什么呢?
- 4. 分析一个展开后的模板函数
- 5.小结:
前言: std::ratio 是c++11中引入的模板类.表示比例.
靠,连个比值都定义一个类,我相信c++已经做了很多类了.
想干什么事,一般找到调用方法就可以了.
这个类先假定它很复杂, 等读完会发现它很简单!
先给一个简单的例子来研究.
$ cat main.cpp
#include <iostream>
#include <ratio>
using namespace std;
// 定义一个模板函数,接受一个 std::ratio 类型参数
// 这个类型有静态变量num 和den, 由于是静态变量,会直接在代码中用常数展开.
// 这个模板函数将来会生成很多个函数,不同的类型就会生成不同的函数代码,是静态编译多态性的一种
template <typename Ratio>
void print_ratio() {printf("num:%ld,den:%ld\n",Ratio::num,Ratio::den);
}int main() {using r=std::ratio<1,1000>; //using 就是typedef, 编译时直接替换// std::ratio<1,1000> 是个类型, 而非数值,//这里类型名会与函数名共同构成一个代码中的导出函数名叫print_ratio<std::ratio<1l,1000l>>print_ratio<r>(); // 输出 Numerator: 1, Denominator: 1000// 这里std::ratio<1,10>时另一种类型,是分子为1,分母为10的类型print_ratio<std::ratio<1,10>>();//用类型实例化对象std::ratio<1,1000> obj;cout<<"num:"<<obj.num<<",den:"<<obj.den<<endl;cout<<"size_type:"<<sizeof(std::ratio<1,1000>)<<",size_obj:"<<sizeof(obj)<<endl;return 0;
}
执行:
./temp3
num:1,den:1000
num:1,den:10
num:1,den:1000
size_type:1,size_obj:1
代码中, std::ratio<1,1000> 是什么意思呢?
1. 它是一种数值吗?
它不是数值
看一下gdb 中的打印
p std::ratio<1,1000>
Attempt to use a type name as an expression
2. 它是一种类型吗?
它是一种类型.
看一下 gdb 的打印
ptype std::ratio<1,1000>
type = struct std::ratio<1, 1000> {
static const intmax_t num;
static const intmax_t den;
}
这个类型包含两个静态成员变量
p std::ratio<1,1000>::num
$1 = 1
p std::ratio<1,1000>::den
$2 = 1000
可见类型也可以包含属于自己的数值. 这2个数值在函数中用Ratio::num, Ratio::den引用过.
3. std:ratio 是什么呢?
显然它是一个模板类,因为有<>,有两个模板参数,因为给了1,1000
这样就可以由这个模板类,根据模板参数的不同,创建很多个类.
这里所说的类,就是类型.注意,只是类型,不是对象. 你可以用类型实例化一个对象,就像代码中那样.
下面看模板类的定义:
template<intmax_t _Num, intmax_t _Den = 1>struct ratio{static constexpr intmax_t num =_Num * __static_sign<_Den>::value / __static_gcd<_Num, _Den>::value;static constexpr intmax_t den =__static_abs<_Den>::value / __static_gcd<_Num, _Den>::value;typedef ratio<num, den> type;};
这个结构看起来还是有点复杂,
static 修饰的变量叫静态变量,它是属于类的而不属于对象.
constexpr 表示它会在编译期计算出数值
intmax_t 是 long int 的别名
__static_sign<_Den>::value, 这也是一个模板类,拿到这个类的带符号的分母的value值.
__static_gcd<_Num, _Den>::value, 拿到gcd模板类的value值,该值是_Num,_Den 的最大公约数.
__static_abs<_Den>::value, 分母绝对值模板类的value值
我们假定分子,分母互质, 即gcd(num,den)=1, 分子,分母都为正数,则上式可以简化为:
template<intmax_t _Num, intmax_t _Den = 1>
struct ratio
{
static constexpr intmax_t num = _Num;
static constexpr intmax_t den = _Den;
};
这样就看清了,ratio 是一个带2参数的模板类, constexpr 的含义是编译时就付给值.
4. 分析一个展开后的模板函数
模板函数 template
void print_ratio();
当类型std::ratio<1,1000> 与函数名 print_ratio相遇时
共同构成一个完整的导出函数名叫print_ratio<std::ratio<1l,1000l>>
该名称才是真实的函数导出名称(已经被c++filt 过滤过).
真实名称是这样的:_Z11print_ratioISt5ratioILl1ELl1000EEEvv,不是给人看的,给编译器看的.
下面是函数 print_ratio<std::ratio<1l,1000l>> 的反汇编代码.
是的,函数名称是可以带角标及逗号等字符的,比c常用的字符下化线丰富了不少.
重点注释了函数中怎样使用类型std::ratio<1,1000>中的静态成员变量num及den, 就是直接用数值.
/*(gdb) disassemble/m print_ratio<std::ratio<1l, 1000l> >
Dump of assembler code for function print_ratio<std::ratio<1l, 1000l> >():
9 void print_ratio() {0x0000555555555203 <+0>: endbr64 0x0000555555555207 <+4>: push %rbp0x0000555555555208 <+5>: mov %rsp,%rbp10 printf("num:%ld,den:%ld\n",Ratio::num,Ratio::den);0x000055555555520b <+8>: mov $0x3e8,%edx //类中的静态成员,直接给1000数值为第3参数0x0000555555555210 <+13>: mov $0x1,%esi //类中静态成员,直接给1数值为第2参数0x0000555555555215 <+18>: lea 0xde8(%rip),%rdi # 0x5555555560040x000055555555521c <+25>: mov $0x0,%eax0x0000555555555221 <+30>: callq 0x555555555070 <printf@plt>11 }0x0000555555555226 <+35>: nop0x0000555555555227 <+36>: pop %rbp0x0000555555555228 <+37>: retq End of assembler dump.*/
5.小结:
- 模板参数可以是整形数,例如上边的1,1000
- 类型名称可以带角标. 例:std::ratio<1,1000>
- 类型可以包含属于自己的数值. 例Ratio::num, Ratio::den
- 类型可以与函数名一起构成实例化后的模板函数名称. print_ratio<std::ratio<1l,1000l>>
- 类型实例化后的对象可以不包含任何数值. sizeof(obj)=1
- 类型的运行期大小也可以为0. sizeof(std::ratio<1,1000>=1,
说明它包含的类成员变量都是编译器的常数. 例Ratio::num=1,Ratio::den=1000 - 一个模板类std::ratio<intmax_t num,intmax_t den>,能实例化出很多实例化类,
std::ratio<1,1000>只是其中之一. 可以随心所欲创造很多种类型,例std::ratio<1,10)等. - 类型是跟gcc 的一种约定,你定义了这种类型,gcc就认识了这种类型,你就可以使用这种类型.
言之不尽,c++相较与c而言, 有更多的内涵需要让gcc明白, gcc会理解我们更复杂的表达方式.
相关文章:
std::ratio<1,1000> 是什么意思?
author: hjjdebug date: 2025年 05月 14日 星期三 09:45:24 CST description: std::ratio<1,1000> 是什么意思? 文章目录 1. 它是一种数值吗?2. 它是一种类型吗?3. std:ratio 是什么呢?4. 分析一个展开后的模板函数5.小结: …...

基于Llama3的开发应用(二):大语言模型的工业部署
大语言模型的工业部署 0 前言1 ollama部署大模型1.1 ollama简介1.2 ollama的安装1.3 启动ollama服务1.4 下载模型1.5 通过API调用模型 2 vllm部署大模型2.1 vllm简介2.2 vllm的安装2.3 启动vllm模型服务2.4 API调用 3 LMDeploy部署大模型3.1 LMDeploy简介3.2 LMDeploy的安装3.3…...
2025.05.17淘天机考笔试真题第三题
📌 点击直达笔试专栏 👉《大厂笔试突围》 💻 春秋招笔试突围在线OJ 👉 笔试突围OJ 03. 奇偶平衡树分割问题 问题描述 K小姐是一位园林设计师,她设计了一个由多个花坛组成的树形公园。每个花坛中种植了不同数量的花…...

windows 10 做服务器 其他电脑无法访问,怎么回事?
一般我们会先打开win10自己的防火墙策略,但是容易忽略 电脑之间 路由器上的防火墙,此时也需要查看一下,可以尝试先关闭路由器防火墙,如果可以了,再 设置路由器上的防火墙规则。 将路由器的上网设置 改成 路由模式 &a…...

Linux进程信号处理(26)
文章目录 前言一、信号的处理时机处理情况“合适”的时机 二、用户态与内核态概念重谈进程地址空间信号的处理过程 三、信号的捕捉内核如何实现信号的捕捉?sigaction 四、信号部分小结五、可重入函数六、volatile七、SIGCHLD 信号总结 前言 这篇就是我们关于信号的最…...
【从设置到上传的全过程】本地多个hexo博客,怎么设置ssh才不会互相影响
偶然间,想多建一个博客,但电脑已经有一个博客了,怎么设置ssh才不会互相影响呢? 在 Windows 系统上设置多个 Hexo 博客的 SSH 配置,避免互相影响,通常户就需要为每个博客配置不同的 SSH 密钥,并…...
顶层架构 - 消息集群推送方案
一、推送基础概念简述 在即时通讯(IM)系统中,最基础的一件事就是“如何把消息推送给用户”。为了实现这个过程,我们要先了解两种常见的网络通信方式:HTTP 和 WebSocket。 1. HTTP 是什么? HTTP 就像一次性…...
Python训练打卡Day26
函数专题1:函数定义与参数 知识点回顾: 函数的定义变量作用域:局部变量和全局变量函数的参数类型:位置参数、默认参数、不定参数传递参数的手段:关键词参数传递参数的顺序:同时出现三种参数类型时 到目前为…...
构建优雅对象的艺术:Java 建造者模式的架构解析与工程实践
一、建造者模式的本质与核心价值 在面向对象的软件设计中,创建复杂对象一直是一个需要精心处理的问题。当一个对象的构建需要多个步骤,并且这些步骤具有不同的组合方式时,传统的构造函数方式会显得力不从心。建造者模式(Builder …...

报表控件stimulsoft教程:如何在报表和仪表板中创建热图
Stimulsoft Ultimate (原Stimulsoft Reports.Ultimate)是用于创建报表和仪表板的通用工具集。该产品包括用于WinForms、ASP.NET、.NET Core、JavaScript、WPF、PHP、Java和其他环境的完整工具集。无需比较产品功能,Stimulsoft Ultimate包含了…...
(8)python开发经验
文章目录 1 下载python2 pip安装依赖无法访问3 系统支持4 下载python文档5 设置虚拟环境6 编译安装python 更多精彩内容👉内容导航 👈👉Qt开发 👈👉python开发 👈 1 下载python 下载地址尽量不要下载最新版…...
0x08.Redis 支持事务吗?如何实现?
回答重点 Redis 支持事务,但它的事务与 MySQL 等关系型数据库的事务有着本质区别。MySQL 中的事务严格遵循 ACID 特性,而 Redis 中的事务主要保证的是命令执行的原子性和隔离性,即所有命令在一个不可分割的操作中顺序执行,不会被其他客户端的命令请求所打断。 最关键的区…...

win32相关(字符编码)
字符编码 ASCII编码 ASCII(American Standard Code for Information Interchange,美国信息交换标准代码)是最基础的字符编码标准,用于在计算机和其他设备中表示文本 基本概念 7位编码: ASCII使用7位二进制数&#x…...

使用Langfuse和RAGAS,搭建高可靠RAG应用
大家好,在人工智能领域,RAG系统融合了检索方法与生成式AI模型,相比纯大语言模型,提升了准确性、减少幻觉且更具可审计性。不过,在实际应用中,当建好RAG系统投入使用时,如何判断接收信息是否正确…...
VSCode + Cline AI辅助编程完全指南
VSCode Cline AI辅助编程完全指南 在当今AI快速发展的时代,程序员可以通过AI工具极大地提高工作效率。本教程将详细介绍如何使用VSCode结合Cline(Claude AI助手)进行AI辅助编程,帮助你提高开发效率,解决复杂问题。 …...

android studio导入项目
如果 gradle-8.0-bin.zip 没有下载成功 可以点击进入这个网站:https://services.gradle.org/distributions/ 找到和自己本版相同的gradle-8.0-bin.zip文件找到自己版本进行下载; 如果下载依赖失败, 可以手动下载依赖编译过程中的jar https://repo.maven.apache.org/…...

Autosar Nvm下电存储实现方式-基于ETAS工具
文章目录 前言Autosar Nvm相关定义Nvm Ram Block States状态切换Nvm_WriteAll函数NvBlock配置生成代码分析及使用总结前言 Nvm中存储的数据,一般有两种存储方式,一个是立即存,一个是下电存,之前介绍过立即存的配置,本文介绍下电存的配置及实现 Autosar Nvm相关定义 Nvm…...

c# 数据结构 树篇 入门树与二叉树的一切
事先声明,本文不适合对数据结构完全不懂的小白 请至少学会链表再阅读 c# 数据结构 链表篇 有关单链表的一切_c# 链表-CSDN博客 数据结构理论先导:《数据结构(C 语言描述)》也许是全站最良心最通俗易懂最好看的数据结构课(最迟每周五更新~~&am…...

Python Bug 修复案例分析:asyncio 事件循环异常引发的程序崩溃 两种修复方法
在 Python 异步编程的工作中,asyncio库为我们提供了高效处理并发任务的强大工具。然而,asyncio在使用过程中也可能因为一些细节处理不当而引发 Bug。下面,我们就来深入分析一个因asyncio事件循环异常导致程序崩溃的典型案例。兴趣的友友可以借…...

题单:递归求和
宣布一个重要的事情,我的洛谷有个号叫 题目描述 给一个数组 a:a[0],a[1],...,a[n−1]a:a[0],a[1],...,a[n−1] 请用递归的方式出数组的所有数之和。 提示:递推方程 f(x)f(x−1)a[x]f(x)f(x−1)a[x]; 输入格式 第一行一个正整数 n (n≤100)n (n≤100)…...
融智学视域下的系统性认知增强框架——基于文理工三类AI助理赋能HI四阶跃迁路径
融智学视域下的系统性认知增强框架 ——基于文理工三类AI助理赋能HI四阶跃迁路径 一、如何排除50个认知偏差:消除50类偏差的精准矫正系统 1. 技术架构 文科AI: 构建文化语义场(Cultural Semantic Field, CSF),通过…...

怎么在excel单元格1-5行中在原来内容前面加上固定一个字?
环境: WPS 2024 问题描述: 怎么在excel单元格1-5行中在原来内容前面加上固定一个字? 解决方案: 1.在Excel中,如果您想在单元格的内容前面添加一个固定的字,可以通过以下几种方法实现: 方法…...
使用 Vue Tour 封装一个统一的页面引导组件
项目开发过程中需要实现用户引导功能,经过调研发现一个好用的 Vue 插件 vue-tour,今天就来分享一下我是如何基于 vue-tour 封装一个统一的引导组件,方便后续在多个页面复用的。 📦 第一步:安装 vue-tour 插件 首先安装…...

OpenHarmony 开源鸿蒙南向开发——linux下使用make交叉编译第三方库——mqtt库
准备工作 请依照这篇文章搭建环境 OpenHarmony 开源鸿蒙南向开发——linux下使用make交叉编译第三方库——环境配置_openharmony交叉编译-CSDN博客 下载 wget ftp://ftp.gnutls.org/gcrypt/gnutls/v3.5/gnutls-3.5.9.tar.xz 解压 tar -xf mkdir ./out cd ./out Cmake命…...

数据结构 -- 顺序查找和折半查找
查找的基本概念 基本概念 查找:在数据集合中寻找满足某种条件的数据元素的过程 查找表(查找结构):用于查找的数据集合称为查找表,它由同一类型的数据结构元素(或记录)组成 关键字࿱…...

信息收集+初步漏洞打点
目标:理解信息收集在渗透测试中的意义,熟悉常用工具用法,完成基本打点测试 一.理论学习: 模块内容说明信息收集分类主动信息收集 vs 被动信息收集目标发现子域名、IP、端口、子站点、目录、接口技术指纹识别Web框架(如…...
2025年01月10日浙江鑫越系统科技前端面试
目录 vue2 和 vue3 的区别vue 怎么封装组件js 怎么把一个数组置空怎么组件自己调用自己的组件v-bind:attribute 和 v-bind“{attribute}” 的区别var let const 的区别this 指向作用域链闭包原型链事件循环 1. vue2 和 vue3 的区别 Vue 2 和 Vue 3 在多个方面存在区别&#…...

JavaScript【5】DOM模型
1.概述: DOM (Document Object Model):当页面被加载时,浏览器会创建页面的文档对象模型,即dom对象;dom对象会被结构化为对象树,如一个HTML文档会被分为head,body等部分,而每个部分又…...

Cloudflare防火墙拦截谷歌爬虫|导致收录失败怎么解决?
许多站长发现网站突然从谷歌搜索结果中“消失”,背后很可能是Cloudflare防火墙误拦截了谷歌爬虫(Googlebot),导致搜索引擎无法正常抓取页面。 由于Cloudflare默认的防护规则较为严格,尤其是针对高频访问的爬虫IP&…...
鸿蒙OSUniApp 实现的表单验证与提交功能#三方框架 #Uniapp
UniApp 实现的表单验证与提交功能 前言 在移动端应用开发中,表单是用户与应用交互的重要媒介。一个好的表单不仅布局合理、使用方便,还应该具备完善的验证与提交功能,以确保用户输入的数据准确无误。本文将分享如何在 UniApp 中实现表单验证…...