在开发嵌入式系统时,尤其是处理大数时,会遇到取值范围的问题。51单片机通常没有内建大整数支持,因此我们需要采用不同的方法来解决这一问题
00 两种可行方法分别是:
- 使用数组存储每一位数据并进行进位运算:通过将大数按位拆分成数组,然后实现逐位加法、进位等操作。
- 使用符号变量进行计算:将数值分成低位和高位,分别用符号变量进行计算。
01:使用数组存储数据大小并处理进位
在这种方法中,我们将数值拆分成若干个位数(数组中的元素),并通过数组存储每一位数字,手动处理进位。
假设我们要处理的数值是存款金额 principal 和最终计算的利息,数值可能非常大,需要使用一个数组来按位存储。每个数组元素存储一个数字(从0到9),并手动实现加法和进位。
代码实现:
#include <reg52.h>
#include <math.h>
#include <stdio.h>#define MAX_DIGITS 20 // 数字的最大位数// 复利计算函数,采用逐位运算
void multiply_by_factor(int* number, double factor) {double carry = 0; // 用于存储进位for (int i = 0; i < MAX_DIGITS; i++) {double digit_value = number[i] * factor + carry;number[i] = (int)digit_value % 10; // 当前位carry = digit_value / 10; // 进位}
}// 打印数组中的大数
void print_large_number(int* number) {int i = MAX_DIGITS - 1;while (i >= 0 && number[i] == 0) {i--; // 去掉前导零}if (i == -1) {uart_send_char('0'); // 如果数组全是0,则显示0} else {while (i >= 0) {uart_send_char(number[i] + '0');i--;}}
}// 使用字符数组存储和计算复利
void calculate_large_interest(double principal, double rate, int years) {int large_number[MAX_DIGITS] = {0}; // 数字存储数组int i;// 将存款金额(principal)转换为数组形式for (i = MAX_DIGITS - 1; i >= 0; i--) {large_number[i] = (int)principal % 10; // 存储个位数principal /= 10; // 除以10,获得下一个位}// 计算复利double factor = 1 + rate / 100;for (i = 0; i < years; i++) {multiply_by_factor(large_number, factor);}// 打印结果print_large_number(large_number);
}void main() {double principal = 1000.00; // 假设存款金额为1000元double rate = 3.5; // 假设年利率为3.5%int years = 5; // 假设存款年数为5年uart_init(); // 初始化串口// 计算并打印复利结果calculate_large_interest(principal, rate, years);
}
large_number[MAX_DIGITS]:用于存储存款金额的每一位数字,MAX_DIGITS定义了存储的最大位数。multiply_by_factor():该函数逐位计算大数与利率的乘积,并处理每一位的进位。通过逐位相乘并累加进位来实现复利计算。print_large_number():将存储的大数数组打印到串口。calculate_large_interest():这是主计算函数,首先将本金principal转换为大数数组,然后逐年计算复利。
02:使用符号变量(高位和低位分别存储)
这种方法相对简单,通过将数值拆分为低位和高位两部分来计算。符号变量可以帮助处理较大范围的数值。虽然不如数组方法灵活,但可以应对一些场景。
代码实现:
#include <reg52.h>
#include <stdio.h>
#include <math.h>#define MAX_INT 0xFFFF // 假设使用16位符号变量// 定义一个结构体用于存储大数
typedef struct {unsigned int low; // 低位unsigned int high; // 高位
} BigNumber;// 将两个整数合并成一个大数
BigNumber multiply_big_number(BigNumber num, double factor) {double result_low = num.low * factor;double result_high = num.high * factor;// 处理低位进位unsigned int new_low = (unsigned int)result_low;unsigned int carry = (unsigned int)(result_low / MAX_INT);// 处理高位和进位unsigned int new_high = (unsigned int)(result_high + carry);BigNumber result = { new_low, new_high };return result;
}// 打印大数
void print_big_number(BigNumber num) {uart_send_char((num.high >> 8) & 0xFF); // 打印高位字节uart_send_char(num.high & 0xFF);uart_send_char((num.low >> 8) & 0xFF); // 打印低位字节uart_send_char(num.low & 0xFF);
}void calculate_large_interest(double principal, double rate, int years) {// 初始化大数BigNumber principal_big = { (unsigned int)(principal), 0 };// 计算复利BigNumber result = principal_big;double factor = 1 + rate / 100;for (int i = 0; i < years; i++) {result = multiply_big_number(result, factor);}// 打印结果print_big_number(result);
}void main() {double principal = 1000.00; // 假设存款金额为1000元double rate = 3.5; // 假设年利率为3.5%int years = 5; // 假设存款年数为5年uart_init(); // 初始化串口// 计算并打印复利结果calculate_large_interest(principal, rate, years);
}
解释:
BigNumber结构体:用于存储一个大数的低位和高位。multiply_big_number():将大数与利率相乘,并处理低位和高位的进位。print_big_number():将大数通过串口输出,打印低位和高位。calculate_large_interest():计算复利的主函数,初始化本金的高低位数据,逐年进行复利计算。
其他需要注意的问题:
2. 51内存大小限制
需减少数组大小和double的使用
同时注意char 类型数组需要以'\0'结尾

3.复利年数过多时,使用数组单个计算理论可行,但程序疑似跑飞
4.无法使用#include <math.h>
pow((1 + rate / 100), years)
总结:
- 方案1(数组存储和逐位计算):适用于处理更大范围的数值,灵活性更高,可以根据需要扩展位数。
- 方案2(符号变量存储):相对简单,通过高低位分离来处理较大的数值,但计算灵活性较差。
两种方法各有优劣,选择哪一种方法取决于系统资源和所需的计算精度。如果存储空间和计算能力有限,使用符号变量(方案2)可能更为高效。如果需要处理更大范围的数值或更高精度,使用数组存储和逐位计算(方案1)会更适合
相关文章:
在开发嵌入式系统时,尤其是处理大数时,会遇到取值范围的问题。51单片机通常没有内建大整数支持,因此我们需要采用不同的方法来解决这一问题
00 两种可行方法分别是: 使用数组存储每一位数据并进行进位运算:通过将大数按位拆分成数组,然后实现逐位加法、进位等操作。使用符号变量进行计算:将数值分成低位和高位,分别用符号变量进行计算。 01:使用…...
【Compose multiplatform教程20】在应用程序中使用多平台资源
为项目设置资源后,生成项目以生成提供资源访问权限的特殊类。要重新生成类和所有资源访问器,请再次生成项目或在 IDE 中重新导入项目。ResRes 之后,您可以使用生成的类从您的代码或外部库访问配置的多平台资源。 自定义访问器类生成 您可以使…...
深入浅出:从入门到精通大模型Prompt、SFT、RAG、Infer、Deploy、Agent
阅读原文 渐入佳境 我们都知道,通过编写一个提示词(prompt),我们可以引导大模型生成回答,从而开启愉快的人工智能对话,比如让模型介绍一下卡皮巴拉。上边简图描述了这个过程,我们拆成两部分 pr…...
紫光同创-盘古200pro+开发板
本原创文章由深圳市小眼睛科技有限公司创作,版权归本公司所有,如需转载,需授权并注明出处(www.meyesemi.com) 一、开发系统介绍 开发系统概述 MES2L676-200HP 开发板采用紫光同创 logos2 系列 FPGA,型号:…...
iOS 中的 nil、Nil、NULL、NSNull 僵尸对象和野指针
iOS 中的 nil、Nil、NULL、NSNull 僵尸对象和野指针-CSDN博客 类型含义使用场景示例nil表示一个指向 Objective - C 对象的空指针。在 Objective - C 和 Swift(与 Objective - C 交互时)中用于表示对象不存在。当一个对象变量没有指向任何有效的对象实例…...
【优选算法】有效三角形的个数(双指针算法)
优质专栏:算法_云边有个稻草人的博客-CSDN博客 目录 【611. 有效三角形的个数 - 力扣(LeetCode)】 解法一: 解法二: 【611. 有效三角形的个数 - 力扣(LeetCode)】 解法一: 三层for…...
中介者模式(Mediator Pattern)、桥接模式(Bridge Pattern) 和 策略模式(Strategy Pattern)
中介者模式(Mediator Pattern)、桥接模式(Bridge Pattern) 和 策略模式(Strategy Pattern) 都是常见的设计模式,它们解决不同类型的问题。我们将通过 Swift 示例来说明它们的使用场景࿰…...
客户案例:基于慧集通打通聚水潭电商ERP与用友U8系统集成之路
一、引言 本原型客户是 生物科技公司,其公司系列抗菌抗病毒产品广泛应用于医疗用品、纺织服饰、家纺用品、母婴护理、女性用品、个人防护等多个领域。在知识产权方面,公司在专业领域已获得商标和专利近百项,创新能力得到了国家及行业内普遍认…...
阿里云clb是什么
传统型负载均衡服务 阿里云CLB(Classic Load Balancer)是阿里云提供的一种传统型负载均衡服务,主要用于将访问流量根据转发策略分发到后端多台云服务器。 CLB的定义和功能 CLB是一种流量分发控制服务,通过设置虚拟服务地…...
【Cursor编辑器】自用经验和实操(迭代更新)
1.启动composer crtl I 2.生成直接一直问加载 3. 实操 生成个知识图谱,使用csv文件里面的数据创关系和节点。...
【学习笔记】ChatGPT原理与应用开发——基础科普
HuggingLLM(ChatGPT原理与应用开发) 原文链接:HuggingLLM(ChatGPT原理与应用开发)-课程详情 | Datawhale 此处仅为学习记录和总结 1:基础科普 1.1:自然语言背景 图灵测试 如果一个人&#x…...
基于Web的实验中心工作管理网站的设计与实现
写作任务 一、课题背景 实验中心承担了全校计算机公共基础课程和学院专业课程,需要对实验中心工作进行有效的管理。 二、课题任务 本课题设计和实现实验中心工作管理系统。 系统的主要内容包括: (1)人员管理; &am…...
docker 安装minio
docker pull minio/minio #启动 mkdir -p /root/minio/config mkdir -p /root/minio/datadocker run -d \--name minio \-p 9002:9000 \-p 9001:9001 \--restartalways \-v /root/minio/data:/data \-v /root/minio/config:/root/.minio \-e "MINIO_ACCESS_KEYminioadmin…...
ubuntu下ipmi的使用(4028)
参考ubuntu系统下配置IPMI_ubuntu ipmi-CSDN博客 参考:ipmitool ubuntu 安装_ipmi centos ubuntu使用总结-CSDN博客 1.安装 sudo apt-get -y install ipmitool 2.加载 modprobe ipmi_msghandlermodprobe ipmi_devintfmodprobe ipmi_si 3.使用,查看不到的话&am…...
周记-唐纳德的《计算机程序设计艺术》
用代码生成代码 开发一个协议,字段有些多,每个字段是QT的属性,需要写Q_PROPERTY,一个一个编辑的话比较繁琐,耗费时间。后来就用代码生成了头文件和源文件,get和set还有signal函数,内容基本都是…...
极品飞车6的快捷键与车辆等级
极品飞车,英文全称为Need for Speed,是EA公司于1994年开始研发的赛车类竞技游戏。从1996年的《极品飞车-特别版》、2002年的《极品飞车:闪电追踪2》、2005年的《极品飞车:地下狂飙2》、到2024年《极品飞车:集结》,是70后、80年、90年等几代人…...
计算机毕业设计Python+知识图谱大模型AI医疗问答系统 健康膳食推荐系统 食谱推荐系统 医疗大数据 机器学习 深度学习 人工智能 爬虫 大数据毕业设计
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
纯真社区版IP库CZDB数据格式使用教程
1. 概述 纯真社区版IP库是一种免费且公开的IP地理位置数据库,主要面向非商业用途。其最新推出的CZDB格式是一种全新的数据文件格式,自2024年10月起将成为官方维护和更新的唯一版本。该格式支持同时存储IPv4和IPv6地址信息,具备以下优点&…...
Linux(Centos 7.6)软件包安装
Linux软件安装,常见的有三种方式,rpm方式、yum方式、源码编译安装方式。其中rpm方式可能存在依赖方式,可能会比较麻烦;源码编译安装同样可能会缺少一些编译需要的软件需要安装,也会比较麻烦;相对比较好的方…...
[WASAPI]音频API:从Qt MultipleMedia走到WASAPI,相似与不同
[WASAPI] 从Qt MultipleMedia 来看WASAPI 最近在学习有关Windows上的音频驱动相关的知识,在正式开始说WASAPI之前,我想先说一说Qt的Multiple Media,为什么呢?因为Qt的MultipleMedia实际上是WASAPI的一层封装,它在是线…...
Linux应用开发之网络套接字编程(实例篇)
服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …...
云原生核心技术 (7/12): K8s 核心概念白话解读(上):Pod 和 Deployment 究竟是什么?
大家好,欢迎来到《云原生核心技术》系列的第七篇! 在上一篇,我们成功地使用 Minikube 或 kind 在自己的电脑上搭建起了一个迷你但功能完备的 Kubernetes 集群。现在,我们就像一个拥有了一块崭新数字土地的农场主,是时…...
Qt Widget类解析与代码注释
#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this); }Widget::~Widget() {delete ui; }//解释这串代码,写上注释 当然可以!这段代码是 Qt …...
基础测试工具使用经验
背景 vtune,perf, nsight system等基础测试工具,都是用过的,但是没有记录,都逐渐忘了。所以写这篇博客总结记录一下,只要以后发现新的用法,就记得来编辑补充一下 perf 比较基础的用法: 先改这…...
高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
反射获取方法和属性
Java反射获取方法 在Java中,反射(Reflection)是一种强大的机制,允许程序在运行时访问和操作类的内部属性和方法。通过反射,可以动态地创建对象、调用方法、改变属性值,这在很多Java框架中如Spring和Hiberna…...
Spring AI 入门:Java 开发者的生成式 AI 实践之路
一、Spring AI 简介 在人工智能技术快速迭代的今天,Spring AI 作为 Spring 生态系统的新生力量,正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务(如 OpenAI、Anthropic)的无缝对接&…...
MySQL 8.0 OCP 英文题库解析(十三)
Oracle 为庆祝 MySQL 30 周年,截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始,将英文题库免费公布出来,并进行解析,帮助大家在一个月之内轻松通过OCP认证。 本期公布试题111~120 试题1…...
重启Eureka集群中的节点,对已经注册的服务有什么影响
先看答案,如果正确地操作,重启Eureka集群中的节点,对已经注册的服务影响非常小,甚至可以做到无感知。 但如果操作不当,可能会引发短暂的服务发现问题。 下面我们从Eureka的核心工作原理来详细分析这个问题。 Eureka的…...
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的“no matching...“系列算法协商失败问题
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的"no matching..."系列算法协商失败问题 摘要: 近期,在使用较新版本的OpenSSH客户端连接老旧SSH服务器时,会遇到 "no matching key exchange method found", "n…...
