青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

流逝的時光
總有一天我們都會離去 email: zzxhang@gmail.com
posts - 21,comments - 111,trackbacks - 0
   為了全面檢查腳本存在的BUG,必須要寫一個相對復雜的程序測試腳本,應VCZH朋友的建議,我寫了這個表達式計算器,400多行代碼...我想沒人會用我這個腳本寫更長的程序了,對吧?因此,調試通過的同時也就意味著測試工作的暫告一段落,接下來得改進內存管理的一些問題.
  鑒于上篇文章被管理員移除的教訓,我決定稍微介紹一下表達式計算的方法,當然都是些比較簡單的東西,考慮這樣一個表達式:2 + (3 +  2) + 4 * 3,結果等于多少?你很和諧的大腦應該能很快就算出一個數字:19,是的,對了,那么你是怎么計算出來的呢?你的大腦思考過程大概就是:一眼看到括號,計算3+2=5,再計算2+5=7,看到乘號,計算4*3=12,最后相加=19.就是說,你的大腦把這個表達式分成了三項,2,(3+2),4*3,分別計算出值再按順序相加得到結果,用電腦計算表達式的處理過程也類似,首先我們根據優先級把表達式分為三個層次:表達式、項、因子,每個加減號左右的操作數為項,每個乘除號左右的操作數為因子,在上面那個表達式中,項為2,(3 +  2),4 * 3,因子為2,(3+2),4,3,所以,我們的處理順序是:計算因子的值-計算項的值-計算表達式的值,這可以通過遞歸實現,想象我們有這么幾個函數:parseExpr,parseTemp,parseFactor,對上面那個式子,首先ParseFactor處理因子2,處理完后返回到ParseTemp再返回到parseExpr,得到正確的操作符+,接著處理第二個項(3 +  2),因為是個表達式,所以到因子級別時遞歸調用parseExpr...一直到達表達式的末尾,為了方便處理,在編譯時,通常會把這樣的表達式翻譯為后綴表達式,2 + (3 +  2) + 4 * 3翻譯成后綴表達式就是2 3 2 + + 4 3 * +,一旦你把式子翻譯成這樣的形式,接下來的處理就很容易了,我們可以用一個堆棧來保存操作數,遇到操作符,就從堆棧中彈出兩個操作數進行運算操作,2 3 2 + + 4 3 * +的處理過程為:
=>[2]                            2入棧
=>[3 2]                         3入棧
=>[2 3 2]                      2入棧
=>+[ (2 3)  2]             讀到操作符+,彈出最上面兩個數2和3相加,得到5
=>[5 2]                          結果入棧
=>+ [(5 2)]                    讀到操作符+,彈出最上面兩個數5和2相加,得到7
=>[7]                             結果入棧
=>[4 7]                          4入棧
=>[3 4 7]                       3入棧
=>*  [(3 4) 7]                 讀到操作符*,彈出最上面兩個數3和4相乘,得到12
=>[12 7]                         結果入棧
=>+ [(12 7)]                   讀到操作符+,彈出最上面兩個數12和7相乘,得到19
=>[19]                            結果入棧

結果保存在棧頂,這樣就完成了整個表達式的計算過程

最后,按照慣例,我貼出用LuckyScript編寫的表達式計算的完整代碼和運行結果,還是有少許問題的,比如表達式字符間不能包含空格,沒有處理浮點數,不過也懶得再改了
  1/*********************************************************************************
  2
  3LuckyScript測試程序:表達式計算器
  4作者:清風
  5
  6**********************************************************************************/

  7
  8#define TOKEN_TYPE_INT                  0          
  9#define TOKEN_TYPE_FLOAT                1  
 10#define TOKEN_TYPE_OPEN_ROUND_BRACKET  2     
 11#define TOKEN_TYPE_CLOSE_ROUND_BRACKET  3         
 12#define TOKEN_OP_TYPE_ADD               4      
 13#define TOKEN_OP_TYPE_SUB               5      
 14#define TOKEN_OP_TYPE_MUL               6      
 15#define TOKEN_OP_TYPE_DIV               7      
 16#define TOKEN_OP_TYPE_MOD               8     
 17#define TOKEN_TYPE_END                  9 
 18#define TOKEN_TYPE_INVALID              10 
 19
 20#define MAX_STACK_SIZE                  100
 21#define true                            1
 22#define false                           0
 23
 24//================================================================================
 25// Experssion
 26//================================================================================
 27class Experssion
 28{
 29public:
 30    var mExprStr;
 31    var mCurrLexeme;
 32    var mStartIndex;    
 33}
;
 34
 35//================================================================================
 36//Stack
 37//================================================================================
 38class Stack
 39{
 40public:
 41    func init()
 42    {
 43          //清空堆棧
 44        mTop = 0;    
 45    }

 46    
 47    func push(var val)
 48    {
 49        if(mTop >= MAX_STACK_SIZE)
 50        {
 51            print("堆棧溢出!");
 52            _getch();    
 53        }
    
 54        
 55        mData[mTop] = val;
 56        mTop ++;
 57    }

 58    
 59    func pop()
 60    {
 61        if(mTop < 0)
 62        {
 63            print("堆棧已沒有元素!");
 64            _getch();    
 65        }
        
 66        
 67        mTop --;
 68        return mData[mTop];
 69    }

 70    
 71    func getTop()
 72    {
 73        return mTop;    
 74    }

 75    
 76    func printAll()
 77    {
 78        for(var i = 0;i < mTop;i ++)
 79        {
 80              print(mData[i]);
 81        }

 82    }

 83    
 84    func get(var index)
 85    {
 86        return mData[index];
 87    }

 88private:
 89    var mTop;
 90    var mData[MAX_STACK_SIZE];
 91}
;
 92
 93//================================================================================
 94//TokenGetter
 95//================================================================================
 96class TokenGetter
 97{
 98public:
 99    //判斷字符是否數字
100    func isCharNumerber(var cChar)
101      {
102            if((cChar >= "0"&& (cChar <= "9"))
103            {
104                  return true;
105              }

106            else
107            {
108                  return false;
109              }

110      }

111      
112      //判斷字符是否分隔符
113      func isSeparator(var cChar)
114    {
115        return (cChar == "*"|| (cChar == "/"|| \
116        (cChar == "+"|| (cChar == "-"|| (cChar == "%"|| \
117        (cChar == "("|| (cChar == ")");
118    }

119
120    //判斷字符串是否數字
121    func isStringNumber(var strString )
122      {
123            if(strlen(strString) == 0)
124              {
125                  return false;    
126              }

127              
128              var strSize = strlen(strString);
129            for(var iCurrCharIndex = 0; iCurrCharIndex < strSize; iCurrCharIndex ++)
130            {
131                  if((! isCharNumerber(strString[iCurrCharIndex])) && (! (strString[iCurrCharIndex] == "-")))
132                  {
133                        return false;
134                    }

135              }

136
137            for(iCurrCharIndex = 1; iCurrCharIndex < strSize; iCurrCharIndex ++)
138            {
139                  if(strString[iCurrCharIndex] == "-")
140                  {
141                        return false;
142                    }

143              }

144            return true;
145      }

146      
147      //參數expr中保存了狀態,返回在此狀態中的下一個token
148      func getNextToken(var expr)
149      {
150          var startIndex = expr.mStartIndex;
151          var exprStr = expr.mExprStr;
152          
153          var size = strlen(exprStr);
154          expr.mCurrLexeme = "";
155          
156          if(exprStr[startIndex] == "")
157          {
158                //已到達字符串的末尾
159              return TOKEN_TYPE_END;    
160          }

161          
162          for(var index = startIndex;index < size;index ++)
163          {
164              if(isSeparator(exprStr[index]))
165              {
166                if(index == startIndex)
167                {
168                      //單個分割符
169                    expr.mCurrLexeme = exprStr[index];    
170                    index ++;
171                }

172                  break;
173              }

174              
175              //如不是分割符,則加到expr.mCurrLexeme中
176              expr.mCurrLexeme = _contractStr(expr.mCurrLexeme,exprStr[index]);    
177          }

178          
179          expr.mStartIndex = index;
180          //返回token
181          switch(expr.mCurrLexeme)
182          {
183          case "+":
184              return TOKEN_OP_TYPE_ADD;
185          case "-":
186              return TOKEN_OP_TYPE_SUB;
187          case "*":
188              return TOKEN_OP_TYPE_MUL;
189          case "/":
190              return TOKEN_OP_TYPE_DIV;
191          case "%":
192              return TOKEN_OP_TYPE_MOD;
193          case "(":
194              return TOKEN_TYPE_OPEN_ROUND_BRACKET;
195          case ")":
196              return TOKEN_TYPE_CLOSE_ROUND_BRACKET;
197          default:
198              if(isStringNumber(expr.mCurrLexeme))
199              {
200                  return TOKEN_TYPE_INT;
201              }

202              else
203              {
204                  print("unknown type!");    
205                  return TOKEN_TYPE_INVALID;
206            }

207              break;
208          }

209          
210          return TOKEN_TYPE_INVALID;
211      }

212}
;
213
214//================================================================================
215//ExprParser
216//================================================================================
217class ExprParser
218{
219public:
220    //分析因子
221    func parseFactor(var expr)
222    {
223        var token = mTokenGetter.getNextToken(expr);
224        
225        //因子有兩種:數字、括號內表達式
226        switch(token)
227        {
228        case TOKEN_TYPE_INT:
229            mSuffixExpr.push(atoi(expr.mCurrLexeme));
230            break;
231         case TOKEN_TYPE_OPEN_ROUND_BRACKET:
232            //遞歸調用
233            parseExpr(expr);
234            token = mTokenGetter.getNextToken(expr);
235            if(token != TOKEN_TYPE_CLOSE_ROUND_BRACKET)
236            {
237                print("expect ')'");
238                _getch();
239            }

240            break;
241        default:
242            print("invalid operand!");
243            _getch();
244            break;        
245        }

246    }

247    
248    //分析項
249    func parseTemp(var expr)
250    {
251        var token;
252        var op;
253        var out = false;
254       
255        //分析第一個因子
256        parseFactor(expr);
257          while(1)
258          
259            token = mTokenGetter.getNextToken(expr);
260            
261            switch(token)
262            {
263            case TOKEN_OP_TYPE_MUL:
264                op = "*";
265                break;
266            case TOKEN_OP_TYPE_DIV:
267                op = "/";
268                break;
269            default:
270                expr.mStartIndex -= strlen(expr.mCurrLexeme);
271                out = true;
272                break;        
273            }

274               
275            if(out)
276            {
277                break;    
278            }

279             
280            parseFactor(expr);
281            
282            //為構造后綴表達式,運算符最后推入棧
283            mSuffixExpr.push(op);
284          }

285    }

286    
287    //分析表達式
288    func parseExpr(var expr)
289    {
290        var token;
291        var op;
292        var out = false;
293       
294        //分析第一項
295        parseTemp(expr);
296        
297          while(1)
298          
299            token = mTokenGetter.getNextToken(expr);
300            switch(token)
301            {
302            case TOKEN_OP_TYPE_ADD:
303                op = "+";
304                break;
305            case TOKEN_OP_TYPE_SUB:
306                op = "-";
307                break;
308             case TOKEN_OP_TYPE_MOD:
309                op = "%";
310                break;
311            case TOKEN_TYPE_END:
312                out = true;
313                break;
314            case TOKEN_TYPE_CLOSE_ROUND_BRACKET:
315                out = true;
316                expr.mStartIndex -= strlen(expr.mCurrLexeme);
317                break;
318            default:
319                out = true;
320                print("invalid operator!");
321                _getch();
322                break;        
323            }

324            
325            if(out)
326            {
327                break;    
328            }

329            
330            parseTemp(expr);
331         
332            //運算符入棧
333            mSuffixExpr.push(op);
334          }

335    }

336    
337    func parse(var expr)
338    {
339          //初始化堆棧
340        mSuffixExpr.init();
341        
342        parseExpr(expr);
343        
344        //返回保存了后綴表達式的堆棧
345        return mSuffixExpr;       
346    }

347    
348private:
349    var mTokenGetter = TokenGetter;
350    var mSuffixExpr = Stack;
351}
;
352
353//================================================================================
354//ExprCalculator
355//================================================================================
356class ExprCalculator
357{
358public:
359    //計算后綴表達式的值
360    func calculate(var suffixExpr)
361    {
362#define PUSH_VALUE(operator) \
363oprend1 = mStack.pop();\
364oprend2 = mStack.pop();\
365mStack.push(oprend1 operator oprend2);          
366
367          var top = suffixExpr.getTop();
368          var element;
369          var oprend1;
370          var oprend2;
371          
372          mStack.init();
373        
374          //得到每一個element,判斷類型,如是操作符就從堆棧中彈出兩個操作數
375          //進行相關運算,如是操作數則推棧保存
376        for(var i = 0;i < top;i ++)
377        {
378              element = suffixExpr.get(i);
379                    
380              switch(element)
381              {
382              case "+":
383                  PUSH_VALUE(+);
384                  break;
385              case "-":
386                  PUSH_VALUE(-);
387                  break;
388              case "*":
389                  PUSH_VALUE(*);
390                  break;
391              case "/":
392                  PUSH_VALUE(/);
393                  break;
394               case "%":
395                  PUSH_VALUE(%);
396                  break;
397              default:
398                  mStack.push(element);
399                  break;
400              }

401        }
    
402        
403        return mStack.get(0);
404    }
    
405    
406private:
407    var mStack = Stack;    
408}
;
409
410//================================================================================
411//Main
412//================================================================================
413func Main()
414{    
415    var parser = new ExprParser();
416    var expr = new Experssion("","",0,0);
417    var calculator = new ExprCalculator();
418    var tokenGetter = new TokenGetter();
419    var suffixExpr;
420    
421    print("**********************************************");
422    newLine();
423    print("LuckyScript測試程序: 表達式計算器");
424    newLine();
425    print("**********************************************");
426    newLine();
427    newLine();
428    
429    while(1)
430    {
431          expr.mStartIndex = 0;
432          print("請輸入一個表達式,如要結束,輸入q: ");
433        expr.mExprStr = getInputString();
434              
435        if(expr.mExprStr == "q")
436        {
437            break;    
438        }

439        
440        suffixExpr = parser.parse(expr);
441        
442        print("后綴表達式:");
443        //輸出后綴表達式
444        suffixExpr.printAll();
445       
446        newLine();
447        
448        print("返回結果:");
449        //計算結果
450        print(calculator.calculate(suffixExpr));
451        
452        newLine();
453    }

454}

455

運行結果:


posted on 2009-03-25 19:08 清風 閱讀(1550) 評論(3)  編輯 收藏 引用 所屬分類: LuckyScript

FeedBack:
# re: LuckyScript測試程序:表達式分析計算
2009-03-25 19:31 | 陳梓瀚(vczh)
囧,居然支持宏,你還不如去支持全局常量和閉包呢……  回復  更多評論
  
# re: LuckyScript測試程序:表達式分析計算
2009-03-25 21:06 | 清風徐來
@陳梓瀚(vczh)
宏處理起來更容易,這個腳本語法上盡量向C++靠攏  回復  更多評論
  
# re: LuckyScript測試程序:表達式分析計算
2009-11-15 19:39 | liubaosen
我的程序不但能解析表達式,還能解析函數:http://armlinux.uueasy.com/read.php?tid=31  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            午夜国产精品视频| 老司机67194精品线观看| 国产尤物精品| 久久永久免费| 亚洲午夜国产成人av电影男同| 久久不射中文字幕| 日韩系列欧美系列| 国产一区二区三区久久久久久久久 | 亚洲精品在线看| 久久久久久国产精品mv| 日韩视频在线观看免费| 国产精品亚洲不卡a| 美女国产一区| 亚洲精品欧美精品| 久久人人精品| 午夜在线电影亚洲一区| 亚洲国产美女精品久久久久∴| 国产精品久久网| 欧美激情精品久久久久久蜜臀| 午夜一区二区三区在线观看| 欧美国产免费| 久久久久成人精品| 亚洲欧美99| 99视频在线精品国自产拍免费观看 | 欧美影院久久久| 99热这里只有精品8| 欧美成人日韩| 久久精品成人| 亚洲一区二区三区在线播放| 亚洲国产精品日韩| 精品999久久久| 国产区亚洲区欧美区| 欧美日韩午夜| 欧美激情精品久久久久久免费印度 | 亚洲大胆人体视频| 欧美一区二区在线免费观看| 亚洲精品美女在线| 在线观看日韩专区| 国内精品亚洲| 国产日韩欧美不卡| 国产欧美日韩一区二区三区在线观看| 欧美三级第一页| 欧美日韩国产综合网| 欧美大色视频| 欧美成人免费全部| 另类综合日韩欧美亚洲| 久久精品国产亚洲高清剧情介绍| 亚洲欧洲av一区二区| 亚洲图片欧美一区| 在线综合亚洲| 亚洲综合色婷婷| 亚洲欧洲日产国产综合网| 美女主播一区| 蜜桃伊人久久| 欧美激情在线| 亚洲激情电影中文字幕| 91久久精品美女| 亚洲人妖在线| 亚洲经典一区| 亚洲精品久久久久中文字幕欢迎你| 亚洲国产91| 亚洲国产精品日韩| 亚洲免费成人av| 亚洲国产天堂久久综合网| 欧美国产日韩一区二区在线观看| 欧美激情视频网站| 亚洲人成网站777色婷婷| 日韩亚洲欧美一区| 亚洲视频在线播放| 欧美一级片在线播放| 久久riav二区三区| 另类专区欧美制服同性| 欧美激情中文不卡| 国产精品www网站| 国产精品综合网站| 在线精品视频一区二区三四| 亚洲精品影视| 亚洲欧美国产精品桃花| 久久精品成人一区二区三区| 美女网站在线免费欧美精品| 亚洲国产精品久久久久久女王| 99精品视频免费全部在线| 亚洲欧美成人一区二区在线电影| 久久精品视频在线看| 欧美成人福利视频| 国产精品a久久久久久| 国产午夜精品一区理论片飘花 | 亚洲午夜一区二区三区| 香蕉久久久久久久av网站| 久久阴道视频| 亚洲精品影院在线观看| 欧美一区二区三区男人的天堂| 久久精品国产第一区二区三区最新章节 | 欧美日韩一区综合| 国模精品一区二区三区色天香| 亚洲免费观看高清完整版在线观看熊| 亚洲欧美www| 麻豆91精品| 一区二区三区四区五区精品| 久久精品一区二区国产| 欧美色精品在线视频| 黑人巨大精品欧美一区二区| 亚洲视频精选在线| 久久精品国产v日韩v亚洲| 亚洲黄色毛片| 久久国产精品色婷婷| 欧美精品一区二区三区一线天视频| 国产精自产拍久久久久久蜜| 亚洲精品欧美一区二区三区| 久久久九九九九| 99国产精品久久久| 美女999久久久精品视频| 国产日韩精品一区二区三区在线| 在线精品视频一区二区| 一区二区三区四区五区精品视频| 狼人社综合社区| 亚洲欧美日韩国产成人精品影院| 男男成人高潮片免费网站| 国产日韩欧美一区二区| 亚洲午夜av电影| 欧美激情女人20p| 久久久精品免费视频| 国产欧美在线| 亚洲欧美日韩国产一区二区三区| 亚洲国产成人在线| 久久精品国产免费看久久精品| 国产精品一区视频网站| 亚洲欧美一区二区三区在线| 在线一区日本视频| 国产精品久久久久久久久久免费看| 一区二区欧美在线观看| 亚洲久色影视| 欧美小视频在线| 亚洲综合电影| 亚洲一区二区在线免费观看视频| 国产精品乱码人人做人人爱| 午夜精品三级视频福利| 亚洲女优在线| 国产午夜亚洲精品不卡| 久久综合五月| 老妇喷水一区二区三区| 亚洲人成在线播放网站岛国| 亚洲欧洲一区二区天堂久久| 欧美日韩免费观看一区三区| 亚洲永久在线| 亚洲欧美视频一区| 伊人久久综合| 亚洲第一在线综合网站| 欧美日韩国产欧美日美国产精品| 99热这里只有精品8| 一本一本久久| 国产日韩免费| 欧美mv日韩mv国产网站| 欧美激情综合在线| 亚洲欧美一区在线| 欧美在线视频二区| 亚洲精品一区在线观看香蕉| 一本色道**综合亚洲精品蜜桃冫 | 亚洲黄色精品| 99在线热播精品免费| 国产情人综合久久777777| 美女精品网站| 欧美日韩亚洲视频| 欧美诱惑福利视频| 老色鬼精品视频在线观看播放| 一区二区国产在线观看| 午夜精品久久久久久久| 亚洲丰满在线| 夜夜嗨av色一区二区不卡| 国产午夜精品久久久久久免费视| 欧美成人三级在线| 国产精品伦一区| 欧美国产日本| 国产精品国内视频| 欧美jizz19性欧美| 欧美性事免费在线观看| 美日韩精品免费观看视频| 欧美日韩视频不卡| 久久久精品久久久久| 欧美—级在线免费片| 久久久99免费视频| 欧美伦理91i| 久久综合狠狠| 国产精品久久久久国产精品日日| 牛牛精品成人免费视频| 国产精品日韩久久久久| 亚洲级视频在线观看免费1级| 国产私拍一区| 日韩视频在线永久播放| 影音先锋成人资源站| 亚洲图片欧洲图片日韩av| 亚洲欧洲精品天堂一级| 香蕉尹人综合在线观看| 夜夜嗨av一区二区三区网站四季av | 日韩一级片网址| 久久久蜜桃精品| 欧美亚洲尤物久久| 欧美日韩色综合| 亚洲黄色免费电影| 在线免费不卡视频|