gpt4 book ai didi

c# - 在 C# 中,如何在运行时创建值类型变量?

转载 作者:行者123 更新时间:2023-11-30 17:27:50 25 4
gpt4 key购买 nike

我正在尝试实现如下方法:

(Func<T> getFn, Action<T> setFn) MakePair<T>(T initialVal) {
}

它将返回两个运行时生成的 lambda,使用 Expression 获取和设置动态创建的变量树来创建代码。

我目前的解决方案是使用一个元素动态创建一个类型的数组,并引用它:

(Func<T> getFn, Action<T> setFn) MakePair<T>(T initialVal) {
var dynvar = Array.CreateInstance(typeof(T), 1);
Expression<Func<Array>> f = () => dynvar;
var dynref = Expression.Convert(f.Body, typeof(T).MakeArrayType());
var e0 = Expression.Constant(0);
var getBody = Expression.ArrayIndex(dynref, e0);
var setParam = Expression.Parameter(typeof(T));
var setBody = Expression.Assign(Expression.ArrayAccess(dynref, e0), setParam);

var getFn = Expression.Lambda<Func<T>>(getBody).Compile();
var setFn = Expression.Lambda<Action<T>>(setBody, setParam).Compile();

return (getFn, setFn);
}

有没有比使用数组更好的方法来创建可以在运行时读取/写入的值类型变量?

除了使用 lambda 创建用于 ArrayIndex 的(字段?)引用之外,是否有更好的方法来引用运行时创建的数组?/ArrayAccess方法调用?

过多的背景信息对于那些想知道的人,最终这是为了尝试创建类似 Perl 自动验证 Perl 散列左值的东西。

假设您有一个 List<T>有重复的元素,想创建一个 Dictionary<T,int>这使您可以查找每个唯一 T 的计数在列表中。可以用几行代码来统计(这里的T就是int):

var countDict = new Dictionary<int, int>();
foreach (var n in testarray) {
countDict.TryGetValue(n, out int c);
countDict[n] = c + 1;
}

但我想用 LINQ 来做这件事,我想避免双重索引 countDict (有趣的是,ConcurrentDictionaryAddOrUpdate 用于此目的)所以我使用聚合:

var countDict = testarray.Aggregate(new Dictionary<int,int>(), (d, n) => { ++d[n]; return d; });

但这有几个问题。一、Dictionary不会为缺失值创建值,因此您需要一种新类型的 Dictionary使用例如自动创建缺失值种子 lambda:

var countDict = testarray.Aggregate(new SeedDictionary<int, Ref<int>>(() => Ref.Of(() => 0)), (d, n) => { var r = d[n]; ++r.Value; return d; });

但是你仍然有左值问题,所以你替换了普通的 int柜台 Ref类(class)。不幸的是,C# 无法创建 C++ 第一类 Ref类,但是使用一个基于从 getter lambda(使用表达式树)自动创建 setter lambda 的类已经足够接近了。 (不幸的是,C# 仍然不接受 ++d[n].Value;,即使它应该是有效的,所以你必须创建一个临时的。)

但是现在您遇到了创建多个运行时整数变量来保存计数的问题。我扩展了 Ref<>类采用返回常量 (ConstantExpression) 的 lambda 并创建一个运行时变量,并构建一个以常量为初始值的 getter 和 setter。

最佳答案

我同意一些问题评论者的观点,即表达式树似乎是不必要的,所以这里是一个简单的 API 实现,没有它们:

struct Box<T> {
public T Value;
}

(Func<T> getFn, Action<T> setFn) MakePair<T>(T initialVal) {
var box = new Box<T> { Value = initialVal };
return (() => box.Value, v => box.Value = v);
}

作为对所述问题的回答(如何在没有 lambda 的情况下定义 dynref),那么,对 dynvar 的以下修改是否有问题动态引用?

var dynvar = new T[] { initialVal };
var dynref = Expression.Constant(dynvar);

关于c# - 在 C# 中,如何在运行时创建值类型变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54120167/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com