Android计算器功能的实现
Android计算器功能的实现
指尖残雪 发表于1年前
Android计算器功能的实现
  • 发表于 1年前
  • 阅读 25
  • 收藏 1
  • 点赞 2
  • 评论 0

腾讯云 十分钟定制你的第一个小程序>>>   

刚刚学习Android,学着别人写一个计算器,花了好久的时间。这个计算器比较全面,使用网格布局,也许装在你的手机上会出现屏幕适配的问题,这个原因是因为我没有获取屏幕大小,然后平均分配给每一个网格合适的长度,如果想修复的同学可以自行添加上,关于获取屏幕的大小可以参考http://blog.csdn.net/bq1073100909/article/details/39318673

源代码我已经上传:http://download.csdn.net/detail/bq1073100909/8086767

布局文件:

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:background="#ff808080"
    >
    <!-- 结果显示框 -->
     <EditText 
        android:id="@+id/input"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:gravity="right"
        android:editable = "false"
        android:cursorVisible="false"
        android:text="0"                              
        />   
     <!-- 接下去采TableRow的格式进行布局设计 -->
     <TableRow 
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >
		<!-- 用于显示存储结果 -->
         <TextView 
             android:layout_width="65sp"
             android:layout_height="wrap_content"
             android:text="  MEM :"
             />
         <!-- 默认显示字符串0 -->
         <TextView 
             android:id="@+id/mem"
             android:layout_width="fill_parent"
             android:layout_height="wrap_content"
             android:text="0"
             />
     </TableRow>
     
     <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >
         <!-- 显示当前是角度还是弧度,默认是角度 -->
         <TextView android:id="@+id/_drg"
                android:layout_width="65sp"
                android:layout_height="wrap_content"
                android:text="   DEG"                 
                />
            <!-- 清除存储结果 -->
         <Button android:id="@+id/mc"
             android:text="MC"
             android:layout_width="106sp"
             android:layout_height="wrap_content"   
             /> 
          <!-- 清除输出窗口的所有内容 -->
         <Button android:id="@+id/c"
             android:text="C"
             android:layout_width="fill_parent"
             android:layout_height="wrap_content"  
             />
     </TableRow>
     <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >  
         	<!-- 在角度和弧度之间切换 -->
         <Button android:id="@+id/drg"
             android:text="DRG"
             android:layout_width="65sp"
             android:layout_height="wrap_content" 
             /> 
            <!-- 正弦计算 -->
            <Button android:id="@+id/sin"
             android:text="sin"
             android:layout_width="65sp" 
             android:layout_height="wrap_content" 
             />
            <!-- 余弦计算 -->
            <Button android:id="@+id/cos"
             android:text="cos"
             android:layout_width="65sp"
             android:layout_height="wrap_content" 
             />
            <!-- 正切计算 -->
         <Button android:id="@+id/tan"
             android:text="tan"
             android:layout_width="65sp"
             android:layout_height="wrap_content" 
             />
            <!-- 阶乘计算 -->
         <Button android:id="@+id/factorial"
             android:text="n!"
             android:layout_width="65sp"
             android:layout_height="wrap_content" 
             />
            <!-- 退格键 -->
         <Button android:id="@+id/bksp"
             android:text="Bksp"
             android:layout_width="65sp"
             android:layout_height="wrap_content" 
             />                                                                                                              
    </TableRow>  
     <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >
         	<!-- 数字7 -->
            <Button android:id="@+id/seven"
             	android:text="7"
                android:layout_width="65sp" 
                android:layout_height="wrap_content" 
                />
            <!-- 数字8 -->
            <Button android:id="@+id/eight"
             	android:text="8"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />    
            <!-- 数字9 -->
            <Button android:id="@+id/nine"
             android:text="9"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />    
            <!-- 除号 -->
            <Button android:id="@+id/divide"
             	android:text="÷"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 左括号 -->     
            <Button android:id="@+id/left"
             	android:text="("
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 右括号 -->   
            <Button android:id="@+id/right"
            	android:text=")"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
     </TableRow>
     <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >  
         	<!-- 数字4 -->
            <Button android:id="@+id/four"
             android:text="4"
                android:layout_width="65sp" 
                android:layout_height="wrap_content" 
                />
            <!-- 数字5 -->
            <Button android:id="@+id/five"
             android:text="5"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 数字6 -->    
            <Button android:id="@+id/six"
             android:text="6"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 乘号 -->    
            <Button android:id="@+id/mul"
             android:text="×"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 开方 -->  
            <Button android:id="@+id/sqrt"
             android:text="√"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 乘方 -->
            <Button android:id="@+id/square"
             android:text="^"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />                                                                                                               
    </TableRow>            
    <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >
            <!-- 数字1 -->   
            <Button android:id="@+id/one"
             android:text="1"
                android:layout_width="65sp" 
                android:layout_height="wrap_content" 
                />
            <!-- 数字2 -->
            <Button android:id="@+id/two"
             android:text="2"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 数字3 -->    
            <Button android:id="@+id/three"
             android:text="3"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 减号 -->    
            <Button android:id="@+id/sub"
             android:text="-"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 对数 -->     
            <Button android:id="@+id/log"
             android:text="log"
                android:layout_width="65sp"
                android:layout_height="wrap_content"
                />
            <!-- 自然对数 -->
            <Button android:id="@+id/ln"
             android:text="ln"
                android:layout_width="65sp" 
                android:layout_height="wrap_content" 
                />                                                                                                   
    </TableRow>         
    <TableLayout
       android:layout_width="fill_parent"
       android:layout_height="57sp"
       >                      
    	<TableRow
         	android:layout_width="fill_parent"
       	 	android:layout_height="wrap_content"
         	>
         	<!-- 数字0 -->  
            <Button android:id="@+id/zero"
             android:text="0"
                android:layout_width="65sp" 
                android:layout_height="wrap_content" 
                />
            <!-- 小数点 -->
            <Button android:id="@+id/dot"
             android:text="."
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 等号 -->
            <Button android:id="@+id/equal"
             android:text="="
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 加号 -->     
            <Button android:id="@+id/add"
             android:text="+"
                android:layout_width="65sp"
                android:layout_height="wrap_content" 
                />
            <!-- 退出计算器 -->
            <Button android:id="@+id/exit"
             android:text="exit"
                android:layout_width="130sp"
                android:layout_height="wrap_content" 
                />                                                                                                      
    	</TableRow>  
    </TableLayout>    
    <TableRow
         android:layout_width="fill_parent"
         android:layout_height="wrap_content"
         >
    		<!-- 用于提示,告诉用户如何使用计算器的一些功能等 -->
            <TextView android:id="@+id/T"
                android:layout_width="45sp"
                android:layout_height="wrap_content"
                android:text="提示:"                 
                />
            <TextView android:id="@+id/tip"
                android:layout_width="fill_parent"
                android:layout_height="wrap_content"
                android:text="欢迎使用!"  
                /> 
    </TableRow>
     
</LinearLayout>

MainActivity.java文件

package org.dyb.acivity;


import java.text.DecimalFormat;
import java.util.StringTokenizer;

import org.dyb.activity.R;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.app.Activity;

public class MainActivity extends Activity {
	//0~9十个按键
    private Button[] btn = new Button[10];
    //显示器,用于显示输出结果
    private EditText input;
    //显示器下方的记忆器,用于记录上一次计算结果
    private TextView mem;
    //三角计算时标志显示:角度还是弧度
    private TextView _drg;
    //小提示,用于加强人机交互的弱检测、提示 
    private TextView tip;                            
    private Button
        div, mul, sub, add, equal,            // ÷ × - + =
        sin, cos, tan, log, ln,               //函数
        sqrt, square, factorial, bksp,  	  //根号  平方  阶乘   退格
        left, right, dot, exit, drg,          //(     )  .  退出     角度弧度控制键
        mc, c;                                // mem清屏键    input清屏键
    //保存原来的算式样子,为了输出时好看,因计算时,算式样子被改变
    public String str_old;
    //变换样子后的式子
    public String str_new;
    //输入控制,true为重新输入,false为接着输入
    public boolean vbegin = true;
    //控制DRG按键,true为角度,false为弧度
    public boolean drg_flag = true;
    //π值:3.14
    public double pi=4*Math.atan(1);
    //true表示正确,可以继续输入;false表示有误,输入被锁定
    public boolean tip_lock = true;
    //判断是否是按=之后的输入,true表示输入在=之前,false反之
    public boolean equals_flag = true;  

    public void onCreate(Bundle icicle)
    {
        super.onCreate(icicle);
        setContentView(R.layout.activity_main);
        //获取界面元素
        input = (EditText)findViewById(R.id.input);
        mem = (TextView)findViewById(R.id.mem);
        tip = (TextView)findViewById(R.id.tip); 
        _drg = (TextView)findViewById(R.id._drg);
        btn[0] = (Button)findViewById(R.id.zero);
         btn[1] = (Button)findViewById(R.id.one);
         btn[2] = (Button)findViewById(R.id.two);
         btn[3] = (Button)findViewById(R.id.three);
         btn[4] = (Button)findViewById(R.id.four);
         btn[5] = (Button)findViewById(R.id.five);
         btn[6] = (Button)findViewById(R.id.six);
         btn[7] = (Button)findViewById(R.id.seven);
        btn[8] = (Button)findViewById(R.id.eight);
        btn[9] = (Button)findViewById(R.id.nine);
        div = (Button)findViewById(R.id.divide);
        mul = (Button)findViewById(R.id.mul);
        sub = (Button)findViewById(R.id.sub);
        add = (Button)findViewById(R.id.add);
        equal = (Button)findViewById(R.id.equal);
        sin = (Button)findViewById(R.id.sin);
        cos = (Button)findViewById(R.id.cos);
        tan = (Button)findViewById(R.id.tan);
        log = (Button)findViewById(R.id.log);
        ln = (Button)findViewById(R.id.ln);
        sqrt = (Button)findViewById(R.id.sqrt);
        square = (Button)findViewById(R.id.square);
        factorial = (Button)findViewById(R.id.factorial);
        bksp = (Button)findViewById(R.id.bksp);
        left = (Button)findViewById(R.id.left);
        right = (Button)findViewById(R.id.right);
        dot = (Button)findViewById(R.id.dot);
        exit = (Button)findViewById(R.id.exit);
        drg = (Button)findViewById(R.id.drg);
        mc = (Button)findViewById(R.id.mc);
        c = (Button)findViewById(R.id.c);
        //为数字按键绑定监听器
        for(int i = 0; i < 10; ++i) {
            btn[i].setOnClickListener(actionPerformed);
        }
        //为+-x÷等按键绑定监听器
        div.setOnClickListener(actionPerformed);
        mul.setOnClickListener(actionPerformed);
        sub.setOnClickListener(actionPerformed);
        add.setOnClickListener(actionPerformed);
        equal.setOnClickListener(actionPerformed);
        sin.setOnClickListener(actionPerformed);
        cos.setOnClickListener(actionPerformed);
        tan.setOnClickListener(actionPerformed);
        log.setOnClickListener(actionPerformed);
        ln.setOnClickListener(actionPerformed);
        sqrt.setOnClickListener(actionPerformed);
        square.setOnClickListener(actionPerformed);    
        factorial.setOnClickListener(actionPerformed);
        bksp.setOnClickListener(actionPerformed);    
        left.setOnClickListener(actionPerformed);
        right.setOnClickListener(actionPerformed);
        dot.setOnClickListener(actionPerformed);    
        exit.setOnClickListener(actionPerformed);
        drg.setOnClickListener(actionPerformed);
        mc.setOnClickListener(actionPerformed);    
        c.setOnClickListener(actionPerformed);
    }
    
    /*
     * 键盘命令捕捉
     */
    //命令缓存,用于检测输入合法性
    String[] Tipcommand = new String[500]; 
    //Tipcommand的指针
    int tip_i = 0;         
    private OnClickListener actionPerformed = new OnClickListener() {
        public void onClick(View v) {
        	//按键上的命令获取
            String command = ((Button)v).getText().toString();
            //显示器上的字符串
            String str = input.getText().toString();  
            //检测输入是否合法
            if(equals_flag == false && "0123456789.()sincostanlnlogn!+-×÷√^".indexOf(command) != -1) {
            	//检测显示器上的字符串是否合法
                if(right(str)) {
                    if("+-×÷√^)".indexOf(command) != -1) {
                        for(int i =0 ; i < str.length(); i++) {
                            Tipcommand[tip_i] = String.valueOf(str.charAt(i));
                            tip_i++;
                        }
                        vbegin = false;
                    }
                } else {
                    input.setText("0");
                    vbegin = true;
                    tip_i = 0;
                    tip_lock = true;
                    tip.setText("欢迎使用!");
                }

                equals_flag = true;
            }
            if(tip_i > 0)
                TipChecker(Tipcommand[tip_i-1] , command);
            else if(tip_i == 0) {
                    TipChecker("#" , command);
            }
            if("0123456789.()sincostanlnlogn!+-×÷√^".indexOf(command) != -1 && tip_lock) {
                Tipcommand[tip_i] = command;
                tip_i++;
            } 
            //若输入正确,则将输入信息显示到显示器上
            if("0123456789.()sincostanlnlogn!+-×÷√^".indexOf(command) != -1 
                    && tip_lock) { //共25个按键
                print(command);
            //若果点击了“DRG”,则切换当前弧度角度制,并将切换后的结果显示到按键上方。
            } else if(command.compareTo("DRG") == 0 && tip_lock) {
                if(drg_flag == true) {
                    drg_flag = false;
                    _drg.setText("   RAD");
                } else {
                    drg_flag = true;
                    _drg.setText("   DEG");
                }
                //如果输入时退格键,并且是在按=之前
            } else if(command.compareTo("Bksp") == 0 && equals_flag) {
            	//一次删除3个字符
                if(TTO(str) == 3) {
                    if(str.length() > 3)
                        input.setText(str.substring(0, str.length() - 3)); 
                    else if(str.length() == 3) {
                        input.setText("0");   
                        vbegin = true;
                        tip_i = 0;
                        tip.setText("欢迎使用!");
                    }
                //依次删除2个字符
                } else if(TTO(str) == 2) {
                    if(str.length() > 2)
                        input.setText(str.substring(0, str.length() - 2)); 
                    else if(str.length() == 2) {
                        input.setText("0");   
                        vbegin = true;
                        tip_i = 0;
                        tip.setText("欢迎使用!");
                    }
                //依次删除一个字符
                } else if(TTO(str) == 1) {
                	//若之前输入的字符串合法则删除一个字符
                    if(right(str)) {
                        if(str.length() > 1)
                            input.setText(str.substring(0, str.length() - 1)); 
                        else if(str.length() == 1) {
                            input.setText("0");   
                            vbegin = true;
                            tip_i = 0;
                            tip.setText("欢迎使用!");
                        }
                        //若之前输入的字符串不合法则删除全部字符
                    } else {
                        input.setText("0");   
                        vbegin = true;
                        tip_i = 0;
                        tip.setText("欢迎使用!");
                    }
                }
                if(input.getText().toString().compareTo("-") == 0 || equals_flag == false) {
                    input.setText("0");
                    vbegin = true;
                    tip_i = 0;
                    tip.setText("欢迎使用!");
                }
                tip_lock = true;
                if(tip_i > 0)
                    tip_i--;
            //如果是在按=之后输入退格键
            } else if(command.compareTo("Bksp") == 0 && equals_flag ==false) {
            	//将显示器内容设置为0
                input.setText("0");
                vbegin = true;
                tip_i = 0;
                tip_lock = true;
                tip.setText("欢迎使用!");
            //如果输入的是清除键
            } else if(command.compareTo("C") == 0) {
            	//将显示器内容设置为0
                input.setText("0");
                //重新输入标志置为true
                vbegin = true;
                //缓存命令位数清0
                tip_i = 0;
                //表明可以继续输入
                tip_lock = true;
                //表明输入=之前
                equals_flag = true;
                tip.setText("欢迎使用!");
                //如果输入的是”MC“,则将存储器内容清0
            } else if(command.compareTo("MC") == 0) {
                mem.setText("0");
                //如果按”exit“则退出程序
            } else if(command.compareTo("exit") == 0) {
                System.exit(0);
            //如果输入的是=号,并且输入合法
            } else if(command.compareTo("=") == 0 && tip_lock && right(str) && equals_flag) {
                tip_i = 0;
                //表明不可以继续输入
                tip_lock = false;
                //表明输入=之后
                equals_flag = false;
                //保存原来算式样子
                str_old = str;
                //替换算式中的运算符,便于计算
                str = str.replaceAll("sin", "s");
                str = str.replaceAll("cos", "c");
                str = str.replaceAll("tan", "t");
                str = str.replaceAll("log", "g");
                str = str.replaceAll("ln", "l");
                str = str.replaceAll("n!", "!");
                //重新输入标志设置true
                vbegin = true;
                //将-1x转换成-
                str_new = str.replaceAll("-", "-1×");
                //计算算式结果
                new calc().process(str_new);
            }
            //表明可以继续输入
            tip_lock = true;
        }
    };
    //向input输出字符
    private void print(String str) {
    	//清屏后输出
        if(vbegin)
            input.setText(str);
        //在屏幕原str后增添字符
        else
            input.append(str);
        vbegin = false;
    }  
    /*
     * 判断一个str是否是合法的,返回值为true、false
     * 只包含0123456789.()sincostanlnlogn!+-×÷√^的是合法的str,返回true
     * 包含了除0123456789.()sincostanlnlogn!+-×÷√^以外的字符的str为非法的,返回false
     */
    private boolean right(String str) {
        int i = 0;
        for(i = 0;i < str.length();i++) {
            if(str.charAt(i)!='0' && str.charAt(i)!='1' && str.charAt(i)!='2' && 
                    str.charAt(i)!='3' && str.charAt(i)!='4' && str.charAt(i)!='5' && 
                    str.charAt(i)!='6' && str.charAt(i)!='7' && str.charAt(i)!='8' && 
                    str.charAt(i)!='9' && str.charAt(i)!='.' && str.charAt(i)!='-' &&
                    str.charAt(i)!='+' && str.charAt(i)!='×' && str.charAt(i)!='÷' &&
                    str.charAt(i)!='√' && str.charAt(i)!='^' && str.charAt(i)!='s' &&
                    str.charAt(i)!='i' && str.charAt(i)!='n' && str.charAt(i)!='c' &&
                    str.charAt(i)!='o' && str.charAt(i)!='t' && str.charAt(i)!='a' &&
                    str.charAt(i)!='l' && str.charAt(i)!='g' && str.charAt(i)!='(' &&
                    str.charAt(i)!=')' && str.charAt(i)!='!') 
                break;
        }
        if(i == str.length()) {
            return true;
        } else {
            return false;
        }
    }
    /*
     * 检测函数,返回值为3、2、1  表示应当一次删除几个?  Three+Two+One = TTO
     * 为Bksp按钮的删除方式提供依据
     * 返回3,表示str尾部为sin、cos、tan、log中的一个,应当一次删除3个
     * 返回2,表示str尾部为ln、n!中的一个,应当一次删除2个
     * 返回1,表示为除返回3、2外的所有情况,只需删除一个(包含非法字符时要另外考虑:应清屏)
     */
    private int TTO(String str) {  
        if((str.charAt(str.length() - 1) == 'n' && 
                str.charAt(str.length() - 2) == 'i' &&
                str.charAt(str.length() - 3) == 's') ||
                (str.charAt(str.length() - 1) == 's' && 
                str.charAt(str.length() - 2) == 'o' &&
                str.charAt(str.length() - 3) == 'c') ||
                (str.charAt(str.length() - 1) == 'n' && 
                str.charAt(str.length() - 2) == 'a' &&
                str.charAt(str.length() - 3) == 't') ||
                (str.charAt(str.length() - 1) == 'g' && 
                str.charAt(str.length() - 2) == 'o' &&
                str.charAt(str.length() - 3) == 'l')) {
            return 3;
        } else if((str.charAt(str.length() - 1) == 'n' && 
                        str.charAt(str.length() - 2) == 'l') ||
                        (str.charAt(str.length() - 1) == '!' && 
                        str.charAt(str.length() - 2) == 'n')) {
                    return 2;
        } else { return 1; }
    }  
    /*
     * 检测函数,对str进行前后语法检测
     * 为Tip的提示方式提供依据,与TipShow()配合使用
     *  编号 字符    其后可以跟随的合法字符
     *   1  (                 数字|(|-|.|函数
     *   2   )                算符|)|√ ^
     *   3  .      数字|算符|)|√ ^
     *   4   数字        .|数字|算符|)|√ ^
     *   5   算符             数字|(|.|函数
     *   6 √ ^     ( |. | 数字
     *   7  函数           数字|(|.
     *   
     * 小数点前后均可省略,表示0
     * 数字第一位可以为0
     */
    private void TipChecker(String tipcommand1,String tipcommand2) {
    	//Tipcode1表示错误类型,Tipcode2表示名词解释类型
    	int Tipcode1 = 0 , Tipcode2 = 0;
        //表示命令类型
        int tiptype1 = 0 , tiptype2 = 0;
        //括号数
        int bracket = 0;
        //“+-x÷√^”不能作为第一位
        if(tipcommand1.compareTo("#") == 0 && (tipcommand2.compareTo("÷") == 0 || 
                tipcommand2.compareTo("×") == 0 || tipcommand2.compareTo("+") == 0 || 
                tipcommand2.compareTo(")") == 0 || tipcommand2.compareTo("√") == 0 ||
                tipcommand2.compareTo("^") == 0)) {
            Tipcode1 = -1;
        }
        //定义存储字符串中最后一位的类型
        else if(tipcommand1.compareTo("#") != 0) {
            if(tipcommand1.compareTo("(") == 0) {
                tiptype1 = 1;
            } else if(tipcommand1.compareTo(")") == 0) {
                tiptype1 = 2;
            } else if(tipcommand1.compareTo(".") == 0) {
                tiptype1 = 3;
            } else if("0123456789".indexOf(tipcommand1) != -1) {
                tiptype1 = 4;
            } else if("+-×÷".indexOf(tipcommand1) != -1) {
                tiptype1 = 5;
            } else if("√^".indexOf(tipcommand1) != -1) {
                tiptype1 = 6;
            } else if("sincostanloglnn!".indexOf(tipcommand1) != -1) {
                tiptype1 = 7;
            }
            //定义欲输入的按键类型
            if(tipcommand2.compareTo("(") == 0) {
                tiptype2 = 1;
            } else if(tipcommand2.compareTo(")") == 0) {
                tiptype2 = 2;
            } else if(tipcommand2.compareTo(".") == 0) {
                tiptype2 = 3;
            } else if("0123456789".indexOf(tipcommand2) != -1) {
                tiptype2 = 4;
            } else if("+-×÷".indexOf(tipcommand2) != -1) {
                tiptype2 = 5;
            } else if("√^".indexOf(tipcommand2) != -1) {
                tiptype2 = 6;
            } else if("sincostanloglnn!".indexOf(tipcommand2) != -1) {
                tiptype2 = 7;
            }
            
            switch(tiptype1) {
            case 1:
            	//左括号后面直接接右括号,“+x÷”(负号“-”不算),或者"√^"
                if(tiptype2 == 2 || (tiptype2 == 5 && tipcommand2.compareTo("-") != 0) || 
                        tiptype2 == 6) 
                    Tipcode1 = 1;
                break;
            case 2:
            	//右括号后面接左括号,数字,“+-x÷sin^...”
                if(tiptype2 == 1 || tiptype2 == 3 || tiptype2 == 4 || tiptype2 == 7)
                	Tipcode1 = 2;
                break;
            case 3:
            	//“.”后面接左括号或者“sincos...”
                if(tiptype2 == 1 || tiptype2 == 7)
                	Tipcode1 = 3;
                //连续输入两个“.”
                if(tiptype2 == 3)
                	Tipcode1 = 8;
                break;            
            case 4:
            	//数字后面直接接左括号或者“sincos...”
                if(tiptype2 == 1 || tiptype2 == 7)
                	Tipcode1 = 4;
                break;
            case 5:
            	//“+-x÷”后面直接接右括号,“+-x÷√^”
                if(tiptype2 == 2 || tiptype2 == 5 || tiptype2 == 6)
                	Tipcode1 = 5;
                break;
            case 6:
            	//“√^”后面直接接右括号,“+-x÷√^”以及“sincos...”
                if(tiptype2 == 2 || tiptype2 == 5 || tiptype2 == 6 || tiptype2 == 7)
                	Tipcode1 = 6;
                break;
            case 7:
            	//“sincos...”后面直接接右括号“+-x÷√^”以及“sincos...”
                if(tiptype2 == 2 || tiptype2 == 5 || tiptype2 == 6 || tiptype2 == 7)
                	Tipcode1 = 7;
                break;
            }
        }
        //检测小数点的重复性,Tipconde1=0,表明满足前面的规则
        if(Tipcode1 == 0 && tipcommand2.compareTo(".") == 0) {
            int tip_point = 0;
            for(int i = 0;i < tip_i;i++) {
            	//若之前出现一个小数点点,则小数点计数加1
                if(Tipcommand[i].compareTo(".") == 0) {
                    tip_point++;
                }
                //若出现以下几个运算符之一,小数点计数清零
                if(Tipcommand[i].compareTo("sin") == 0 || Tipcommand[i].compareTo("cos") == 0 ||
                        Tipcommand[i].compareTo("tan") == 0 || Tipcommand[i].compareTo("log") == 0 ||
                        Tipcommand[i].compareTo("ln") == 0 || Tipcommand[i].compareTo("n!") == 0 || 
                        Tipcommand[i].compareTo("√") == 0 || Tipcommand[i].compareTo("^") == 0 || 
                        Tipcommand[i].compareTo("÷") == 0 || Tipcommand[i].compareTo("×") == 0 || 
                        Tipcommand[i].compareTo("-") == 0 || Tipcommand[i].compareTo("+") == 0 || 
                        Tipcommand[i].compareTo("(") == 0 || Tipcommand[i].compareTo(")") == 0 ) {
                    tip_point = 0;
                }
            }
            tip_point++;
            //若小数点计数大于1,表明小数点重复了
            if(tip_point > 1) {
                Tipcode1 = 8;
            }
        }
        //检测右括号是否匹配
        if(Tipcode1 == 0 && tipcommand2.compareTo(")") == 0) {
            int tip_right_bracket = 0;
            for(int i = 0;i < tip_i;i++) {
            	//如果出现一个左括号,则计数加1
                if(Tipcommand[i].compareTo("(") == 0) {
                    tip_right_bracket++;
                }
                //如果出现一个右括号,则计数减1
                if(Tipcommand[i].compareTo(")") == 0) {
                    tip_right_bracket--;
                }
            }
            //如果右括号计数=0,表明没有响应的左括号与当前右括号匹配
            if(tip_right_bracket == 0) {
                Tipcode1 = 10; 
            }
        }
        //检查输入=的合法性
        if(Tipcode1 == 0 && tipcommand2.compareTo("=") == 0) {
        	//括号匹配数
        	int tip_bracket = 0; 
            for(int i = 0;i < tip_i;i++) {
                if(Tipcommand[i].compareTo("(") == 0) {
                    tip_bracket++;
                }
                if(Tipcommand[i].compareTo(")") == 0) {
                    tip_bracket--;
                }
            }
            //若大于0,表明左括号还有未匹配的
            if(tip_bracket > 0) {
                Tipcode1 = 9; 
                bracket = tip_bracket;
            } else if(tip_bracket == 0) {
            	//若前一个字符是以下之一,表明=号不合法
                if("√^sincostanloglnn!".indexOf(tipcommand1) != -1) {
                    Tipcode1 = 6;
                }
                //若前一个字符是以下之一,表明=号不合法
                if("+-×÷".indexOf(tipcommand1) != -1) {
                    Tipcode1 = 5;
                }
            }
        }
        //若命令式以下之一,则显示相应的帮助信息
        if(tipcommand2.compareTo("MC") == 0) Tipcode2 = 1;
        if(tipcommand2.compareTo("C") == 0) Tipcode2 = 2;
        if(tipcommand2.compareTo("DRG") == 0) Tipcode2 = 3;
        if(tipcommand2.compareTo("Bksp") == 0) Tipcode2 = 4;
        if(tipcommand2.compareTo("sin") == 0) Tipcode2 = 5;
        if(tipcommand2.compareTo("cos") == 0) Tipcode2 = 6;
        if(tipcommand2.compareTo("tan") == 0) Tipcode2 = 7;
        if(tipcommand2.compareTo("log") ==0) Tipcode2 = 8;
        if(tipcommand2.compareTo("ln") == 0) Tipcode2 = 9;
        if(tipcommand2.compareTo("n!") == 0) Tipcode2 = 10;
        if(tipcommand2.compareTo("√") == 0) Tipcode2 = 11;
        if(tipcommand2.compareTo("^") == 0) Tipcode2 = 12;
        //显示帮助和错误信息
        TipShow(bracket , Tipcode1 , Tipcode2 , tipcommand1 , tipcommand2);
    }
    
    /*
     * 反馈Tip信息,加强人机交互,与TipChecker()配合使用
     */
    private void TipShow(int bracket , int tipcode1 , int tipcode2 , 
            String tipcommand1 , String tipcommand2) {
        String tipmessage = "";
        if(tipcode1 != 0) 
        	tip_lock = false;//表明输入有误
        switch(tipcode1) {
        case -1:
            tipmessage = tipcommand2 + "  不能作为第一个算符\n";
            break;
        case 1:
            tipmessage = tipcommand1 + "  后应输入:数字/(/./-/函数 \n";
            break;
        case 2:
            tipmessage = tipcommand1 + "  后应输入:)/算符 \n";
            break;
        case 3:
            tipmessage = tipcommand1 + "  后应输入:)/数字/算符 \n";
            break;
        case 4:
            tipmessage = tipcommand1 + "  后应输入:)/./数字 /算符 \n";
            break;
        case 5:
            tipmessage = tipcommand1 + "  后应输入:(/./数字/函数 \n";
            break;
        case 6:
            tipmessage = tipcommand1 + "  后应输入:(/./数字 \n";
            break;
        case 7:
            tipmessage = tipcommand1 + "  后应输入:(/./数字 \n";
            break;
        case 8:
            tipmessage = "小数点重复\n";
            break;
        case 9:
            tipmessage = "不能计算,缺少 "+ bracket +" 个 )";
            break;
        case 10:
            tipmessage = "不需要  )";
            break;
        }        
        switch(tipcode2) {
        case 1:
            tipmessage = tipmessage + "[MC 用法: 清除记忆 MEM]";
            break;
        case 2:
            tipmessage = tipmessage + "[C 用法: 归零]";
            break;
        case 3:
            tipmessage = tipmessage + "[DRG 用法: 选择 DEG 或 RAD]";
            break;
        case 4:
            tipmessage = tipmessage + "[Bksp 用法: 退格]";
            break;
        case 5:
            tipmessage = tipmessage + "sin 函数用法示例:\n" +
                    "DEG:sin30 = 0.5      RAD:sin1 = 0.84\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "sin(cos45),而不是sincos45" ;
            break;
        case 6:
            tipmessage = tipmessage + "cos 函数用法示例:\n" +
                    "DEG:cos60 = 0.5      RAD:cos1 = 0.54\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "cos(sin45),而不是cossin45" ;
            break;
        case 7:
            tipmessage = tipmessage + "tan 函数用法示例:\n" +
                    "DEG:tan45 = 1      RAD:tan1 = 1.55\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "tan(cos45),而不是tancos45" ;
            break;
        case 8:
            tipmessage = tipmessage + "log 函数用法示例:\n" +
                    "log10 = log(5+5) = 1\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "log(tan45),而不是logtan45" ;
            break;
        case 9:
            tipmessage = tipmessage + "ln 函数用法示例:\n" +
                    "ln10 = le(5+5) = 2.3   lne = 1\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "ln(tan45),而不是lntan45" ;
            break;
        case 10:
            tipmessage = tipmessage + "n! 函数用法示例:\n" +
                    "n!3 = n!(1+2) = 3×2×1 = 6\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "n!(log1000),而不是n!log1000" ;
            break;
        case 11:
            tipmessage = tipmessage + "√ 用法示例:开任意次根号\n" +
                    "如:27开3次根为  27√3 = 3\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "(函数)√(函数) , (n!3)√(log100) = 2.45";
            break;
        case 12:
            tipmessage = tipmessage + "^ 用法示例:开任意次平方\n" +
                    "如:2的3次方为  2^3 = 8\n" +
                    "注:与其他函数一起使用时要加括号,如:\n" +
                    "(函数)√(函数) , (n!3)^(log100) = 36";
            break;
        }
        //将提示信息显示到tip
        tip.setText(tipmessage);
    }
    /*
     * 整个计算核心,只要将表达式的整个字符串传入calc().process()就可以实行计算了
     * 算法包括以下几部分:
     * 1、计算部分           process(String str)  当然,这是建立在查错无错误的情况下
     * 2、数据格式化      FP(double n)         使数据有相当的精确度
     * 3、阶乘算法           N(double n)          计算n!,将结果返回
     * 4、错误提示          showError(int code ,String str)  将错误返回
     */
    public class calc {
         public calc(){
             
        }
         final int MAXLEN = 500;
            /* 
             * 计算表达式
             * 从左向右扫描,数字入number栈,运算符入operator栈
             * +-基本优先级为1,×÷基本优先级为2,log ln sin cos tan n!基本优先级为3,√^基本优先级为4
             * 括号内层运算符比外层同级运算符优先级高4
             * 当前运算符优先级高于栈顶压栈,低于栈顶弹出一个运算符与两个数进行运算
             * 重复直到当前运算符大于栈顶
             * 扫描完后对剩下的运算符与数字依次计算
             */
            public void process(String str) {
                int weightPlus = 0, topOp = 0,
                topNum = 0, flag = 1, weightTemp = 0;
                //weightPlus为同一()下的基本优先级,weightTemp临时记录优先级的变化
                //topOp为weight[],operator[]的计数器;topNum为number[]的计数器
                //flag为正负数的计数器,1为正数,-1为负数
                int weight[];  //保存operator栈中运算符的优先级,以topOp计数
                double number[];  //保存数字,以topNum计数
                char ch, ch_gai, operator[];//operator[]保存运算符,以topOp计数
                String num;//记录数字,str以+-×÷()sctgl!√^分段,+-×÷()sctgl!√^字符之间的字符串即为数字
                weight = new int[MAXLEN];
                number = new double[MAXLEN];
                operator = new char[MAXLEN];
                String expression = str;
                StringTokenizer expToken = new StringTokenizer(expression,"+-×÷()sctgl!√^");
                int i = 0;
                while (i < expression.length()) {
                    ch = expression.charAt(i);
                    //判断正负数
                    if (i == 0) {
                        if (ch == '-')
                        	flag = -1;
                    } else if(expression.charAt(i-1) == '(' && ch == '-')
                    	flag = -1;
                    //取得数字,并将正负符号转移给数字
                    if (ch <= '9' && ch >= '0'|| ch == '.' || ch == 'E') {
                        num = expToken.nextToken();
                        ch_gai = ch;
                        Log.e("guojs",ch+"--->"+i);
                        //取得整个数字
                        while (i < expression.length() && 
                                (ch_gai <= '9' && ch_gai >= '0'|| ch_gai == '.' || ch_gai == 'E')) 
                        {
                            ch_gai = expression.charAt(i++);
                            Log.e("guojs","i的值为:"+i);
                        }
                        //将指针退回之前的位置
                        if (i >= expression.length()) i-=1; else {i-=2;}
                        if (num.compareTo(".") == 0) number[topNum++] = 0;
                        //将正负符号转移给数字
                        else {
                            number[topNum++] = Double.parseDouble(num)*flag;
                            flag = 1;
                        }
                    }
                    //计算运算符的优先级
                    if (ch == '(') weightPlus+=4;
                    if (ch == ')') weightPlus-=4;
                    if (ch == '-' && flag == 1 || ch == '+' || ch == '×'|| ch == '÷' || 
                            ch == 's' ||ch == 'c' || ch == 't' || ch == 'g' || ch == 'l' || 
                            ch == '!' || ch == '√' || ch == '^') {
                        switch (ch) {
                        	//+-的优先级最低,为1
                            case '+':
                            case '-':
                                weightTemp = 1 + weightPlus;
                                break;
                            //x÷的优先级稍高,为2
                            case '×':
                            case '÷':
                                weightTemp = 2 + weightPlus;
                                break;
                            //sincos之类优先级为3
                            case 's':
                            case 'c':
                            case 't':
                            case 'g':
                            case 'l':
                            case '!':
                                weightTemp = 3 + weightPlus;
                                break;
                            //其余优先级为4
                            //case '^':
                            //case '√':
                            default:
                                weightTemp = 4 + weightPlus;
                                break;
                        }
                        //如果当前优先级大于堆栈顶部元素,则直接入栈
                        if (topOp == 0 || weight[topOp-1] < weightTemp) {
                            weight[topOp] = weightTemp;
                            operator[topOp] = ch;
                            topOp++;
                        //否则将堆栈中运算符逐个取出,直到当前堆栈顶部运算符的优先级小于当前运算符
                        }else {
                             while (topOp > 0 && weight[topOp-1] >= weightTemp) {
                                switch (operator[topOp-1]) {
                                	//取出数字数组的相应元素进行运算
                                    case '+':
                                        number[topNum-2]+=number[topNum-1];
                                        break;
                                    case '-':
                                        number[topNum-2]-=number[topNum-1];
                                        break;
                                    case '×':
                                        number[topNum-2]*=number[topNum-1];
                                        break;
                                    //判断除数为0的情况
                                    case '÷':
                                        if (number[topNum-1] == 0) {
                                            showError(1,str_old);
                                            return;
                                        }
                                        number[topNum-2]/=number[topNum-1];
                                        break;
                                    case '√':
                                        if(number[topNum-1] == 0 || (number[topNum-2] < 0 &&
                                                number[topNum-1] % 2 == 0)) {
                                            showError(2,str_old);
                                            return;    
                                        } 
                                        number[topNum-2] = 
                                            Math.pow(number[topNum-2], 1/number[topNum-1]);
                                        break;
                                    case '^':
                                        number[topNum-2] = 
                                            Math.pow(number[topNum-2], number[topNum-1]);
                                        break;
                                    //计算时进行角度弧度的判断及转换
                                    //sin
                                    case 's':
                                        if(drg_flag == true) {
                                            number[topNum-1] = Math.sin((number[topNum-1]/180)*pi);
                                        } else {
                                            number[topNum-1] = Math.sin(number[topNum-1]);
                                        }
                                        topNum++;
                                        break;
                                    //cos
                                    case 'c':
                                        if(drg_flag == true) {
                                            number[topNum-1] = Math.cos((number[topNum-1]/180)*pi);
                                        } else {
                                            number[topNum-1] = Math.cos(number[topNum-1]);
                                        }
                                        topNum++;
                                        break;
                                    //tan
                                    case 't':
                                        if(drg_flag == true) {
                                            if((Math.abs(number[topNum-1])/90)%2 == 1) {
                                                showError(2,str_old);
                                                return;
                                            }
                                            number[topNum-1] = Math.tan((number[topNum-1]/180)*pi);
                                        } else {
                                            if((Math.abs(number[topNum-1])/(pi/2))%2 == 1) {
                                                showError(2,str_old);
                                                return;
                                            }
                                            number[topNum-1] = Math.tan(number[topNum-1]);
                                        }
                                        topNum++;
                                        break;
                                    //log
                                    case 'g':
                                        if(number[topNum-1] <= 0) {
                                            showError(2,str_old);
                                            return;    
                                        } 
                                        number[topNum-1] = Math.log10(number[topNum-1]);
                                        topNum++;
                                        break;
                                    //ln
                                    case 'l':
                                        if(number[topNum-1] <= 0) {
                                            showError(2,str_old);
                                            return;    
                                        } 
                                        number[topNum-1] = Math.log(number[topNum-1]);
                                        topNum++;
                                        break;
                                    //阶乘
                                    case '!':
                                        if(number[topNum-1] > 170) {
                                            showError(3,str_old);
                                            return;    
                                        } else if(number[topNum-1] < 0) {
                                            showError(2,str_old);
                                            return;
                                        }
                                        number[topNum-1] = N(number[topNum-1]);
                                        topNum++;
                                        break;
                                }
                                //继续取堆栈的下一个元素进行判断
                                topNum--;
                                topOp--;
                            }
                            //将运算符如堆栈
                            weight[topOp] = weightTemp;
                            operator[topOp] = ch;
                            topOp++;
                        }
                    }
                    i++;
                }
                //依次取出堆栈的运算符进行运算
                while (topOp>0) {
                	//+-x直接将数组的后两位数取出运算
                    switch (operator[topOp-1]) {
                    case '+':
                        number[topNum-2]+=number[topNum-1];
                        break;
                    case '-':
                        number[topNum-2]-=number[topNum-1];
                        break;
                    case '×':
                        number[topNum-2]*=number[topNum-1];
                        break;
                    //涉及到除法时要考虑除数不能为零的情况
                    case '÷':
                        if (number[topNum-1] == 0) {
                            showError(1,str_old);
                            return;
                        }
                        number[topNum-2]/=number[topNum-1];
                        break;
                    case '√':
                        if(number[topNum-1] == 0 || (number[topNum-2] < 0 &&
                                number[topNum-1] % 2 == 0)) {
                            showError(2,str_old);
                            return;    
                        } 
                        number[topNum-2] = 
                            Math.pow(number[topNum-2], 1/number[topNum-1]);
                        break;
                    case '^':
                        number[topNum-2] = 
                            Math.pow(number[topNum-2], number[topNum-1]);
                        break;
                    //sin
                    case 's':
                        if(drg_flag == true) {
                            number[topNum-1] = Math.sin((number[topNum-1]/180)*pi);
                        } else {
                            number[topNum-1] = Math.sin(number[topNum-1]);
                        }
                        topNum++;
                        break;
                    //cos
                    case 'c':
                        if(drg_flag == true) {
                            number[topNum-1] = Math.cos((number[topNum-1]/180)*pi);
                        } else {
                            number[topNum-1] = Math.cos(number[topNum-1]);
                        }
                        topNum++;
                        break;
                    //tan
                    case 't':
                        if(drg_flag == true) {
                            if((Math.abs(number[topNum-1])/90)%2 == 1) {
                                showError(2,str_old);
                                return;
                            }
                            number[topNum-1] = Math.tan((number[topNum-1]/180)*pi);
                        } else {
                            if((Math.abs(number[topNum-1])/(pi/2))%2 == 1) {
                                showError(2,str_old);
                                return;
                            }
                            number[topNum-1] = Math.tan(number[topNum-1]);
                        }
                        topNum++;
                        break;
                    //对数log
                    case 'g':
                        if(number[topNum-1] <= 0) {
                            showError(2,str_old);
                            return;    
                        } 
                        number[topNum-1] = Math.log10(number[topNum-1]);
                        topNum++;
                        break;
                    //自然对数ln
                    case 'l':
                        if(number[topNum-1] <= 0) {
                            showError(2,str_old);
                            return;    
                        } 
                        number[topNum-1] = Math.log(number[topNum-1]);
                        topNum++;
                        break;
                    //阶乘
                    case '!':
                        if(number[topNum-1] > 170) {
                            showError(3,str_old);
                            return;    
                        } else if(number[topNum-1] < 0) {
                            showError(2,str_old);
                            return;
                        }
                        number[topNum-1] = N(number[topNum-1]);
                        topNum++;
                        break;
                    }
                    //取堆栈下一个元素计算
                    topNum--;
                    topOp--;
                }
                //如果是数字太大,提示错误信息
                if(number[0] > 7.3E306) {
                    showError(3,str_old);
                    return;
                }
                //输出最终结果
                input.setText(String.valueOf(FP(number[0]))); 
                tip.setText("计算完毕,要继续请按归零键 C");
                mem.setText(str_old+"="+String.valueOf(FP(number[0])));
            }
            
            /*
             * FP = floating point 控制小数位数,达到精度
             * 否则会出现 0.6-0.2=0.39999999999999997的情况,用FP即可解决,使得数为0.4
             * 本格式精度为15位
             */
            public double FP(double n) {
                //NumberFormat format=NumberFormat.getInstance();  //创建一个格式化类f
                //format.setMaximumFractionDigits(18);    //设置小数位的格式
                DecimalFormat format = new DecimalFormat("0.#############");
                return Double.parseDouble(format.format(n));
            }
            
            /*
             * 阶乘算法
             */
            public double N(double n) { 
                int i = 0;
                double sum = 1;
                //依次将小于等于n的值相乘
                for(i = 1;i <= n;i++) {
                    sum = sum*i;
                }
                return sum;
            }
            
            /*
             * 错误提示,按了"="之后,若计算式在process()过程中,出现错误,则进行提示
             */
            public void showError(int code ,String str) {
                String message="";
                switch (code) {
                case 1:
                    message = "零不能作除数";
                    break;
                case 2:
                    message = "函数格式错误";
                    break;
                case 3:
                    message = "值太大了,超出范围";
                }
                input.setText("\""+str+"\""+": "+message);
                tip.setText(message+"\n"+"计算完毕,要继续请按归零键 C");
            }  
        }
}


共有 人打赏支持
粉丝 8
博文 73
码字总数 0
×
指尖残雪
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: