Skip to content

简单的计算器

1. 如何实现计算器

实现计算器一般会有两种方法:一种是使用前缀和后缀(中缀)表达式,一种是定义栈内和栈外的优先级。 我使用了第二种方法: 1)将输入的运算式转化为由运算符和运算数组成的数组; 2) 准备好运算数的栈和运算符的栈; 3)依次读入数组中的元素,若是运算数则直接压入运算数栈,若是运算符则将该运算符的优先级和运算符栈顶元素优先级比较,有以下情况: a)若前者大,则该运算符直接入栈; b)若后者大,则弹出栈顶运算符,并弹出两个运算数,将运算结果压入运算数栈; c)若两者相等,则两者抵消。 如此直到运算符栈清空,得到最后结果。

思路 栈内栈外的优先级也有些意思的。括号在平时的计算中拥有最高优先级,然而表达式有左右括号之分,故不得不将一个括号优先级设为最低,另一个设为最高(它们在栈内栈外对立时必须优先级相等)。加减号优先级本来相等,但是显然栈内运算优先,故一般运算符栈内优先级高1。

这个计算器只是实现了基本思路,比如负号,除以零处理等等都还没来得及做,以后会补上。

2. 代码实现

首先是一个处理栈等基本功能的类。


//定义一个处理基本操作的类
class CalculateTool: NSObject {
    var opndStack:Array<String>?    //运算数栈
    var optrStack:Array<String>?    //运算符栈

    //初始化运算数栈
    func initOpndStack()
    {
        opndStack = Array<String>()
    }
    //初始化运算符栈
    func initOptrStack(){
        optrStack = Array<String>()
    }
    //压入运算数
    func pushOperand(operand:String)
    {
        opndStack?.append(operand)
    }
    //压入运算符
    func pushOperator(optr:String)
    {
        optrStack?.append(optr)
    }
    //弹出一个运算数,返回弹出数:小数点也是数
    func popOperand() -> Float
    {
        if opndStack?.count > 0
        {
             let last = opndStack?.last
            opndStack?.removeLast()
            let lastString = last! as NSString
            return lastString.floatValue
        }
        return 0.0
    }
    //弹出一个运算符
    func popOperator() -> String
    {
        if optrStack?.count>0
        {
            let last = optrStack?.last
            optrStack?.removeLast()
            return last!
        }
        return ""
    }
    //取栈顶运算数
    func getTopOperand() -> Float
    {
        if opndStack?.count > 0
        {
            let last = opndStack?.last
            let lastString = last! as NSString
            return lastString.floatValue
        }
        return 0.0
    }
    //取栈顶运算符
    func getTopOperator() -> String
    {
        if optrStack?.count>0
        {
            let last = optrStack?.last
            return last!
        }
        return ""
    }
    //判断字符串是否为运算符
    func isOperator(c:String)->Bool
    {
        if c=="+" || c=="÷" || c=="×" || c=="-" || c=="=" || c=="(" || c==")"
        {
            return true
        }
        else
        {
            return false
        }
    }

    //判断运算符的优先级:栈内和栈外的区别
    func precedeInStack(s:String) -> Int
    {
        switch s {
        case "-":
            fallthrough
        case "+":
            return 2
        case "×":
            fallthrough
        case "÷":
            return 4
        case "(":
            return 0
        case ")":
            return 7
        default:
            return -1
        }
    }
    func precedeOutStack(s:String) -> Int
    {
        switch s {
        case "-":
            fallthrough
        case "+":
            return 1
        case "×":
            fallthrough
        case "÷":
            return 3
        case "(":
            return 7
        case ")":
            return 0
        default:
            return -1
        }
    }

    //进行单独的运算
    func operate(x:Float,opr:String,y:Float) -> Float
    {
        var result:Float?
        switch opr
        {
        case "+":
            result = x+y
            break
        case "-":
            result = x-y
            break
        case "×":
            result = x*y
            break
        case "÷":
            if y != 0
            {
                result = x/y
            }
            else
            {
                return PI
            }
            break
        default:
            result = 0
            break
        }
        return result!
    }
}

接着处理运算表达式:

class CalculateResult: NSObject {
    var calculateTool:CalculateTool!

    override init() {
        super.init()
        calculateTool = CalculateTool()
        calculateTool.initOptrStack()
        calculateTool.initOpndStack()
    }
    //如何排除异常的表达式,包括小数点,除以0,其他格式
    func calculateWithString(s:String) -> Float
    {
        var calculateString = stringToArray(s)
        var result:Float = 0.0//初始化一个结果
        while calculateString.count>0
        {
            let firstElement = calculateString.first
            if firstElement?.componentsSeparatedByString(".").count>2
            {
                return PI
            }
            else if Float(firstElement!)==nil//如果是运算符,则比较优先级;多个小数点的情况
            {
                let topOptr = calculateTool.getTopOperator()
                //如果栈内运算符优先级较低,则运算符入栈
                if calculateTool.precedeInStack(topOptr)<calculateTool.precedeOutStack(firstElement!)
                {
                    calculateTool.pushOperator(firstElement!)
                    calculateString.removeAtIndex(0)
                    continue
                }
                //从符号栈弹出一个运算符,从对象栈弹出2个操作数,运算结果压入对象栈
                else if calculateTool.precedeInStack(topOptr)>calculateTool.precedeOutStack(firstElement!)
                {
                    let popOptr = calculateTool.popOperator()
                    let secondNumber = calculateTool.popOperand()
                    let firstNumber = calculateTool.popOperand()
                    let tempResult = calculateTool.operate(firstNumber, opr: popOptr, y: secondNumber)
                    if tempResult==PI//如果除以0
                    {
                        calculateTool.initOptrStack()
                        calculateTool.initOpndStack()
                        return PI
                    }

                    calculateTool.pushOperand(String(tempResult))
                    continue
                }
                else//两个运算符相等时,从运算栈抵消
                {
                    let a = calculateTool.popOperator()
                    print("弹出的运算符为\(a)")
                    calculateString.removeAtIndex(0)
                    continue
                }
            }
            else//如果运算数则直接入栈,如果有两个小数点需要报错
            {
                calculateTool.pushOperand(firstElement!)
                calculateString.removeAtIndex(0)
                continue
            }
        }
        //上述步骤结束后那么剩下的运算符必然有序排列:从运算符栈弹出一个运算符,从运算数栈弹出两个运算数,将结果压入运算数
        while calculateTool.getTopOperator() != ""
        {
            let popOptr = calculateTool.popOperator()
            let secondNumber = calculateTool.popOperand()
            let firstNumber = calculateTool.popOperand()
            let tempResult = calculateTool.operate(firstNumber, opr: popOptr, y: secondNumber)
            calculateTool.pushOperand(String(tempResult))
        }
        result = calculateTool.getTopOperand()
        calculateTool.initOptrStack()
        calculateTool.initOpndStack()
        return result
    }


    //将输入的字符串输出为数组
    func stringToArray(s:String) -> Array<String>
    {
        var stringArray = Array<String>()
        var mutableStr = s
        while mutableStr.characters.count>0
        {
            let firstCharacter = mutableStr.substringToIndex(mutableStr.startIndex.advancedBy(1))
            if Int(firstCharacter)>=0 &&  Int(firstCharacter)<10//如果是数字
            {
                let length = mutableStr.characters.count

                //从前向后扫描
                for i in 0...length-1
                {
                    let character = mutableStr.substringWithRange(Range(start:mutableStr.startIndex.advancedBy(i),end:mutableStr.startIndex.advancedBy(i+1)))
                    print("该字符的整数值为\(Int(character))")
                    if Int(character)==nil && character != "."
                    {
                        let numberStr = mutableStr.substringToIndex(mutableStr.startIndex.advancedBy(i))
                        stringArray.append(numberStr)
                        mutableStr = mutableStr.substringFromIndex(mutableStr.startIndex.advancedBy(i))
                        break//跳出当次循环
                    }
                }
            }
            else//如果是运算符
            {
                stringArray.append(firstCharacter)
                mutableStr = mutableStr.substringFromIndex(mutableStr.startIndex.advancedBy(1))
            }
        }

        return stringArray
    }

}

界面用的UICollectionView,很容易实现。