【iOS】——仿写计算器

news2024/11/16 3:45:10

文章目录

  • 一、实现思路
  • 二、实现方法
  • 三、判错处理


一、实现思路

先搭建好MVC框架,接着在各个模块中实现各自的任务。首先要创建好UI界面,接着根据UI界面的元素来与数据进行互动,其中创建UI界面需要用到Masonry布局。
请添加图片描述

二、实现方法

在calculationView文件中只涉及到UI界面。
首先创建两个UITextField对象,一个用来显示输入到表达式,一个用来输出运算结果,接着创建一个数组用来存放button的名称,接着循环创建button对象即可。

创建button时需要设置button的tag属性,因为后面需要通过button的tag属性来进行逻辑判断。

self.textField01 = [[UITextField alloc] init];
self.textField01.backgroundColor = [UIColor blackColor];
self.textField02 = [[UITextField alloc] init];
self.textField02.backgroundColor = [UIColor blackColor];
self.buttonArray = [NSMutableArray arrayWithObjects:@"AC", @"(", @")", @"/", @"1", @"2", @"3", @"+", @"4", @"5", @"6", @"-", @"7", @"8", @"9", @"*",  @"0", @".", @"=", nil];
 for (int i = 0; i < 4; i++) {
        for (int j = 0; j < 4; j++) {
            self.calculatorButton = [UIButton buttonWithType:UIButtonTypeCustom];
            NSString* buttonString = self.buttonArray[i * 4 + j];
            [self.calculatorButton setTitle:buttonString forState:UIControlStateNormal];
            self.calculatorButton.titleLabel.font = [UIFont systemFontOfSize:43];
            self.calculatorButton.titleLabel.textColor = [UIColor whiteColor];
            self.calculatorButton.backgroundColor = [UIColor grayColor];
            [self.calculatorButton addTarget:self action:@selector(pressButton:) forControlEvents:UIControlEventTouchUpInside];
            self.calculatorButton.layer.borderWidth = 2.0;
            self.calculatorButton.layer.cornerRadius = ButtonSize / 2;
            self.calculatorButton.layer.masksToBounds = YES;
            self.calculatorButton.tag = 100 + j + i*4;
            if (i == 0 && j < 3) {
                self.calculatorButton.backgroundColor = [UIColor colorWithWhite:0.6 alpha:0.9];
                self.calculatorButton.titleLabel.textColor = [UIColor blackColor];
            }
            if (j == 3) {
                self.calculatorButton.backgroundColor = [UIColor orangeColor];
            }
            [self addSubview:self.calculatorButton];
            [self.calculatorButton mas_makeConstraints:^(MASConstraintMaker *make) {
                make.top.equalTo(self).offset(90 + HEIGHT / 6 + HEIGHT / 8 + (ButtonSize + SideSize) * i);
                make.left.equalTo(self).offset(SideSize + (ButtonSize + SideSize) * j);
                make.width.equalTo(@ButtonSize);
                make.height.equalTo(@ButtonSize);
            }];
        }
    }

还需要在当前文件触发button的事件函数,因为要通过button的事件函数将当前button传值到CalculationViewController文件中。

我使用的协议传值进行

定义协议及协议方法

@protocol ButttonDelegate <NSObject>

- (void)returnButton:(UIButton*)button;

@end

定义代理对象

@property (nonatomic, weak)id<ButttonDelegate>delegate;

button事件函数触发代理方法

- (void)pressButton:(UIButton*)button {
    [self.delegate returnButton:button];
}

在CalculationViewController中定义协议方法

- (void)returnButton:(UIButton*)button;

在CalculationViewController中设置代理对象

 self.calculationView.delegate = self;

在CalculationViewController中实现协议方法

- (void)returnButton:(UIButton*)button {
- NSString* buttonStr = button.titleLabel.text;
    NSInteger buttonTag = button.tag;
    if (buttonTag == 100) {
        self.calculationView.textField01.text = @"";
        self.calculationView.textField02.text = @"";
        self.yunsuanStr = [[NSMutableString alloc] init];
        self.jieguoStr = [[NSString alloc] init];
    } else if (buttonTag == 118) {
    	//四则运算
    }
	//
}

对于四则运算,我使用中缀表达式转后缀表达式,再利用后缀表达式进行计算。中缀转后缀定义一个符号栈,一个结果栈。

		char resultStack[100];
		int resultTop = -1;
		char operationStack[100];
		int operationTop = -1;
 		NSString* ocStr =  self.yunsuanStr;
        const char* strs = [ocStr UTF8String];
        char str[1000] ;
        strcpy(str, strs) ;
        int length = (int)strlen(str);

后缀进行四则运算时定义一个新栈。

			double Stack[100];
            int StackTop = -1;
              char* token = strtok(resultStack, " ");
            while (token != NULL) {
                if (strcmp(token, "+") == 0) {
                    double a = Stack[StackTop--];
                    double b = Stack[StackTop--];
                    double c = b + a;
                    Stack[++StackTop] = c;
                }
                else if (strcmp(token, "-") == 0) {
                    double a = Stack[StackTop--];
                    double b = Stack[StackTop--];
                    double c = b - a;
                    Stack[++StackTop] = c;
                }
                else if (strcmp(token, "*") == 0) {
                    double a = Stack[StackTop--];
                    double b = Stack[StackTop--];
                    double c = b * a;
                    Stack[++StackTop] = c;
                }
                else if (strcmp(token, "/") == 0) {
                    double a = Stack[StackTop--];
                    double b = Stack[StackTop--];
                    double c = b / a;
                    Stack[++StackTop] = c;
                }
                else if (strcmp(token, "!") == 0) {
                    double c = Stack[StackTop--];
                    c = -c;
                    Stack[++StackTop] = c;
                }
                else {
                    double num = atof(token);
                    Stack[++StackTop] = num;
                }

                token = strtok(NULL, " ");
            }
            double result = Stack[StackTop--];

这里使用了atof()函数可以将字符串中的数字识别为double类型的浮点数。使用strtok()函数将结果栈的元素通过空格分割方便进行识别和运算。

对于负数的运算,我是将与“(”相邻的“-”转换为“!”,并用“!”进行负数转变,其中需要将“!“的优先级设置为最高

int Compare(char str) {
    if (str == '(' || str == ')') {
        return 0;
    }
    else if (str == '+' || str == '-') {
        return 1;
    }
    else if (str == '*' || str == '/') {
        return 2;
    } else if (str == '!') {
        return 3;
    } else {
        return -1;
    }
}

接着在输入字符串时进行识别

for (int i = 0; i < length; ++i) {
                if (str[i] == '(' && str[i + 1] == '-') {
                    str[i + 1] = '!';
                }
            }

三、判错处理

对于括号匹配问题,我使用栈来处理,当读到“(”则入栈,读到“)”则栈顶元素出栈,最后判断栈是否为空即可。

- (int)kuoHaoMatch:(NSMutableString*)mutableString {
    const char* cstr = [mutableString UTF8String];
    int length = (int)strlen(cstr);
    char stack[100];
    int stackTop = -1;
    for (int i = 0; i < length; i++) {
        if (cstr[i] == '(') {
            stack[++stackTop] = cstr[i];
        }
        if (cstr[i] == ')') {
            stack[stackTop--];
        }
    }
    if (stackTop == -1) {
        return 1;
    }
    return 0;
}

对于符号匹配问题,两个运算符不能相邻,第一个位置不能为运算符(-除外),最后一个位置不能为运算符

- (int)fuHaoMatch:(NSMutableString*)mutableString {
    const char* cstr = [mutableString UTF8String];
    int length = (int)strlen(cstr);
    if (cstr[0] == '*' || cstr[0] == '/' || cstr[0] == '+') {
        return 0;
    }
    for (int i = 0; i < length; i++) {
        if ((cstr[i] == '*' && cstr[i+1] == '/') || (cstr[i] == '/' && cstr[i+1] == '*')) {
            return 0;
        }
        if ((cstr[i] == '*' && cstr[i+1] == '-') || (cstr[i] == '*' && cstr[i+1] == '+')) {
            return 0;
        }
        if ((cstr[i] == '/' && cstr[i+1] == '-') || (cstr[i] == '/' && cstr[i+1] == '+')) {
            return 0;
        }
        if ((cstr[i] == '+' && cstr[i+1] == '/') || (cstr[i] == '+' && cstr[i+1] == '*')) {
            return 0;
        }
        if ((cstr[i] == '-' && cstr[i+1] == '/') || (cstr[i] == '-' && cstr[i+1] == '*')) {
            return 0;
        }
        if ((cstr[i] == '-' && cstr[i+1] == '-') || (cstr[i] == '-' && cstr[i+1] == '+')) {
            return 0;
        }
        if ((cstr[i] == '+' && cstr[i+1] == '+') || (cstr[i] == '+' && cstr[i+1] == '-')) {
            return 0;
        }
    }
    if (cstr[length - 1] == '+' || cstr[length - 1] == '-' ||cstr[length - 1] == '*' || cstr[length - 1] == '/') {
        return 0;
    }
    return 1;
}

对于数字和括号的相邻问题,数字后不能直接跟左括号,右括号不能直接跟数字

- (int)numberAndkuoHao:(NSMutableString*)mutableString {
    const char* cstr = [mutableString UTF8String];
    int length = (int)strlen(cstr);
    int flag = 0;
    for (int i = 0; i < length; i++) {
        if ((cstr[i]>= '0' && cstr[i] <= '9') &&cstr[i+1] == '(') {
            flag++;
        }
        if (cstr[i] == ')' && (cstr[i + 1]>= '0' && cstr[i + 1] <= '9')) {
            flag++;
        }
    }
    if (flag == 0) {
        return 1;
    }
    return 0;
}

对于只有括号没有数字的问题,直接遍历整个字符串如果有数字则标志变量加一,最后判断标志变量

- (int)numberJudge:(NSMutableString*)mutableString {
    const char* cstr = [mutableString UTF8String];
    int length = (int)strlen(cstr);
    int flag = 0;
    for (int i = 0; i < length; i++) {
        if (cstr[i]>= '0' && cstr[i] <= '9') {
            flag++;
        }
    }
    if (flag != 0) {
        return 1;
    }
    return 0;;
}

对于除零问题

if ([self.jieguoStr isEqualToString:@"nan"]) {
                self.jieguoStr = @"错误";
            }
            if ([self.jieguoStr isEqualToString:@"inf"]) {
                self.jieguoStr = @"错误";
            }

对于小数点异常处理

int pointFlag = 0;
for (int i = 0; i < resultTop; i++) {
                if (resultStack[i] == '.') {
                    for (int j = i + 1; resultStack[j] != ' '; j++) {
                        if (resultStack[j] == '.') {
                            pointFlag++;
                        }
                    }
                }
            }
             if (pointFlag != 0) {
                self.jieguoStr = @"错误";
                pointFlag = 0;
            }

对于如何删除多余的零

NSDecimalNumber *number = [NSDecimalNumber decimalNumberWithString: self.jieguoStr];
            NSDecimalNumberHandler *roundHandler = [NSDecimalNumberHandler decimalNumberHandlerWithRoundingMode:NSRoundUp scale:8 raiseOnExactness:NO raiseOnOverflow:NO raiseOnUnderflow:NO raiseOnDivideByZero:NO];
            NSDecimalNumber *roundedNumber = [number decimalNumberByRoundingAccordingToBehavior:roundHandler];
            self.jieguoStr = [roundedNumber stringValue];

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1067222.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

糖基转移酶特异性 -bg

糖基转移酶特异性 大多数糖基转移酶对其供体和受体底物都具有高度的特异性&#xff0c;这导致Saul Roseman及其同事提出了“一种酶 - 一种连接”假说。人B血型α1-3半乳糖基转移酶就是这一概念的例证。该酶催化糖基化反应&#xff0c;其中半乳糖以α键添加到受体底物上半乳糖残…

[0xGame 2023] week1

整理一下&#xff0c;昨天该第二周了。今天应该9点结束提交&#xff0c;等我写完就到了。 PWN 找不到且不对劲的flag 第1题是个nc测试&#xff0c;但也不完全是&#xff0c;因为flag在隐含目录里 高端的syscall 程序使用了危险函数&#xff0c;并且没有canary阻止&#xff0…

后端解决跨域(极速版)

header(Access-Control-Allow-Origin: *); header(Access-Control-Allow-Methods:*); 代表接收全部的请求&#xff0c;"POST,GET"//允许访问的方式 指定域&#xff0c;如http://172.20.0.206//宝塔的域名&#xff0c;注意不是&#xff1a;http://wang.jingyi.icu等…

文件上传笔记

一、上传的简单绕过&#xff1a; 1、若是上传的文件只在前端的代码中进行了过滤&#xff1a; &#xff08;1&#xff09;可以直接在开发者工具中删除相关代码&#xff1a; &#xff08;2&#xff09;也可以通过 burpsuite 绕过: 上传时&#xff0c;先提前修改 php 文件的后缀…

QT之可自由折叠和展开的布局

介绍和功能分析 主要是实现控件的折叠和展开&#xff0c;类似抽屉控件&#xff0c;目前Qt自带的控件QToolBox具有这个功能&#xff0c;但是一次只能展开一个&#xff0c;所以针对自己的需求可以自己写一个类似的功能&#xff0c;这里实现的方法比较多&#xff0c;其实原理也比较…

upload文件上传

参数格式 <el-upload :http-request"upload" class"upload" :show-file-list"false" :on-success"handleUploadSuccess"><el-button size"default" type"primary" style"margin:0 20px;">上…

分类预测 | MATLAB实现POA-CNN鹈鹕算法优化卷积神经网络多特征分类预测

分类预测 | MATLAB实现POA-CNN鹈鹕算法优化卷积神经网络多特征分类预测 目录 分类预测 | MATLAB实现POA-CNN鹈鹕算法优化卷积神经网络多特征分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现POA-CNN鹈鹕算法优化卷积神经网络多特征分类预测&#xff0…

深入解读redis的zset和跳表【源码分析】

1.基本指令 部分指令&#xff0c;涉及到第4章的api&#xff0c;没有具体看实现&#xff0c;但是逻辑应该差不多。 zadd <key><score1><value1><score2><value2>... 将一个或多个member元素及其score值加入到有序集key当中。根据zslInsert zran…

02.Cesium源码编译及搭建开发环境

开始之前&#xff0c;默认你已经掌握了一定的前端知识&#xff0c;文章中用到的一些前端知识不再展开解释&#xff0c;如果你有不明白的地方&#xff0c;请自行学习。 另外&#xff0c;本篇文章及后续的文章首先会使用原生JS的方式 进行实例的开发&#xff0c;Vue版本会在后期文…

办公技巧:Excel日常高频使用技巧

目录 1. 快速求和&#xff1f;用 “Alt ” 2. 快速选定不连续的单元格 3. 改变数字格式 4. 一键展现所有公式 “CTRL ” 5. 双击实现快速应用函数 6. 快速增加或删除一列 7. 快速调整列宽 8. 双击格式刷 9. 在不同的工作表之间快速切换 10. 用F4锁定单元格 1. 快速求…

快速掌握批量合并视频

在日常的工作和生活中&#xff0c;我们经常需要对视频进行编辑和处理&#xff0c;而合并视频、添加文案和音频是其中常见的操作。如何快速而简便地完成这些任务呢&#xff1f;今天我们介绍一款强大的视频编辑软件——“固乔智剪软件”&#xff0c;它可以帮助我们轻松实现批量合…

ACE综述

1、ACE综述 ACE自适配通信环境&#xff08;ADAPTIVE Communication Environment&#xff09;是可自由使用、开放源码的面向对象&#xff08;OO&#xff09;框架&#xff08;framework&#xff09;&#xff0c;它实现了许多用于并发通信软件的核心模式。ACE提供了一组丰…

【juc】countdownlatch实现游戏进度

目录 一、截图示例二、代码示例 一、截图示例 二、代码示例 package com.learning.countdownlatch;import java.util.Arrays; import java.util.Random; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurr…

C++:模板(非类型模板参数、类模板的特化、模板的分离编译)

本文是 C&#xff1a;模板&#xff08;函数模板、类模板&#xff09; 该文的进阶部分&#xff0c;主要介绍模板非类型模板参数、类模板的特化、模板的分离编译这三部分。 目录 一、非类型模板参数 二、模板的特化 1.概念 2.函数模板特化 3.类模板特化 1.全特化 2.偏特…

Lua系列文章(1)---Lua5.4参考手册学习总结

windows系统上安装lua,下载地址&#xff1a; Github 下载地址&#xff1a;https://github.com/rjpcomputing/luaforwindows/releases 可以有一个叫SciTE的IDE环境执行lua程序 1 – 简介 Lua 是一种强大、高效、轻量级、可嵌入的脚本语言。 它支持过程编程&#xff0c; 面向对…

VScode配置Jupyter

环境 安装步骤 1、插件安装 2、更改pip加速源 pip config set global.index-url https://pypi.tuna.tsinghua.edu.cn/simple 参考&#xff1a;vscode python配置pip源 ​​​​​​​ 【Python学习】Day-00 Python安装、VScode安装、pip命令、镜像源配置、虚拟环境 3、建…

202. 最幸运的数字

202. 最幸运的数字 - AcWing题库 #include<bits/stdc.h> #define IOS ios::sync_with_stdio(0);cin.tie(0);cout.tie(0); #define endl \nusing namespace std;typedef pair<int, int> PII; typedef long long ll; typedef long double ld;ll qmi(ll a, ll k, ll m…

集中发现服务DCPSInfoRepo通信端口和ORB交互流程

OpenDDS集中发现服务DCPSInfoRepo,为OpenDDS的pub和sub通信终端提供主题匹配和通信协商和中介服务,是基于TAO的ORB机制完成的,GIOP协议。 1、集中发现服务DCPSInfoRepo的相关通信端口 1)集中发现服务DCPSInfoRepo通信端口 DCPSInfoRepo -ORBListenEndpoints iiop://192.…

RK3568的CAN驱动适配

目录 背景&#xff1a; 1.内核驱动模块配置 2.设备树配置 3.功能测试 4.bug修复 背景&#xff1a; 某个项目上使用RK3568的芯片&#xff0c;需要用到4路CAN接口进行通信&#xff0c;经过方案评审后决定使用RK3568自带的3路CAN外加一路spi转的CAN实现功能&#xff0c;在这个…

SMT求解器Q3B——在WSL上的Docker配置

SMT求解器Q3B——在WSL上的Docker配置 1、配置wsl下的Docker2、在github上下载Q3B3、更换配置文件4、安装docker镜像5、编译Q3B6、使用Q3B 1、配置wsl下的Docker WSL 2 上的 Docker 远程容器入门 2、在github上下载Q3B Q3B下载地址 3、更换配置文件 下载完后&#xff0c;将…