admin管理员组文章数量:1794759
MFC实现浮点/进制转换计算器
简介
VS2019编写的计算器,支持括号和小数运算和进制转换 当进制发生转换时,如果表达式框内有表达式,会计算后转化,支持小数转换,如果表达式有误,将直接清空。
界面 实现方法表达式框添加控制变量CString类型的m_sEquation,除了求值运算其他按键都可以认为是对字符串的操作
下面重点介绍值的运算,为此写了一个MyCalculator类,它有两个成员函数:
bool MyCalculator::Calc(CString Equation, CString &ans, int radix) 传入表达式Equation,和进制radix(默认为十进制),将计算结果存入ans中 当表达式有错误时,如"(1+1++1)","(1+6",函数返回值FALSE,否则返回TRUE 被除数为0时,ans赋值为"除0异常"
为了获取小数,需要下面的GetNum函数,传入表达式Eq,起始位置i,进制radix,注意这个函数不可以直接处理负数
- 对于整数部分,和普通整数变换一样 x = x ∗ 10 + E q [ i ] − ′ 0 ′ x=x*10+Eq[i]-'0' x=x∗10+Eq[i]−′0′,但是现在不一定是十进制,换成 ∗ r a d i *radi ∗radix就可以;
- 判断是否有小数点
- 如果有,小数部分第i位要乘上 1 / r a d i x 1/radix 1/radix的i次方
计算表达式的整体过程是通过表达式栈实现 num数字表达式 opt操作符表达式 因为有负数,已经要判断"2++3"之类错误,不方便直接判断某一位到底是数还是操作符,但是所有表达式都有一个特点:除了括号数字后面一定是操作符,操作符后面一定是数字,左括号左是操作符右是数字,右括号左是数字又是操作符,所以可以写以下循环
w h i l e while while 表达式未处理完:
- 判断左括号,可能没有
- 得到数字(先判断符号再GetNum),入栈
- 判断右括号,如果有,出栈至左括号
- 得到操作符,把操作符栈顶优先级不比它低的出栈,自己入栈
最后所有操作符出栈
这里的出栈指取出操作符和两个操作数,运算后结果数入栈这一整个过程
bool MyCalculator::Calc(CString Equation, CString &ans, int radix) { double num[MAXNUM]; int ntop = 0; char opt[MAXOPT]; int otop = 0; int Len = Equation.GetLength(); for (int i = 0 ; i < Len; ) { if (Equation[i] == '(') { opt[++otop] = Equation[i++]; if (i >= Len) return FALSE; } bool flag = false; if (Equation[i] == '-') { flag = true; i++; if (i >= Len) return FALSE; } if (!IsNum(Equation[i])) return FALSE; num[++ntop] = GetNum(Equation, i, radix) * (flag ? -1 : 1); if (i >= Len) break; if (Equation[i] == ')') { while (otop > 0 && opt[otop] != '(') { INT cas = GetOutSta(num, ntop, opt, otop); if (cas == -1) return FALSE; if (cas == 0) { ans = _T("除0异常"); return TRUE; } } if (otop == 0) return FALSE; otop--; i++; if (i >= Len) break; } if (!IsOpt(Equation[i])) return FALSE; while (otop > 0 && opt[otop]!='(' && !Greater(Equation[i],opt[otop])) { INT cas = GetOutSta(num, ntop, opt, otop); if (cas == -1) return FALSE; if (cas == 0) { ans = _T("除0异常"); return TRUE; } } opt[++otop] = Equation[i++]; if (i >= Len) return FALSE; } //return 0; while (otop) { INT cas = GetOutSta(num,ntop,opt,otop); if (cas == -1) return FALSE; if (cas == 0) { ans = _T("除0异常"); return TRUE; } } if (ntop != 1) return FALSE; ans.Format(_T("%g"), num[ntop]); if (radix != 10) { ConvertRad(ans, 10, radix); } return TRUE; }bool MyCalculator::ConvertRad(CString& Num, int radixFrom, int radixTo) 传入原数字Num,转换前的进制radixFrom,转换后的进制radixTo,转换失败时(Num不是数字)返回FALSE
首先用GeNum转换为double型dtmp
- 如果是负数dtmp=-dtmp,ans加‘-’
- 整数部分用自带的_itoa_s转换
- 小数部分不断乘radixTo,顺序取整数位,默认最多转成8位小数
CSDN下载
码云
版权声明:本文标题:MFC实现浮点进制转换计算器 内容由林淑君副主任自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.xiehuijuan.com/baike/1687003687a127620.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论