字典值参考

TJH*_*vel 5 c#

在我们的应用程序中,我们有一些来自翻译的字符串,它们可以包含变量。例如,在Can i have a {beverage}?{beverage}部分应具有可变代替。我当前的实现通过拥有一个包含所有变量的名称和值的字典来工作,并且只替换正确的字符串。但是我想通过引用注册变量,这样如果值被更改,结果字符串也会更改。通常用ref关键字传递参数就可以了,但我不确定如何将它们存储在字典中。

翻译解析器:

static class TranslationParser
{
    private const string regex = "{([a-z]+)}";
    private static Dictionary<string, object> variables = new Dictionary<string,object>();

    public static void RegisterVariable(string name, object value)
    {
        if (variables.ContainsKey(name))
            variables[name] = value;
        else
            variables.Add(name, value);
    }

    public static string ParseText(string text)
    {
        return Regex.Replace(text, regex, match =>
        {
            string varName = match.Groups[1].Value;

            if (variables.ContainsKey(varName))
                return variables[varName].ToString();
            else
                return match.Value;
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

主文件

        string bev = "cola";
        TranslationParser.RegisterVariable("beverage", bev);
        //Expected: "Can i have a cola?"
        Console.WriteLine(TranslationParser.ParseText("Can i have a {beverage}?")); 
        bev = "fanta";
        //Expected: "Can i have a fanta?"
        Console.WriteLine(TranslationParser.ParseText("Can i have a {beverage}?")); 
Run Code Online (Sandbox Code Playgroud)

这是可能的,还是我只是错误地解决了问题?我担心唯一的解决方案会涉及不安全的代码(指针)。

简而言之,我想在字典中存储一个变量,更改原始变量并从字典中获取更改后的值。就像您对ref关键字所做的那样。

tuk*_*aef 2

使用包装器的另一种方法。每次注册变量时都可以包装它们。

class ObjectWrapper
{
    private object _value;

    public ObjectWrapper(object value) 
    {
        _value = value;
    }

    public override string ToString()
    {
        return _value.ToString();
    }
}

static class TranslationParser
{
    private const string regex = "{([a-z]+)}";
    private static Dictionary<string, ObjectWrapper> variables = new Dictionary<string, ObjectWrapper>();

    public static void RegisterVariable(string name, object value)
    {
        var wrapped = new ObjectWrapper(value);
        if (variables.ContainsKey(name))
            variables[name] = wrapped;
        else
            variables.Add(name, wrapped);
    }

    public static string ParseText(string text)
    {
        return Regex.Replace(text, regex, match =>
        {
            string varName = match.Groups[1].Value;

            if (variables.ContainsKey(varName))
                return variables[varName].ToString();
            else
                return match.Value;
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑:

但实际上,我认为如果没有不安全的代码,就不可能以您想要的方式跟踪变量。值类型和对引用类型的引用存储在堆栈中,如果您只是替换引用,则不会影响堆中的实际对象(对存储在字典中的该对象的引用)。所以你需要有堆栈内存的引用(比如指针)。

再次编辑:我错了!

可以使用表达式跟踪任何变量:

class Wrapper
{
    private readonly Dictionary<string, MemberExpression> _registrations = 
        new Dictionary<string, MemberExpression>();

    public void Register<T>(string name, Expression<Func<T>> expr)
    {
        _registrations[name] = (MemberExpression)expr.Body;
    }

    public object GetValue(string name)
    {
        var expr = _registrations[name];
        var fieldInfo = (FieldInfo)expr.Member;
        var obj = ((ConstantExpression)expr.Expression).Value;
        return fieldInfo.GetValue(obj);
    }
}
private static void Main(string[] args)
{
    var wrapper = new Wrapper();
    int x = 0;
    storage.Register("x", () => x);
    Console.WriteLine(wrapper.GetValue("x")); //0
    x = 1;
    Console.WriteLine(wrapper.GetValue("x")); //1
}
Run Code Online (Sandbox Code Playgroud)