【Qt应用开发】复刻经典:基于Qt实现Windows风格计算器(加减乘除、删除、归零功能全解析)

简介: 在Qt中,"栈"的概念主要体现在两个层面:一是程序设计中的数据结构——栈(Stack),二是用户界面管理中的QStackedWidget控件。下面我将分别简要介绍这两个方面:


目录

展示效果:

1. 数据结构:栈

算法思路概览

思路图

代码段:


展示效果:

image.gif 编辑

在Qt中,"栈"的概念主要体现在两个层面:一是程序设计中的数据结构——栈(Stack),二是用户界面管理中的QStackedWidget控件。下面我将分别简要介绍这两个方面:

1. 数据结构:栈

尽管Qt本身并不直接提供一个名为"Stack"的容器类,但你可以使用C++标准库中的std::stack,或者利用数组、向量等容器配合特定的操作来模拟栈的行为。栈是一种后进先出(Last In, First Out, LIFO)的数据结构,主要操作有压栈(push)、弹栈(pop)和查看栈顶元素。

算法思路概览

本算法利用栈的先进后出特性,通过分离数字和运算符,实现复杂表达式的高效解析与计算。具体步骤如下:

  1. 初始化栈结构
  • 数字栈:用于存储待计算的数值。
  • 运算符栈:存储运算符,初始时包含一个等于号(=),作为终止计算的标志。
  1. 解析输入字符串
  • 将输入的表达式视为由数字和运算符组成的序列。
  • 逐字符扫描输入字符串,识别数字和运算符。
  1. 数字栈操作
  • 遇到数字时,连续读取构成整数的所有字符,直至遇到运算符或字符串结束。
  • 将形成的整数压入数字栈。
  1. 运算符栈操作
  • 当遇到运算符时,与栈顶运算符比较优先级。
  • 若栈外运算符优先级高于栈顶运算符,将其压入运算符栈。
  • 若低于或等于栈顶运算符优先级,执行以下步骤:
  • 弹出运算符栈顶的运算符。
  • 从数字栈弹出两个数值(后弹出的为右操作数,先弹出的为左操作数)。
  • 根据弹出的运算符执行相应的数学运算。
  • 将计算结果重新压入数字栈。
  • 重复上述步骤,直至栈外运算符被压入栈或运算符栈为空。
  1. 终止条件
  • 当扫描至字符串末尾或遇到等于号(=)时,进行最终的计算,直至运算符栈清空。

思路图

我个人认为最主要的还是利用了栈这个先进后出的特性,可以很好的匹配这个算法的需求。

将输入的内容看为一串字符串,遍历字符串寻找自己所寻求的字符,利用C++面向对象的特性类型转变起来也更加容易

image.gif 编辑

代码段:

#include "widget.h"
#include "ui_widget.h"
//按键优化
#define DEFULT_BUTTON_STYLE "\
QPushButton{\
color:#000000;\
border:1px solid #AAAAAA;\
    border-radius:0;\
    background-color:#FFFFFF;\
}\
QPushButton:pressed{\
color:#FFFFFF;\
    background-color:#AAAAAA;\
}\
"
Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    this->setFixedSize(415,489);
    this->setWindowTitle("计算器");
    temp = '0';
    ui->Text_2->setPlainText(temp);
    ui->zerp_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->one_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->two_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->three_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->four_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->five_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->six_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->seven_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->eight_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->nine_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->sum_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->sub_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->max_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->mini_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->del_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->equal_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->C_2->setStyleSheet(DEFULT_BUTTON_STYLE);
    ui->lable_2->setStyleSheet(DEFULT_BUTTON_STYLE);
}
Widget::~Widget()
{
    delete ui;
}
//拆分字符
double Widget::splitNum(QString str,int &i)
{
    QString temp;
    double num;
    while(str[i] >= '0' && str[i] <= '9')
    {
        temp += str[i++];
    }
    num = temp.toDouble();
    return num;
}
//栈内优先级
int Widget::inprior(char ch)
{
    int in = 0;
    switch (ch)
    {
    case '=':
        break;
    case '+':
        in = 3;
        break;
    case '-':
        in = 3;
        break;
    case '*':
        in = 5;
        break;
    case '/':
        in = 5;
        break;
    default:
        break;
    }
    return in;
}
//栈外优先级
int Widget::outprior(char str)
{
    int out = 0;
    switch (str)
    {
    case '=':
        break;
    case '+':
        out = 2;
        break;
    case '-':
        out = 2;
        break;
    case '*':
        out = 4;
        break;
    case '/':
        out = 4;
        break;
    default:
        break;
    }
    return out;
}
//比较优先级
int Widget::cmpprior(int in, int out)
{
    if(in > out)
    {
        return 1;
    }
    else if (in < out)
    {
        return -1;
    }
    else
    {
        return 0;
    }
}
//计算公式
int Widget::caculator(int num1, int num2, char op)
{
    switch (op)
    {
    case '+':
        return num1 + num2;
        break;
    case '-':
        return num2 - num1;
        break;
    case '*':
        return num1 * num2;
        break;
    case '/':
        return num2 / num1;
        break;
    default:
        break;
    }
}
//等于
void Widget::on_equal_2_clicked()
{
    int i = 0;
    int num = 0;
    int num1 = 0,num2 = 0;
    int result = 0;
    char top,del,buf;
    optr.clear();
    numtr.clear();
    optr.push('=');
    infix = ui->Text_2->toPlainText();
    if(infix.isEmpty())
        return;
    while(!optr.empty())
    {
        if(infix[i] == '0' ||infix[i] == '1' ||infix[i] == '2' ||infix[i] == '3' ||infix[i] == '4' ||infix[i] == '5' ||infix[i] == '6' ||infix[i] == '7' ||infix[i] == '8' ||infix[i] == '9')
        {
            num = splitNum(infix,i);
            numtr.push(num);
        }
        else
        {
            top = optr.top();
            switch (cmpprior(inprior(top),outprior(infix[i].toLatin1())))
            {
            case 1:
                buf = optr.pop();
                num1 = numtr.pop();
                num2 = numtr.pop();
                numtr.push(caculator(num1,num2,buf));
                break;
            case -1:
                optr.push(infix[i].toLatin1());
                i++;
                break;
            case 0:
                del = optr.pop();
                if(del != '=')
                    i++;
                break;
            default:
                break;
            }
        }
        result = numtr.top();
        if(result == 0)
        {
            qDebug("运算错误");
            return;
        }
        ui->lineEdit_2->setText(QString::number(result));
    }
}
//0
void Widget::on_zerp_2_clicked()
{
    infix += '0';
    ui->Text_2->setPlainText(infix);
}
//1
void Widget::on_one_2_clicked()
{
    infix += '1';
    ui->Text_2->setPlainText(infix);
}
//2
void Widget::on_two_2_clicked()
{
    infix += '2';
    ui->Text_2->setPlainText(infix);
}
//3
void Widget::on_three_2_clicked()
{
    infix += '3';
    ui->Text_2->setPlainText(infix);
}
//4
void Widget::on_four_2_clicked()
{
    infix += '4';
    ui->Text_2->setPlainText(infix);
}
//5
void Widget::on_five_2_clicked()
{
    infix += '5';
    ui->Text_2->setPlainText(infix);
}
//6
void Widget::on_six_2_clicked()
{
    infix += '6';
    ui->Text_2->setPlainText(infix);
}
//7
void Widget::on_seven_2_clicked()
{
    infix += '7';
    ui->Text_2->setPlainText(infix);
}
//8
void Widget::on_eight_2_clicked()
{
    infix += '8';
    ui->Text_2->setPlainText(infix);
}
//9
void Widget::on_nine_2_clicked()
{
    infix += '9';
    ui->Text_2->setPlainText(infix);
}
//+
void Widget::on_sum_2_clicked()
{
    infix += '+';
    ui->Text_2->setPlainText(infix);
}
//-
void Widget::on_sub_2_clicked()
{
    infix += '-';
    ui->Text_2->setPlainText(infix);
}
//*
void Widget::on_max_2_clicked()
{
    infix += '*';
    ui->Text_2->setPlainText(infix);
}
//除法
void Widget::on_mini_2_clicked()
{
    infix += '/';
    ui->Text_2->setPlainText(infix);
}
//.
void Widget::on_lable_2_clicked()
{
    infix += '.';
    ui->Text_2->setPlainText(infix);
}
//归零
void Widget::on_C_2_clicked()
{
    temp = '0';
    infix.clear();
    ui->Text_2->setPlainText(temp);
}
//删除
void Widget::on_del_2_clicked()
{
    infix.remove(infix.length()-1,1);
    if(infix.length() == 0)
    {
        temp = '0';
        infix.clear();
        ui->Text_2->setPlainText(temp);
        return;
    }
    ui->Text_2->setPlainText(infix);
}

image.gif

#include "widget.h"
#include <QApplication>
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Widget w;
    w.show();
    return a.exec();
}

image.gif

#ifndef WIDGET_H
#define WIDGET_H
#include <QtWidgets>
#include <QStack>
namespace Ui {
class Widget;
}
class Widget : public QWidget
{
    Q_OBJECT
public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
    double splitNum(QString,int &i);//拆分数字
    int inprior(char ch);//栈内优先级
    int outprior(char ch);//栈外优先级
    int cmpprior(int in,int out);//比较优先级
    int caculator(int num1,int num2,char op);
private slots:
    void on_zerp_2_clicked();
    void on_one_2_clicked();
    void on_two_2_clicked();
    void on_three_2_clicked();
    void on_four_2_clicked();
    void on_five_2_clicked();
    void on_six_2_clicked();
    void on_seven_2_clicked();
    void on_eight_2_clicked();
    void on_nine_2_clicked();
    void on_sum_2_clicked();
    void on_sub_2_clicked();
    void on_max_2_clicked();
    void on_mini_2_clicked();
    void on_lable_2_clicked();
    void on_C_2_clicked();
    void on_del_2_clicked();
    void on_equal_2_clicked();
private:
    Ui::Widget *ui;
    QStack<char> optr;
    QStack<double> numtr;
    QString infix;
    QString temp;
};
#endif // WIDGET_H

image.gif


相关文章
|
29天前
|
监控 安全 开发工具
鸿蒙HarmonyOS应用开发 | HarmonyOS Next-从应用开发到上架全流程解析
HarmonyOS Next是华为推出的最新版本鸿蒙操作系统,强调多设备协同和分布式技术,提供丰富的开发工具和API接口。本文详细解析了从应用开发到上架的全流程,包括环境搭建、应用设计与开发、多设备适配、测试调试、应用上架及推广等环节,并介绍了鸿蒙原生应用开发者激励计划,帮助开发者更好地融入鸿蒙生态。通过DevEco Studio集成开发环境和华为提供的多种支持工具,开发者可以轻松创建并发布高质量的鸿蒙应用,享受技术和市场推广的双重支持。
307 11
|
1月前
|
数据可视化 数据挖掘 BI
团队管理者必读:高效看板类协同软件的功能解析
在现代职场中,团队协作的效率直接影响项目成败。看板类协同软件通过可视化界面,帮助团队清晰规划任务、追踪进度,提高协作效率。本文介绍看板类软件的优势,并推荐五款优质工具:板栗看板、Trello、Monday.com、ClickUp 和 Asana,助力团队实现高效管理。
52 2
|
6天前
|
搜索推荐 UED Python
实现一个带有昼夜背景切换的动态时钟:从代码到功能解析
本文介绍了一个使用Python和Tkinter库实现的动态时钟程序,具有昼夜背景切换、指针颜色随机变化及整点和半点报时功能。通过设置不同的背景颜色和随机变换指针颜色,增强视觉吸引力;利用多线程技术确保音频播放不影响主程序运行。该程序结合了Tkinter、Pygame、Pytz等库,提供了一个美观且实用的时间显示工具。欢迎点赞、关注、转发、收藏!
123 94
|
29天前
|
物联网 调度 vr&ar
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战
鸿蒙技术分享:HarmonyOS Next 深度解析 随着万物互联时代的到来,华为发布的 HarmonyOS Next 在技术架构和生态体验上实现了重大升级。本文从技术架构、生态优势和开发实践三方面深入探讨其特点,并通过跨设备笔记应用实战案例,展示其强大的分布式能力和多设备协作功能。核心亮点包括新一代微内核架构、统一开发语言 ArkTS 和多模态交互支持。开发者可借助 DevEco Studio 4.0 快速上手,体验高效、灵活的开发过程。 239个字符
215 13
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战
|
28天前
|
自然语言处理 搜索推荐 数据安全/隐私保护
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】
鸿蒙登录页面设计展示了 HarmonyOS 5.0(Next)的未来美学理念,结合科技与艺术,为用户带来视觉盛宴。该页面使用 ArkTS 开发,支持个性化定制和无缝智能设备连接。代码解析涵盖了声明式 UI、状态管理、事件处理及路由导航等关键概念,帮助开发者快速上手 HarmonyOS 应用开发。通过这段代码,开发者可以了解如何构建交互式界面并实现跨设备协同工作,推动智能生态的发展。
144 10
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】
|
6天前
|
人工智能 自然语言处理 搜索推荐
销售易、悟空、神州云动CRM:全方位功能解析与优势特色盘点
销售易CRM、悟空CRM和神州云动CRM各自具备独特的产品功能与优势,适用于不同类型的企业。销售易CRM提供移动化、社交化和AI驱动的全流程管理,适合大型企业及跨国公司;悟空CRM以智能数据分析和移动办公支持见长,适合中大型企业;神州云动CRM则凭借灵活定制和多行业适配能力,特别适合大中型企业。企业在选择时应根据自身需求和发展战略,挑选最适合的CRM系统,以实现客户关系管理的最大化效益。
|
26天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
17天前
|
人工智能 数据处理 C#
AI Dev Gallery:微软开源 Windows AI 模型本地运行工具包和示例库,助理开发者快速集成 AI 功能
微软推出的AI Dev Gallery,为Windows开发者提供开源AI工具包和示例库,支持本地运行AI模型,提升开发效率。
64 13
|
19天前
|
存储 数据库 对象存储
新版本发布:查询更快,兼容更强,TDengine 3.3.4.3 功能解析
经过 TDengine 研发团队的精心打磨,TDengine 3.3.4.3 版本正式发布。作为时序数据库领域的领先产品,TDengine 一直致力于为用户提供高效、稳定、易用的解决方案。本次版本更新延续了一贯的高标准,为用户带来了多项实用的新特性,并对系统性能进行了深度优化。
28 3
|
19天前
|
供应链 数据可视化 数据挖掘
企业服务品牌深度解析:销售易、用友、白码功能与特色对比
在企业服务领域,销售易、用友、白码等品牌凭借独特的产品和解决方案占据重要地位。销售易专注于CRM,提供客户管理、销售自动化、市场营销等功能,提升销售效率与客户满意度。用友作为领先的企业服务提供商,涵盖ERP、财务管理、人力资源管理等,助力企业资源优化配置。白码则以低代码开发平台为核心,支持快速构建业务应用,具备高度可定制化和易于维护的特点。三者各具特色,共同推动企业数字化转型。

推荐镜像

更多