Expression类编译表达式

如何实现自己的编译器,微软已经给大家现成的轮子了。微软的Expression类提供了一套拼接、编译Lambda表达式的完整方法,可以用它轻松定义你自己的语法


RpnExpression方法:将中缀表达式转换为逆波兰表达式。用关键字将表达式字符串分割为一个数组;按照优先级出栈入栈;返回一个逆波兰表达式顺序的字符串列表。

ComplieRpnExp方法:根据逆波兰表达式顺序,依次弹出运算符转换为Expression的各子类如二元表达式BinaryExpression、条件表达式ConditionalExpression、常数表达式ConstantExpression等;参数首先判断是否常数,如果不是,则调用GetTagExpression方法,将字符串转换为方法调用MethodCallExpression,最终会将该参数编译为一个Tag。经过处理最终返回一个LambdaExpression。

Eval方法将LambdaExpression编译为一个委托;相关的Tag加入列表TagList。


源码:

using System.Linq.Expressions;

namespace ConsoleApp2
{

    internal class Program
    {
        static void Main(string[] args)
        {
            var compiler = new ExpressionCompiler();
            string expression = "3+4"; // 忽略等号  
            var lambda = compiler.CompileInfixExp(expression);
            int result = compiler.Eval<int>(lambda);
            Console.WriteLine(result); // 输出 7  
        }
    }

    public class ExpressionCompiler
    {
        // 假设的简单中缀到逆波兰的转换(这里直接返回中缀,因为简单)  
        public List<string> RpnExpression(string infixExpression)
        {
            // 在真实场景中,这里会有更复杂的逻辑来处理运算符优先级和括号  
            return infixExpression.Split(new[] { '+', '-', '*', '/' }, StringSplitOptions.RemoveEmptyEntries).ToList();
        }

        // 将中缀表达式编译为Lambda表达式  
        public LambdaExpression CompileInfixExp(string infixExpression)
        {
            var tokens = infixExpression.Split(new[] { '+', '-', '*', '/' }, StringSplitOptions.RemoveEmptyEntries);
            var expressions = new List<Expression>();

            foreach (var token in tokens)
            {
                if (int.TryParse(token, out int value))
                {
                    expressions.Add(Expression.Constant(value));
                }
            }
            // 这里简化为只处理加法,且假设前面至少有两个操作数  
            var right = expressions.Last();
            expressions.RemoveAt(expressions.Count - 1);
            var left = expressions.Last();
            expressions.RemoveAt(expressions.Count - 1);
            expressions.Add(Expression.Add(left, right));

            // 假设表达式只有一个结果  
            return Expression.Lambda(expressions.First());
        }

        // 执行Lambda表达式并返回结果  
        public T Eval<T>(LambdaExpression lambda)
        {
            return (T)lambda.Compile().DynamicInvoke();
        }

    }
}


本文出自勇哥的网站《少有人走的路》wwww.skcircle.com,转载请注明出处!讨论可扫码加群:

发表评论:

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

会员中心
搜索
«    2025年4月    »
123456
78910111213
14151617181920
21222324252627
282930
网站分类
标签列表
最新留言
    热门文章 | 热评文章 | 随机文章
文章归档
友情链接
  • 订阅本站的 RSS 2.0 新闻聚合
  • 扫描加本站机器视觉QQ群,验证答案为:halcon勇哥的机器视觉
  • 点击查阅微信群二维码
  • 扫描加勇哥的非标自动化群,验证答案:C#/C++/VB勇哥的非标自动化群
  • 扫描加站长微信:站长微信:abc496103864
  • 扫描加站长QQ:
  • 扫描赞赏本站:
  • 留言板:

Powered By Z-BlogPHP 1.7.2

Copyright Your skcircle.com Rights Reserved.

鄂ICP备18008319号


站长QQ:496103864 微信:abc496103864