万位计算器资料文档格式.docx
《万位计算器资料文档格式.docx》由会员分享,可在线阅读,更多相关《万位计算器资料文档格式.docx(53页珍藏版)》请在冰豆网上搜索。
1.大数据处理模块:
这个模块主要是定义我的大数的数据结构,以及完成基于这个数据结构的各种运算,如加减乘除正弦余弦模幂对数等一系列运算
2.表达式处理模块:
这个模块主要是将用户的输入进行检错和转换,将正确的中缀表达式转换成后缀表达式,然后进行求解后缀表达式,最终得到结果
3.界面模块:
这个模块主要是用于与用户交互,良好的界面能够增加应用的实用性,它是连接用户与底层实现的桥梁
总的说来,架构还是相当清晰的,并且根据这个架构进行开发,使得我的开发效率大大提高。
这个部分将较为详细的讲述,按照三个模块的顺序进行分析。
(一)大数据处理模块
数据结构:
publicBigDouble(){
dBaseNumber=0.0;
iIndexNumber=0;
}
上面是最基本的数据结构,分为指数与底数两个模块,全程采用科学计数法进行保存数据。
这样做的好处是可以尽量使得运算变得简单有效,但是却损失了精度,当然,后来我将double替换成了java自带的BigDecimal类,使得进度一下子提高,但是,运行速度一下子变慢了,由于考虑到我要将这个程序写到手机应用里去,效率应该多考虑一点,毕竟,只要通过这个编程,我对计算机对数据的处理的理解跟进一步就行了,我更多想的是这个软件以后我自己还会用,也许说不一定别人也可以用,所以我最终还是选择了double作为底数。
下面是改模块实现的函数列表:
ModifierandType
MethodandDescription
BigDouble
log10(BigDoubleotherBigDouble)
返会以10为底,otherBigDouble为真值的对数
ln(BigDoubleotherBigDouble)
返会以e为底,otherBigDouble为真值的对数
bigAbs(BigDoubleotherBigDouble)
返回otherBigDouble的绝对值
bigSqrt(BigDoubleotherBigDouble)
返回对otherBigDouble的平方根
bigExp(BigDoublebDouble1)
返回e的bDouble1次方
add(BigDoubleotherBigDouble)
返回this和otherBigDouble的加法结果
minus(BigDoubleotherBigDouble)
返回this和otherBigDouble的减法结果
multi(BigDoubleotherBigDouble)
返回this*otherBigDouble
div(BigDoubleotherBigDouble)
返回this/otherBigDouble
sin(BigDoubleotherBigDouble)
返回sin(otherBigDouble)
cos(BigDoubleotherBigDouble)
返回cos(otherBigDouble)
pow(BigDoublebDouble1,BigDoublebDouble2)
返回bDouble1^bDouble2
tan(BigDoubleotherBigDouble)
返回tan(otherBigDouble)
上述涉及的函数型运算全部采用迈克劳林展开式进行运算,
如:
等,这里就不再一一列举了,总之采用麦克劳林方法计算,将指数转化为多项式运算,大大提高了效率。
(二)表达式处理模块
这一块其实算起来是最难的,首先得对输入的字符串表达式进行正确性判断,然后还要将正确的中缀表达式转换成后缀表达式,然后进行计算。
这一块,至今为止,出错控制我仍然没有做的很到位。
该模块的主要流程如下:
使用波兰表达式进行求解,第一步要将表达式拆分成各个基本元素,第二步对拆分成基本元素的中缀表达式进行排序,将其转化为后缀表达式,第三步从左到右依次计算,直到遇到表达式结尾符,输出结果
下面是一个利用中缀表达式求解的样例:
中缀表达式的计算原理是这样的:
将一个普通的中序表达式转换为逆波兰表达式的一般算法是:
(1)首先构造一个运算符栈,此运算符在栈内遵循越往栈顶优先级越高的原则。
(2)读入一个用中缀表示的简单算术表达式,为方便起见,设该简单算术表达式的右端多加上了优先级最低的特殊符号“#”。
(3)从左至右扫描该算术表达式,从第一个字符开始判断,如果该字符是数字,则分析到该数字串的结束并将该数字串直接输出。
(4)如果不是数字,该字符则是运算符,此时需比较优先关系。
做法如下:
将该字符与运算符栈顶的运算符的优先关系相比较。
如果,该字符优先关系高于此运算符栈顶的运算符,则将该运算符入栈。
倘若不是的话,则将栈顶的运算符从栈中弹出,直到栈顶运算符的优先级低于当前运算符,将该字符入栈。
(5)重复上述操作(3)-(4)直至扫描完整个简单算术表达式,确定所有字符都得到正确处理,我们便可以将中缀式表示的简单算术表达式转化为逆波兰表示的简单算术表达式。
最后,计算后缀表达式的时候如果遇到数字就放到栈里,遇到操作符就根据操作符是单目还是双目运算符从栈里取出相应的数据,将计算结果入栈,知道扫描完整个后缀表达式队列后,结果栈内的数据就是表达式的值。
比如:
5-(2+1)*12.1
后缀:
521+12.1*-
运算符的优先级
符号
优先级
#(定义的特殊符号,队列计算中作为标志,无实际意义)
+
1
-
*
2
/
^
3
sin
4
cos
tan
cot
asin
acos
acot
atan
log
ln
exp
abs
sqrt
(
5
)
代码这里就不展示了,代码全部在附件之中,反正就是按照这个流程走的。
(三)用户界面模块:
说实话,界面模块其实是相对简单的,采用了xml超文本进行UI的设计,使得界面开发比较方便,主要特点是部署到android平台:
下面是布局文件的代码片段:
<
EditText
android:
id="
@+id/rstText"
layout_width="
match_parent"
layout_height="
100dp"
layout_alignLeft="
@+id/expText"
layout_alignParentTop="
true"
layout_marginTop="
23dp"
gravity="
bottom|right"
textSize="
18sp"
textColor="
#FFF"
background="
@drawable/edit"
editable="
false"
ems="
10"
/>
layout_below="
hint="
"
25sp"
>
下面是实际视图:
简单做了一些测试,由于采用的是麦克劳林展开式,故有些数据并不是完全是理论上那样,会有一定的损失,这个我已经提前预料到了,比如sin(pi):
但是在理论上仍是正确的,
多项式运算也能够支持:
总之,测试上能够达到预期的目的
实验心得与总结
总的说来,通过这次编程实践,我学到了很多知识,过程中也遇到了很多困难,但是都一一进行了解决,能力得到了较大的提高。
首先最开始我在开始做这个软件之前我就在想,我把这个做完之后又能怎样呢,因为我遇到过很多这样的事,我们有时候做了很多事,但是做完之后就扔那,算是完成任务就再也不管了,但是我不想这样,我想的是我不仅要学到知识,我还要我做的东西以后能够用起来,至少可以供我自己使用,结合这个想法,加之一直对手机应用程序开发一直有点兴趣,但是就是没有合适的机会进行学习,所以这次我选择了android平台下的开发,这样的话,我既可以学习计算机组成与原理,又可以学习手机应用开发,还可以使自己做的东西可以得到应用,这样的感觉很不错。
于是,就这样,我就开始开始做这个手机端的高精度计算器了
从一开始开发的路途就不平坦,准确的说是不怎么平坦,最开始在数据结构方面就来来回回的改,好不容易定下来了,然后到后面开发优势磕磕绊绊,特别是在做麦克劳林时,由于double本身的限制,我一度被卡在精度这一块。
当看着旁边有的同学用python的自带大数已经早早就做完这个计算器时,我变得很着急,因为我的核心类还没完成,但是,我还是坚持了下来,终于,进过一两周的艰苦奋斗,我把大数的核心类写完了。
这个时候,我面临的是做不做多项式运算,因为之前进度在我前面的好几个人都说多项式运算太难,开始是想做,但是最后就只做了一个一次只能有一个运算符的计算器,我当时也有点畏难,本打算也就不做多项式运算了,但是后来又想想,如果多项式运算都不支持,那何来实用性,谁会用一个连用着都不舒服的应用,用户才不管你程序员开发时难不难呢,用户要的是使用体验,如果程序员不能给用户良好的体验,那就不说一个合格的程序员,于是我下定决心要把多项式这一块啃下来,于是我到图书馆借了一些书,并结合不联网进行了这一方面的学习,通过一周左右的时间,我掌握了两种比较好的方法,一种是二叉树计算,另一种就是逆波兰后缀表达式计算,我比较喜欢后者,于是,在前面的基础上,我花了两天就把它写了出来,心里想想,也不是那么难嘛,到了最后一个阶段,就是android界面设计这个模块,虽然界面是比较简单的事,但想做好却并不容易,我从亚马逊上买了一本专门讲androidUI设计的书,边学边开发,于是,最后我的app就出来了。
虽然说只是做了一个app,但是,这个过程中收获实在太多了,不一而足,不是三言两语就能说明白的,这种乐趣,只能在切切实实的实践中才能切身感受得到。
(一)大数据处理模块:
packagecom.example.big;
publicclassBigDouble{
privatestaticfinalintINF=100;
//无穷大,百位计算器
privatestaticfinaldoublePI=3.141592653589793;
privatestaticfinaldoubleEXP=2.718281828459046;
privatedoubledBaseNumber;
privateintiIndexNumber;
/*
*一组构造函数
**/
publicBigDouble(){
}
publicBigDouble(intiNumber){
StringnumberStr=String.valueOf(iNumber);
intiLength=numberStr.length();
intiDotIndex=iLength;
intiFirstNoZeroNumberIndex=iLength;
charnumberStrArray[]=numberStr.toCharArray();
for(inti=iLength-1;
i>
=0;
--i){
if(numberStrArray[i]=='
.'
){
iDotIndex=i;
//小数点的位置
}
if(numberStrArray[i]>
'
0'
&
&
numberStrArray[i]<
='
9'
iFirstNoZeroNumberIndex=i;
//第一个非零数的位置
}
if(iDotIndex<
iFirstNoZeroNumberIndex){//0.1型
this.iIndexNumber=iDotIndex-iFirstNoZeroNumberIndex;
else{//1.01型
this.iIndexNumber=iDotIndex-iFirstNoZeroNumberIndex-1;
doubledNumber=Double.parseDouble(numberStr);
doubledTemp=0.0;
intindex=0;
if(dNumber==0.0){
this.dBaseNumber=dNumber;
this.iIndexNumber=0;
else{
for(index=-INF;
index<
INF;
++index){
dTemp=dNumber/Math.pow(10.0,index);
if(Math.abs(dTemp)>
=1.0&
Math.abs(dTemp)<
10.0){
this.dBaseNumber=dTemp;
//this.iIndexNumber=index;
break;
}
publicBigDouble(doubledNumber){
intindex=0;
doubledTemp=0.0;
if(dNumber==0.0){
1.0&
this.iIndexNumber=index;
publicBigDouble(StringnumberStr){
publicBigDouble(doubledNumber,intiNumber){
this.dBaseNumber=dNumber;
this.iIndexNumber=iNumber;
this.normalization();
*一些内部使用的函数
//返回大数的字符串
publicStringgetBigDoubleStr(){
StringbigDoubleStr=String.valueOf(this.dBaseNumber);
if(this.iIndexNumber<
0){
bigDoubleStr=bigDoubleStr.concat("
e"
);
bigDoubleStr=bigDoubleStr.concat(String.valueOf(this.iIndexNumber));
e+"
returnbigDoubleStr;
//返回非科学计数法的大数
publicStringtoString(){
StringsNumber=Double.toString(Math.abs(this.getBaseNumber()));
charsNumberArray[]=sNumber.toCharArray();
intiIndex=this.getIndexNumber();
if(this.getBaseNumber()<
0.0){
sNumber="
-"
;
if(iIndex>
=0){
for(inti=0;
i<
sNumberArray.length;
++i){
if(sNumberArray[i]!
sNumber+=String.valueOf(sNumberArray[i]);
if(sNumber.length()==iIndex+1&
iIndex<
sNumberArray.length-2&
sNumber.charAt(0)!
-'
sNumber+="
."
elseif(sNumber.length()==iIndex+2&
sNumberArray.length-1&
sNumber.charAt(0)=='