本文實(shí)例講述了Android開(kāi)發(fā)中計(jì)算器的sin、cos及tan值計(jì)算問(wèn)題。分享給大家供大家參考,具體如下:
接到一個(gè)需求 :要求計(jì)算器sin90=1,拿到知道很疑問(wèn) 難道不等于一么?測(cè)試了四五個(gè)手機(jī) ,有的滿足,有的sin90=0.8939…。查了api文檔后發(fā)現(xiàn) jdk中Math.sin/cos/tan ()求值采用弧度值,目前覺(jué)大部分手機(jī)計(jì)算器 如果滿足sin(90)=1就不會(huì)滿足sin(pi/2)=1,因?yàn)槠渌惴ㄈ绻D(zhuǎn)換弧度值(x/180*pi).當(dāng)輸入弧度值算時(shí)會(huì)變?yōu)閟in(弧度值/180*pi)使結(jié)果錯(cuò)誤。實(shí)現(xiàn)計(jì)算器算法使可分sin中是否含pi來(lái)進(jìn)行不同的處理
我的解決辦法如下:
修改代碼途徑
/packages/apps/Calculator/src/com/android/calculator/CalculatorExpressionEvaluator.java
部分源代碼:
輸入的算式經(jīng)過(guò)這個(gè)方法傳入,然后轉(zhuǎn)過(guò)另一個(gè)類(lèi)求出計(jì)算值,該類(lèi)在位置org.javia.arity.Symbols;(被封裝打不開(kāi),只能修改代入值)
public void evaluate(String expr, EvaluateCallback callback) { expr = mTokenizer.getNormalizedExpression(expr); // remove any trailing operators while (expr.length() > 0 && "+-/*".indexOf(expr.charAt(expr.length() - 1)) != -1) { expr = expr.substring(0, expr.length() - 1); } /*try { if (expr.length() == 0 || Double.valueOf(expr) != null) { callback.onEvaluate(expr, null, Calculator.INVALID_RES_ID); return; } } catch (NumberFormatException e) { // expr is not a simple number }*/ if (expr.length() == 0) { callback.onEvaluate(expr, null, Calculator.INVALID_RES_ID); return; } try { /*************代值的代碼在這里**********/ double result = mSymbols.eval(expr); if (Double.isNaN(result)) { callback.onEvaluate(expr, null, R.string.error_nan); } else { /* The arity library uses floating point arithmetic when evaluating the expression leading to precision errors in the result. The method doubleToString hides these errors; rounding the result by dropping N digits of precision.*/ final String resultString = mTokenizer.getLocalizedExpression( Util.doubleToString(result, MAX_DIGITS, ROUNDING_DIGITS)); callback.onEvaluate(expr, resultString, Calculator.INVALID_RES_ID); } } catch (SyntaxException e) { callback.onEvaluate(expr, null, R.string.error_syntax); }}我的解決思路是:
斷某該字符串是否含有”sin( ” ,” cos( ” ,”tan(”字符,并且不含“sin(pi”,“cos(pi”,“tan(pi”, 如果有,在每個(gè)該字符后面添加字符串”pi/180*”
所以我在代入前加了一個(gè)正則表達(dá)式過(guò)濾
public void evaluate(String expr, EvaluateCallback callback) { expr = mTokenizer.getNormalizedExpression(expr); // remove any trailing operators while (expr.length() > 0 && "+-/*".indexOf(expr.charAt(expr.length() - 1)) != -1) { expr = expr.substring(0, expr.length() - 1); } /*try { if (expr.length() == 0 || Double.valueOf(expr) != null) { callback.onEvaluate(expr, null, Calculator.INVALID_RES_ID); return; } } catch (NumberFormatException e) { // expr is not a simple number }*/ if (expr.length() == 0) { callback.onEvaluate(expr, null, Calculator.INVALID_RES_ID); return; } try { /************** 添加的過(guò)濾代碼 ***********/ expr=expr.replaceAll("(?<=(sin|cos|tan)[(])(?!pi)","pi/180*"); double result = mSymbols.eval(expr); if (Double.isNaN(result)) { callback.onEvaluate(expr, null, R.string.error_nan); } else { /* The arity library uses floating point arithmetic when evaluating the expression leading to precision errors in the result. The method doubleToString hides these errors; rounding the result by dropping N digits of precision.*/ final String resultString = mTokenizer.getLocalizedExpression( Util.doubleToString(result, MAX_DIGITS, ROUNDING_DIGITS)); callback.onEvaluate(expr, resultString, Calculator.INVALID_RES_ID); } } catch (SyntaxException e) { callback.onEvaluate(expr, null, R.string.error_syntax); }}然后就能滿足sin90=1了!
PS:這里再為大家推薦幾款計(jì)算工具供大家進(jìn)一步參考借鑒:
在線一元函數(shù)(方程)求解計(jì)算工具:
http://tools.VeVB.COm/jisuanqi/equ_jisuanqi
科學(xué)計(jì)算器在線使用_高級(jí)計(jì)算器在線計(jì)算:
http://tools.VeVB.COm/jisuanqi/jsqkexue
在線計(jì)算器_標(biāo)準(zhǔn)計(jì)算器:
http://tools.VeVB.COm/jisuanqi/jsq
更多關(guān)于Android相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Android開(kāi)發(fā)入門(mén)與進(jìn)階教程》、《Android基本組件用法總結(jié)》、《Android視圖View技巧總結(jié)》、《Android布局layout技巧總結(jié)》及《Android控件用法總結(jié)》
希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。
新聞熱點(diǎn)
疑難解答
圖片精選