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

STM32简易计算机设计

运用 A0上拉按钮和 A1 A2下拉按钮设计按键功能     加上独特的算法检测设计,先计算()内在计算乘除在计算加减的值在计算乘除优先级最后计算加减优先级

#include "stm32f10x.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>#define EXPR_MAX_LEN 64
#define UART_BAUDRATE 115200/* 全局变量声明 */
char expr_buffer[EXPR_MAX_LEN] = {0};
char result_str[32] = {0};
volatile uint8_t expr_index = 0;
volatile uint8_t expr_ready = 0;
float calc_result = 0;
uint8_t result_available = 0;/* 函数原型声明 */
void GPIO_Configuration(void);
void USART1_Init(void);
void USART1_SendString(char *str);
void ProcessExpression(void);
float EvaluateExpression(char *expr);
int GetOperatorPriority(char op);int main(void) {
//    SystemInit();GPIO_Configuration();USART1_Init();USART1_SendString("Calculator Ready!\r\n");while(1) {/* 处理按键事件 */static uint8_t key1_state = 1, key2_state = 1, key3_state = 1;// KEY1 按下计算if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == Bit_SET && key1_state) {key1_state = 0;if(expr_ready) ProcessExpression();} else if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == Bit_SET) key1_state = 1;// KEY2 发送结果if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1) == Bit_RESET && key2_state) {key2_state = 0;if(result_available) USART1_SendString(result_str);} else if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1) == Bit_SET) key2_state = 1;// KEY3 清零if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_2) == Bit_RESET && key3_state) {key3_state = 0;expr_index = 0;expr_ready = 0;result_available = 0;memset(expr_buffer, 0, EXPR_MAX_LEN);USART1_SendString("Cleared\r\n");} else if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_2) == Bit_SET) key3_state = 1;}
}/* 处理接收到的表达式 */
void ProcessExpression(void) {expr_ready = 0;calc_result = EvaluateExpression(expr_buffer);if(isnan(calc_result)) {strcpy(result_str, "Error\r\n");} else {snprintf(result_str, sizeof(result_str), "Result: %.2f\r\n", calc_result);}result_available = 1;
}/* 支持多运算符和浮点数的表达式求值 */
float EvaluateExpression(char *expr) {float num_stack[EXPR_MAX_LEN] = {0};char op_stack[EXPR_MAX_LEN] = {0};int num_top = -1, op_top = -1;char *p = expr;float num = 0;int decimal = 0;float fraction = 1.0f;//进来保证不到最后一个字符不停止while(*p != '\0') {//如果是数字或者小数点就进行字符数字转换成真正的数字的操作if(isdigit(*p) || *p == '.') {if(*p == '.') {decimal = 1;} else {if(decimal) {fraction *= 0.1f;num += (*p - '0') * fraction;} else {num = num * 10 + (*p - '0');}}p++;} //如果是符号就判断这个elseelse {//将前面处理好的数字压入数字堆栈,如果遇到两次符号,也就是当+(..)就是(情况就会压入零代表当前数字,那就不会在主机算区算错因为0不管怎么样都零if(num != 0 || decimal) {num_stack[++num_top] = num;num = 0;   					//如果遇到两次符号,也就是当+(..)就是(情况就会压入零代表当前数字decimal = 0;fraction = 1.0f;}//若果判断到括号(那就记录一下碰到了括号,这样就会订掉一次计算,因为字符加减乘除和括号都是字符,这样加这一段可以防止误判断这里为计算时间上这里只是个括号if(*p == '(') {op_stack[++op_top] = *p;} //判断括号里卖你是不是有数字计算op_stack[op_top] != '('这句就是判断上一个符号如果还是)那就是括号里啥也没有,那就不判断计算else if(*p == ')') {while(op_top >= 0 && op_stack[op_top] != '(') {//那这一段只是处理)出现,代表此时的括号内算式已经结束,只需判断前面括号内的结果诚意最后一个数字的结果//处理两次符号区//如果遇到两次符号,也就是当+(..)就是(情况就会压入零代表当前数字/*辅助计算区*/char op = op_stack[op_top--];float b = num_stack[num_top--];float a = num_stack[num_top--];if(op == '+') num_stack[++num_top] = a + b;else if(op == '-') num_stack[++num_top] = a - b;else if(op == '*') num_stack[++num_top] = a * b;else if(op == '/') num_stack[++num_top] = a / b;/*辅助计算区*/}if(op_top >= 0) op_top--; // 弹出左括号} else {//时时刻刻在进行优先级判断计算运算式得出结果,为什么这里只判断后者与前者符号的差距呢,因为一个数字只要他前后符号优先级一样那就不具备优先级计算1*2*6==6*1*2但1+2*8!=1*8+2//所以有了这一段那你不管是不是括号内算式都可以具有优先级计算  /* 主计算区*/while(op_top >= 0 && GetOperatorPriority(op_stack[op_top]) >= GetOperatorPriority(*p)) {char op = op_stack[op_top--];float b = num_stack[num_top--];float a = num_stack[num_top--];if(op == '+') num_stack[++num_top] = a + b;else if(op == '-') num_stack[++num_top] = a - b;else if(op == '*') num_stack[++num_top] = a * b;else if(op == '/') num_stack[++num_top] = a / b;}/*主计算区*/op_stack[++op_top] = *p;}p++;}}if(num != 0 || decimal) {num_stack[++num_top] = num;}while(op_top >= 0) {char op = op_stack[op_top--];float b = num_stack[num_top--];float a = num_stack[num_top--];if(op == '+') num_stack[++num_top] = a + b;else if(op == '-') num_stack[++num_top] = a - b;else if(op == '*') num_stack[++num_top] = a * b;else if(op == '/') {if(fabs(b) < 1e-6) return NAN;num_stack[++num_top] = a / b;}}return num_top >= 0 ? num_stack[num_top] : NAN;
}/* 获取运算符优先级 */
int GetOperatorPriority(char op) {if(op == '+' || op == '-') return 1;if(op == '*' || op == '/') return 2;return 0;
}/* USART1中断处理 */
void USART1_IRQHandler(void) {if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) {char ch = USART_ReceiveData(USART1);if(ch == '\r' || ch == '\n') {expr_buffer[expr_index] = '\0';expr_ready = 1;expr_index = 0;} else if(expr_index < EXPR_MAX_LEN-1) {expr_buffer[expr_index++] = ch;}USART_ClearITPendingBit(USART1, USART_IT_RXNE);}
}/* 初始化USART1 */
void USART1_Init(void) {GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);// 配置TX(PA9)和RX(PA10)GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;GPIO_Init(GPIOA, &GPIO_InitStructure);USART_InitStructure.USART_BaudRate = UART_BAUDRATE;USART_InitStructure.USART_WordLength = USART_WordLength_8b;USART_InitStructure.USART_StopBits = USART_StopBits_1;USART_InitStructure.USART_Parity = USART_Parity_No;USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;USART_Init(USART1, &USART_InitStructure);USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;NVIC_Init(&NVIC_InitStructure);USART_Cmd(USART1, ENABLE);
}/* GPIO配置 */
void GPIO_Configuration(void) {GPIO_InitTypeDef GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);// KEY1(PA0), KEY2(PA1), KEY3(PA2) 输入GPIO_InitStructure.GPIO_Pin =GPIO_Pin_1 | GPIO_Pin_2;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 ;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;GPIO_Init(GPIOA, &GPIO_InitStructure);
}/* 串口发送字符串 */
void USART1_SendString(char *str) {while(*str) {USART_SendData(USART1, *str++);while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);}
}

相关文章:

STM32简易计算机设计

运用 A0上拉按钮和 A1 A2下拉按钮设计按键功能 加上独特的算法检测设计&#xff0c;先计算&#xff08;&#xff09;内在计算乘除在计算加减的值在计算乘除优先级最后计算加减优先级 #include "stm32f10x.h" #include <stdio.h> #include <stdlib.h>…...

GUI实验

题目&#xff1a; 编程包含一个标签和一个按钮&#xff0c;单击按钮时&#xff0c;标签的内容在"你好"和"再见"之间切换。 分析&#xff1a; 导入所需的Java库&#xff1a;程序使用了 javax.swing 包中的一些类来创建图形用户界面。 创建一个 JFrame 对象…...

量子计算 | 量子密码学的挑战和机遇

量子计算在密码学中的应用现主要体现在对现有加密算法的威胁上。最著名的例子是Shor算法&#xff0c;该算法能够在多项式时间内分解大整数&#xff0c;从而威胁到基于大数分解的加密算法&#xff0c;如RSA加密。此外&#xff0c;量子计算还可以加速某些类型的密码分析&#xff…...

linux系统查看硬盘序列号

Linux系统查看硬盘信息指南 方法一&#xff1a;hdparm工具 sudo hdparm -i /dev/sda输出示例&#xff1a;在返回信息中查找"SerialNo"字段为序列号&#xff0c;"Model"字段为硬盘型号注意&#xff1a;必须使用root权限&#xff0c;普通用户需在命令前加s…...

分享一些多模态文档解析思路

多模态文档解析思路小记 作者&#xff1a;Arlene 原文&#xff1a;https://zhuanlan.zhihu.com/p/1905635679293122466 多模态文档解析内容涉及&#xff1a;文本、表格和图片 解析思路v1 基于mineru框架对pdf文件进行初解析 其具备较完整的布局识别和内容识别&#xff0c;并将…...

CSS 选择器入门

一、CSS 选择器基础&#xff1a;快速掌握核心概念 什么是选择器&#xff1f; CSS 选择器就像 “网页元素的遥控器”&#xff0c;用于定位 HTML 中的特定元素并应用样式。 /* 结构&#xff1a;选择器 { 属性: 值; } */ p { color: red; } /* 选择所有<p>元素&#xff0c;…...

【Django】Django DRF 中如何手动调用分页器返回分页数据(APIView,action场景)

&#x1f4e6; Django DRF 中如何手动调用分页器返回分页数据&#xff08;APIView&#xff0c;action场景&#xff09; 在使用 Django REST Framework&#xff08;DRF&#xff09;时&#xff0c;很多人习惯了用 GenericAPIView 或 ViewSet 自动帮我们处理分页。但在某些场景中…...

AI知识梳理——RAG、Agent、ReAct、LangChain、LangGraph、MCP、Function Calling、JSON-RPC

AI技术I AI技术II RAG &#x1f4cc; 高度凝练表达 RAG &#xff08;检索增强生成&#xff09;是一种结合信息检索与生成式人工智能的技术框架&#xff0c;旨在提升大型语言模型&#xff08;LLM&#xff09;的输出准确性和实用性。通过在生成响应前引入外部知识库的信息&#…...

Vue组件通信方式及最佳实践

1. Props / 自定义事件 (父子通信) 使用场景 父子组件直接数据传递 代码实现 <!-- Parent.vue --> <template><Child :message"parentMsg" update"handleUpdate" /> </template><script setup> import { ref } from vue…...

【实用教程】如何快速搭建一套私有的埋点系统?

这篇教程将基于开源项目-ClkLog&#xff0c;教大家快速搭建一套自有的埋点系统&#xff0c;从0开始完成数据采集、分析与展示&#xff0c;全流程掌控用户行为数据。 ClkLog是一款支持私有化部署的全开源用户行为数据采集与分析系统&#xff0c;兼容Web、App、小程序多端埋点&am…...

深入解析 Uniswap:自动做市商模型的数学推导与智能合约架构

目录 1. 自动做市商&#xff08;AMM&#xff09;模型的数学推导1.1 恒定乘积公式推导1.2 价格影响与滑点 2. Uniswap 智能合约架构解析2.1 核心合约&#xff08;Core&#xff09;2.1.1 工厂合约&#xff08;Factory&#xff09;2.1.2 交易对合约&#xff08;Pair&#xff09; 2…...

spring配置并使用rabbitmq

本篇博客默认你已经成功安装了rabbitmq。如未安装&#xff0c;可参照官网https://www.rabbitmq.com/docs/platforms&#xff0c;选择对应平台进行安装 一、虚拟主机 虚拟主机的概念&#xff1a; 虚拟主机不是指vm虚拟机&#xff0c;而是指一个通过逻辑进行资源隔离和分区的机…...

Android开发——不同布局的定位属性 与 通用属性

目录 不同布局的定位属性1. 线性布局&#xff08;LinearLayout&#xff09;2. 相对布局&#xff08;RelativeLayout&#xff09;3. 约束布局&#xff08;ConstraintLayout&#xff09;4. 表格布局&#xff08;TableLayout&#xff09;5. 网格布局&#xff08;GridLayout&#x…...

React 19版本refs也支持清理函数了。

文章目录 前言一、refs 支持清理函数二、案例演示1.useEffect写法2.React 19改进 的ref写法 总结 前言 React 19版本发布了ref支持清理函数了&#xff0c;这样就可以达到useEffect一样的效果了。为啥需要清理函数呢&#xff0c;这是因为节约内存。 清理事件监听&#xff08;避…...

Python高效网络爬虫开发指南

Python 网络爬虫入门与实战 一、引言 随着互联网数据的爆炸性增长&#xff0c;获取和分析这些数据变得越来越重要。网络爬虫作为数据采集的重要工具&#xff0c;在这其中扮演了不可或缺的角色。 二、环境搭建 首先我们需要安装Python环境以及一些必要的库&#xff1a; req…...

Python爬虫实战:获取国家统计网最新消费数据并分析,为从业者做参考

一、系统定义与架构设计 1.1 系统定义 本系统基于 Python 爬虫技术构建,实现国家数据网消费数据的自动化获取、清洗、分析及可视化。通过定义标准化的数据采集流程、反爬策略、数据分析模型,为经济研究、行业分析等场景提供数据支持。 1.2 架构设计 数据采集层 --> 数据…...

Python中使用uv创建环境及原理详解

Python中使用uv创建环境及原理详解 摘要&#xff1a;本文详细介绍uv工具的功能、安装及使用方法&#xff0c;重点阐述如何利用uv创建和管理Python环境&#xff0c;并深入分析其工作原理。uv作为一款高性能的Python包和项目管理器&#xff0c;凭借其快速的依赖解析、内置虚拟环…...

阿尔泰科技助力电厂——520为爱发电!

当城市的霓虹在暮色中亮起&#xff0c;当千万个家庭在温暖中共享天伦&#xff0c;总有一群默默的 "光明守护者" 在幕后坚守 —— 它们是为城市输送能量的电厂&#xff0c;更是以科技赋能电力行业的阿尔泰科技。值此 520 爱意满满的日子&#xff0c;阿尔泰科技用硬核技…...

【Golang笔记02】函数、方法、泛型、接口学习笔记

Golang笔记02&#xff1a;函数、方法、泛型、接口学习笔记 一、进阶学习 1.1、函数 go中的函数使用func关键字进行定义&#xff0c;go程序的入口函数叫做&#xff1a;main&#xff0c;并且必须是属于main包里面。 1.1.1、定义函数 &#xff08;1&#xff09;普通函数 go中…...

C#语法篇 :基类子类转换,成员变化情况

在C#中&#xff0c;会有从子类对象到基类对象的转换&#xff0c;这属于C#中的向上扩容&#xff0c;一般可以默认转换。 方法的转换 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace ex3._4 …...

【漫话机器学习系列】264.内距(又称四分位差)Interquartile Range

深入理解内距&#xff08;Interquartile Range&#xff0c;IQR&#xff09;——数据分析中的异常值利器 在日常的数据分析中&#xff0c;我们经常需要识别和处理异常值&#xff08;Outliers&#xff09;&#xff0c;而内距&#xff08;Interquartile Range&#xff0c;简称 IQR…...

海外盲盒系统开发:重构全球消费体验的科技引擎

当盲盒文化席卷全球&#xff0c;海外盲盒系统开发已成为重构消费体验的核心赛道。数据显示&#xff0c;2025年全球盲盒市场规模突破120亿&#xff0c;东南亚市场年增长率达4540。我们开发的海外盲盒系统&#xff0c;以技术创新为驱动&#xff0c;打造覆盖全链路的全球化解决方案…...

高噪声下扩展边缘检测算子对检测边缘的影响

目录 一、常见的边缘检测算子 二、扩展边缘检测算子对检测边缘的影响 三、结论 一、常见的边缘检测算子 Sobel 算子: Prewitt算子;...

vuejs处理后端返回数字类型精度丢失问题

标题问题描述 后端返回数据有5.00和3.30这种数据&#xff0c;但是前端展示的时候返回对应分别为5和3.0&#xff0c;小数点后0都丢失了。 接口返回数据展示network-Response&#xff1a; 接口返回数据展示network-Preview&#xff1a; 错误数据效果展示 发现问题 浏览器接口…...

mysql数据库-中间件MyCat

1. MyCat简介 在整个 IT 系统架构中&#xff0c;数据库是非常重要&#xff0c;通常又是访问压力较大的一个服务&#xff0c;除了在程序开发的本身做优化&#xff0c;如&#xff1a; SQL 语句优化、代码优化&#xff0c;数据库的处理本身优化也是非常重要的。主从、热备、分表分…...

手搓四人麻将程序

一、麻将牌的表示 在麻将游戏中&#xff0c;总共有一百四十四张牌&#xff0c;这些牌被分为多个类别&#xff0c;每个类别又包含了不同的牌型。具体来说&#xff0c;麻将牌主要包括序数牌、字牌和花牌三大类。序数牌中&#xff0c;包含有万子、条子和筒子&#xff0c;每种花色…...

PotPlayer 安装 madVR、LAV Filters 以提升解码能力和视频音频效果

PotPlayer自带的解码器并不是最好&#xff0c;如下两张截图都是出自 TOP GUN: Maverick 较暗、灰蒙蒙的一张&#xff0c;是安装插件之前明亮的一张&#xff0c;是安装插件之后 详细安装参考 https://www.bilibili.com/video/BV1UV5qzuE74?spm_id_from333.788.videopod.sectio…...

阿里云域名 绑定 华为云服务器ip

阿里云&#xff08;万网&#xff09;域名转入华为云域名图文教程 上-云社区-华为云 阿里云&#xff08;万网&#xff09;域名转入华为云域名图文教程-云社区-华为云 在阿里云备案了域名&#xff0c;解析到华为云服务器&#xff0c;不在同一个服务商这样可以使用么&#xff1f…...

windows7安装node18

1、要安装windows系统更新 2、node下载&#xff1a;CNPM Binaries Mirror 3、下载zip安装包&#xff0c;解压后将目录添加至环境变量。 4、node -v检查是否成功安装。 5、npm install -g vite4.0.0...

Maven配置安装

&#xff08;2025.1.27&#xff09;最新版MAVEN的安装和配置教程&#xff08;超详细&#xff09;_maven安装-CSDN博客...