DREvalKit 一個使用簡單又強大的表達式計算庫

demo下載地址

一. 簡介

  1. 一個數(shù)學表達式計算器舵匾,能實現(xiàn)和 UIWebView 的 stringByEvaluatingJavaScriptFromString: 一樣的計算效果窖张,但效率要高很多簸淀,可以在子線程中執(zhí)行青扔;

  2. 基本全面覆蓋 NSExpression 的 expressionForFunction:arguments 中的所有function晓猛,使用要比 NSExpression 簡單很多,只需將注意力放大expression表達式的編輯上险领,將任意復(fù)雜度的表達式侨舆,通過eval:方法傳入便可輕松得到計算結(jié)果;

  3. 支持復(fù)雜加減乘除四則運算舷暮,與或非邏輯運算态罪,和大于小于等比較運算噩茄;

  4. 支持三目運算下面;

  5. 表達式中能自動識別處理的函數(shù),基本全部覆蓋NSExpression绩聘,有的未實現(xiàn)的沥割,因為可以自己有數(shù)學表達式表達,比如 a+b凿菩,這個表達式計算最基本功能机杜,無需通過函數(shù)調(diào)用來實現(xiàn);

  6. 以上所述的計算類型在符合數(shù)學表達式邏輯的前提下衅谷,可以組合在一個表達式中椒拗,函數(shù)支持嵌套調(diào)用;

  7. 支持字符串相加(字符串拼接)获黔;

  8. 開發(fā)者可以擴展自己的函數(shù)蚀苛,通過構(gòu)建 DFEvaluatorFunction 對象來聲明自定義的函數(shù),詳細使用方式可以參考 demo玷氏。

二. 使用方式

  1. 引入頭文件

    #import <DFEvalKit/DFEvaluator.h>
    

    DFEvaluator.h 文件中只聲明了4個方法:

    • -(id)eval:(NSString *)expression // 用于傳入表達式進行計算并返回計算結(jié)果
    • -(void)setCustomFunctions:(NSDictionary *)customFunctions // 用于給開發(fā)者注冊自定義方法
    • -(void)setDateFormatter:(NSDateFormatter *)dateFormatter // 設(shè)置支持的日期格式堵未,默認只支持 yyyy-MM-dd HH:mm:ss 格式
    • -(void)withoutFunctionTransfer:(BOOL)withoutFunction; // 不支持函數(shù)調(diào)用,僅用于計算純數(shù)學表達式盏触,默認為支持函數(shù)調(diào)用

開發(fā)者僅需通過這4個 API 來使用表達式計算全部功能

// DFEvaluator.h
@interface DFEvaluator : NSObject

#pragma mark - API
/**
 *  表達式計算
 *
 *  @param expression 需要計算的表達式
 *
 *  @return 計算結(jié)果
*/
- (id)eval:(NSString *)expression;

/**
 *  設(shè)置開發(fā)者自定義的函數(shù)集
 *
*  @param customFunctions 每個函數(shù)用 DFEvaluatorFunction 對象來描述渗蟹,以函數(shù)名為 key
 *                         
 */
- (void)setCustomFunctions:(NSDictionary *)customFunctions;

/**
 *  設(shè)置支持的日期格式块饺,默認只支持 yyyy-MM-dd HH:mm:ss 格式
 *
*  @param dateFormat 日期格式
 */
- (void)setDateFormat:(NSString *)dateFormat;
    
/**
*  不支持函數(shù)調(diào)用,僅用于計算純數(shù)學表達式雌芽,默認為支持函數(shù)調(diào)用
 */
- (void)withoutFunctionTransfer:(BOOL)withoutFunction;

@end
  1. 具體使用方式

    1. 支持的數(shù)學運算操作符和操作數(shù)類型

      typedef NS_ENUM(NSInteger, DFEvaluatorNodeType)
      {
          /**
           * 未知  0
           */
          Unknown,  
            
          /**
           * + 加
           */
          Plus,
          
          /**
           * - 減
           */
          Subtract,
          
          /**
           * * 乘
           */
          MultiPly,
          
          /**
           * / 除
           */
          Divide,
          
          /**
           * ( 左括號
           */
          LParentheses,
          
          /**
           * ) 右括號
           */
          RParentheses,
          
          /**
           * % 求模,取余
           */
          Mod,
          
          /**
           * ^ 冪運算
           */
          Power,
          
          /**
           * << 左移位
           */
          LShift,
          
          /**
           * >> 右移位
           */
          RShift,
          
          /**
           * & 按位與
           */
          BitwiseAnd,
          
          /**
           * | 按位或
           */
          BitwiseOr,
          
          /**
           * && 邏輯與
           */
          And,
          
          /**
           * || 邏輯或
           */
          Or,
          
          /**
           * ! 邏輯非
           */
          Not,
          
          /**
           * == 比較等
           */
          Equal,
          
          /**
           * != 或 <> 比較不等
           */
          Unequal,
          
          /**
           * > 比較大于
           */
          GT,
          
          /**
           * < 比較小于
           */
          LT,
          
          /**
           * >= 比較大于等于
           */
          GTOrEqual,
          
          /**
           * <= 比較小于等于
           */
          LTOrEqual,
          
          /**
           * 數(shù)值
           */
          Numeric,
          
          /**
           * 字符串
           */
          String,
          
          /**
           * 日期時間
           */
          Datetime
      };
      
    2. 使用示例

      // 簡單四則運算
      [DFEvaluator eval:@"22 + 33 * 66 + 3^5"]; // 3^5 3的5次方
      
      // 簡單比較運算
      [DFEvaluator eval:@"5 < 6"];
      
      // 邏輯運算
      [DFEvaluator eval:@"5 < 3 || 6 > 5)"];
      
      // 位運算
      [DFEvaluator eval:@"4 << 5"];
      
      // 字符串相加
      [DFEvaluator eval:@"\"Hello\" + \" \" + \"World\" << 5"];
      
    3. 包含函數(shù)的運算

      1. 支持的函數(shù)清單

        // 邏輯運算類
        ternaryOperation(5<7, \"真\", \"假\") // 三目表達式
        
        日期類處理方法, 日期字符串格式要求為:yyyy-MM-dd或者yyyy-MM-dd HH:mm:ss
        dateDiff(差值類型, 較早日期, 較晚日期) // 時間差值
        getYear(date) // 獲取日期中的年份
        getQuarter(date) // 獲取日期中的第幾季度
        getLocalQuarter(date) // 獲取日期中的中文第幾季度
        getMonth(date) // 獲取日期中的月份
        getLocalMonth(date) // 獲取日期中的中文月份
        getWeek(date) // 獲取日期中的第幾周
        getLocalWeek(date) // 獲取日期中的中文第幾周
        getDayOfWeek(date) // 獲取日期中的星期幾
        getLocalDayOfWeek(date) // 獲取日期中的中文星期幾
        getDay(date) // 獲取日子
        getLocalDay // 獲取中文日子
        now() // 獲取現(xiàn)在時間
        
        // 數(shù)值類
        getLocalMoney(digit) // 將數(shù)值轉(zhuǎn)換為大寫金額
        round(digit) // 數(shù)值四舍五入
        ceil(digit) // 數(shù)值0舍1入
        trunc(digit) // 向下取整
        floor(digit) // 向下取整
        abs(digit) // 求絕對值
        sqrt(digit) // 開平方
        log(digit) // 底數(shù)為e對數(shù)
        ln(digit) // 底數(shù)為e對數(shù)
        log10(digit) // 底數(shù)為10對數(shù)
        log2(digit) // 底數(shù)為2對數(shù)
        raiseToPower(x, n) // 計算 x 的 n 次方
        exp(digit) // 求e的x次方
        bitwiseXor(a, b) // a 異或 b
        onesComplement(a) // a 的補碼
        average(digit, digit, ...) // 求平均
        sum(digit, digit, ...) // 求和
        count(digit, digit, ...) // 計數(shù)
        min(digit, digit, ...) // 找最小值
        max(digit, digit, ...) // 找最大值
        median(digit, digit, ...) // 找中值
        mode(digit, digit, ...) // 一數(shù)組或數(shù)據(jù)區(qū)域中出現(xiàn)頻率最多的數(shù)值
        stddev(digit, digit, ...) // 樣本標準偏差
        random(void) // 獲取隨機數(shù)小數(shù)
        randomn(digit) // 獲取隨機數(shù)整數(shù)
        
        // 字符串類
        contains("待檢字符串", "被包含字符串") // 檢查包含子字符串
        unContains("待檢字符串", "不被包含字符串") // 檢查不包含子字符串
        lowercase("字符串") // 轉(zhuǎn)小寫
        uppercase("字符串") // 轉(zhuǎn)大寫
        
      2. 調(diào)用方式

        // 三目運算函數(shù)
        [DFEvaluator eval:@"ternaryOperation(5<7, \"真\", \"假\")"];
        
        // 獲取大寫金額
        [DFEvaluator eval:@"getLocalMoney(10086)"];
        
        // 復(fù)雜混合運算表達式
        [DFEvaluator eval:@"dateDiff(\"dd\", \"2016-12-17\", now()) * 10 - getYear(now()) + max(11, 22,33,1000) * sqrt(floor(1000.445))"];
        
  2. 自定義函數(shù)的使用授艰,以 demo 為例:
    第一步:自定義方法的OC實現(xiàn)
    demo 中在 ViewController.m 實現(xiàn)了如下四個方法,可以看到返回時世落,均構(gòu)建了 DFEvaluatorFunctionResult 類實例來返回想诅,這是必須的;

    方法中傳入的 param 會根據(jù)表達式中調(diào)用函數(shù)時括號內(nèi)傳入的參數(shù)情況解析成字符串岛心,一維數(shù)組来破,或者二維數(shù)組,具體規(guī)則看如下代碼段的注釋忘古。

    #pragma mark - 自定義函數(shù)測試
    /*
     *  不帶參函數(shù)
     *  在表達式中寫入 test1()
    *
    *  @return 創(chuàng)建 DFEvaluatorResult 實例徘禁,返回函數(shù)運行結(jié)果
     */
    - (DFEvaluatorFunctionResult *)test1
    {
         return [[DFEvaluatorFunctionResult alloc] initWithResult:@"測試不帶參函數(shù)"
                                                                 dataType:DFEvaluatorFunctionResultDataTypeString] ;
    }
            
    /*
     *  帶一個加單參數(shù)的函數(shù)
     *
     *  @param param 如在表達式中寫:test2(123) 則此處 param 為: @"123"
     *
     *  @return 創(chuàng)建 DFEvaluatorFunctionResult 實例,返回函數(shù)運行結(jié)果
    */
    - (DFEvaluatorFunctionResult *)test2:(id)param
    {
         return [[DFEvaluatorFunctionResult alloc] initWithResult:[NSString stringWithFormat:@"測試帶參函數(shù)髓堪,傳入?yún)?shù)為:%@", param]
                                                                 dataType:DFEvaluatorFunctionResultDataTypeString];
    }
            
    /*
     *  一維多參函數(shù)送朱,將所有參數(shù)拼接成一個字符串
     *  如表達式中寫:test3(123, 456, 789...) 數(shù)量根據(jù)自己的需要來定
     *
     *  @param param 此處得到 param 為一維數(shù)組 @[@"123", @"456", @"789"...]
     *
     *  @return 創(chuàng)建 DFEvaluatorFunctionResult 實例,返回函數(shù)運行結(jié)果
     */
    - (DFEvaluatorFunctionResult *)test3:(id)param
    {
        // 將所有參數(shù)拼接成一個字符串
                
         NSMutableString *result = [NSMutableString string];
         for(NSString *str in param)
         {
             [result appendString:str];
          }
                
         return [[DFEvaluatorFunctionResult alloc] initWithResult:result
                                                                 dataType:DFEvaluatorFunctionResultDataTypeString];
    }
            
            /*
             *  二維多參函數(shù)干旁,函數(shù)功能為將所有參數(shù)拼接為字符串
             *  如表達式中寫:test3(123, [456, 789], @"333", [234]...) 數(shù)量根據(jù)自己的需要來定
             *
             *  @param param 此處得到 param 為二維數(shù)組 @[@"123", @[@"456", @"789"], @"333", @[@"234"]...]
             *
             *  @return 創(chuàng)建 DFEvaluatorFunctionResult 實例驶沼,返回函數(shù)運行結(jié)果
             */
            - (DFEvaluatorFunctionResult *)test4:(id)param
            {
                NSMutableString *result = [NSMutableString string];
                
                for(id obj in param)
                {
                    if([obj isKindOfClass:[NSArray class]])
                    {
                        for(NSString *str in (NSArray *)obj)
                        {
                            [result appendString:str];
                        }
                    }
                    else
                    {
                        [result appendString:obj];
                    }
                }
                
                return [[DFEvaluatorFunctionResult alloc] initWithResult:result
                                                                 dataType:DFEvaluatorFunctionResultDataTypeString];
            }
    

    第二步:構(gòu)建 DFEvaluatorFunction 實例
    如下代碼將上述四個 test 方法分別構(gòu)建一個 DFEvaluatorFunction 實例來進行描述,并以用于表達式調(diào)用的函數(shù)名為 key 存入字典争群,準備注入表達式解析計算器中回怜。

    - (NSDictionary *)customFunctions
    {
        if(!_customFunctions)
        {
            _customFunctions = [NSMutableDictionary dictionary];
            // test1 無參函數(shù)
            DFEvaluatorFunction *function = [[DFEvaluatorFunction alloc] initWithFunctionName:@"test1"
                                                                                                                     selector:@selector(test1)
                                                                                                                       target:self];
            [_customFunctions setObject:function forKey:function.functionName];
            // test2 帶一個參數(shù)的函數(shù)
            function = [[DFEvaluatorFunction alloc] initWithFunctionName:@"test2"
                                                                                    selector:@selector(test2:)
                                                                                      target:self];
            [_customFunctions setObject:function forKey:function.functionName];
            // test3 帶多個一維參數(shù)的函數(shù)
            function = [[DFEvaluatorFunction alloc] initWithFunctionName:@"test3"
                                                                                    selector:@selector(test3:)
                                                                                      target:self];
            [_customFunctions setObject:function forKey:function.functionName];
            // test4 帶多個二維參數(shù)的函數(shù)
            function = [[DFEvaluatorFunction alloc] initWithFunctionName:@"test4"
                                                                                    selector:@selector(test4:)
                                                                                      target:self];
            [_customFunctions setObject:function forKey:function.functionName];
                }
            return _customFunctions;
        }
    

    第三步:將構(gòu)建好的 DFEvaluatorFunction 實例注入表達式解析計算器
    代碼如下,即在 demo 中點擊 “計算” 按鈕時執(zhí)行的代碼

    /*
     *  創(chuàng)建表達式計算器對象
    */
    - (DFEvaluator *)evaluator
    {
        if(!_evaluator)
        {
            _evaluator = [[DFEvaluator alloc] init];
            [_evaluator setCustomFunctions:self.customFunctions]; // 注入自定義函數(shù)集
                    
            // 默認就是這個格式
            // [_evaluator setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
                    
            // 默認就是 false换薄,即表達式支持函數(shù)調(diào)用玉雾,當表達式不需要函數(shù)調(diào)用是,調(diào)用該方法置為 true轻要,可以調(diào)高運算效率
            // [_evaluator withoutFunctionTransfer:false];
        }
        return _evaluator;
    }
    

最后上圖看看運行效果

1. 開始運行.png
2. 選擇函數(shù).png
3. 選擇指定測試用例.png
4. 選擇確定輸入到了輸入框.png
5. 表達式無誤完成計算.png
6. 表達式有錯誤計算失敗.png



感興趣請下載demo研究复旬,運行后,點擊快速測試冲泥,快速一睹 DFEvaluator 的風采吧驹碍!

意見建議請聯(lián)系:

QQ: 247159603
**博客:猿視界

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市凡恍,隨后出現(xiàn)的幾起案子志秃,更是在濱河造成了極大的恐慌,老刑警劉巖咳焚,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件洽损,死亡現(xiàn)場離奇詭異,居然都是意外死亡革半,警方通過查閱死者的電腦和手機碑定,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門流码,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人延刘,你說我怎么就攤上這事漫试。” “怎么了碘赖?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵驾荣,是天一觀的道長。 經(jīng)常有香客問我普泡,道長播掷,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任撼班,我火速辦了婚禮歧匈,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘砰嘁。我一直安慰自己件炉,他們只是感情好,可當我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布矮湘。 她就那樣靜靜地躺著斟冕,像睡著了一般。 火紅的嫁衣襯著肌膚如雪缅阳。 梳的紋絲不亂的頭發(fā)上磕蛇,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天,我揣著相機與錄音券时,去河邊找鬼孤里。 笑死伏伯,一個胖子當著我的面吹牛橘洞,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播说搅,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼炸枣,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了弄唧?” 一聲冷哼從身側(cè)響起适肠,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎候引,沒想到半個月后侯养,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡澄干,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年逛揩,在試婚紗的時候發(fā)現(xiàn)自己被綠了柠傍。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡辩稽,死狀恐怖惧笛,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情逞泄,我是刑警寧澤患整,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站喷众,受9級特大地震影響各谚,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜到千,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一嘲碧、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧父阻,春花似錦愈涩、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至斟览,卻和暖如春毁腿,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背苛茂。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工已烤, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人妓羊。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓胯究,卻偏偏與公主長得像,于是被迫代替她去往敵國和親躁绸。 傳聞我的和親對象是個殘疾皇子裕循,可洞房花燭夜當晚...
    茶點故事閱讀 44,979評論 2 355

推薦閱讀更多精彩內(nèi)容