假设我有这样一个类:
[JsonObject(MemberSerialization.OptIn)]
public class TestClass
{
private readonly int _SomeField;
[JsonProperty(nameof(InputInfo))]
public ref readonly int SomeField => ref _SomeField;
}
注意:这个类是一个精简的示例,在我的真实世界场景中,
_SomeField
字段不是成员字段,否则我会刚刚添加了Json属性。该字段是由另一个对象公开的字段,该对象是该类的成员。公共属性只是将该字段暴露给用户,以便更容易访问该值。,实际属性tipe是不是
int
,而是12字节struct
,所以我通过引用返回它为了避免按值进行无用的复制。
我正在使用JsonConvert.SerializeObject(this, Formatting.Indented)
序列化这样的类。
Newtonsoft.Json在转换值时抛出异常,说它无法访问字段/属性值(我猜它是ref
参数会使库使用的反射过程崩溃。)< / p>
我尝试使用自定义JsonConverter
进行试验,但是在使用任何其他转换器之前就发生了崩溃。
我知道一个快速的解决方案是添加一个辅助私有参数,它只返回该字段作为值而不是通过引用,并且只将它用于Json序列化,但对我来说看起来很糟糕(而且我有禁用有关未使用的私有参数的自动VS警告),如果可能的话,我正在寻找更好的解决方案(不引入无用的字段/属性)。
感谢您的帮助!
答案 0 :(得分:2)
这个评论太长了,如果有人发布另一个答案,我会删除它。快速查看,目前无法覆盖它。
问题出现在DynamicValueProvider.cs第110行:
public object GetValue(object target)
{
try
{
if (_getter == null)
{
_getter = DynamicReflectionDelegateFactory.Instance.CreateGet<object>(_memberInfo);
}
return _getter(target); //Line 100
}
catch (Exception ex)
{
throw new JsonSerializationException("Error getting value from '{0}' on '{1}'.".FormatWith(CultureInfo.InvariantCulture, _memberInfo.Name, target.GetType()), ex);
}
}
原因在CreateGet
,它无法生成正确处理这些类型的方法。也许你应该在GitHub上打开一个新问题(如果还没有)。
下面你可以看到一个重现问题的小应用程序:
using System;
using System.Reflection;
using System.Reflection.Emit;
namespace ConsoleApp15
{
public class TestClass
{
public TestClass()
{
_SomeField = 42;
}
private readonly int _SomeField;
public ref readonly int SomeField => ref _SomeField;
}
internal class Program
{
private static void Main(string[] args)
{
var propertyInfo = typeof(TestClass).GetProperty("SomeField");
var getMethod = CreateGet<object>(propertyInfo);
TestClass obj = new TestClass();
var result = getMethod(obj);
}
public static Func<T, object> CreateGet<T>(PropertyInfo propertyInfo)
{
DynamicMethod dynamicMethod = CreateDynamicMethod("Get" + propertyInfo.Name, typeof(object), new[] { typeof(T) }, propertyInfo.DeclaringType);
ILGenerator generator = dynamicMethod.GetILGenerator();
GenerateCreateGetPropertyIL(propertyInfo, generator);
return (Func<T, object>)dynamicMethod.CreateDelegate(typeof(Func<T, object>));
}
private static DynamicMethod CreateDynamicMethod(string name, Type returnType, Type[] parameterTypes, Type owner)
{
DynamicMethod dynamicMethod = new DynamicMethod(name, returnType, parameterTypes, owner, true);
return dynamicMethod;
}
private static void GenerateCreateGetPropertyIL(PropertyInfo propertyInfo, ILGenerator generator)
{
MethodInfo getMethod = propertyInfo.GetGetMethod(true);
if (getMethod == null)
{
throw new ArgumentException("Property " + propertyInfo.Name + " does not have a getter.");
}
if (!getMethod.IsStatic)
{
generator.PushInstance(propertyInfo.DeclaringType);
}
generator.CallMethod(getMethod);
generator.BoxIfNeeded(propertyInfo.PropertyType);
generator.Return();
}
}
internal static class ILGeneratorExtensions
{
public static void PushInstance(this ILGenerator generator, Type type)
{
generator.Emit(OpCodes.Ldarg_0);
if (type.IsValueType)
{
generator.Emit(OpCodes.Unbox, type);
}
else
{
generator.Emit(OpCodes.Castclass, type);
}
}
public static void PushArrayInstance(this ILGenerator generator, int argsIndex, int arrayIndex)
{
generator.Emit(OpCodes.Ldarg, argsIndex);
generator.Emit(OpCodes.Ldc_I4, arrayIndex);
generator.Emit(OpCodes.Ldelem_Ref);
}
public static void BoxIfNeeded(this ILGenerator generator, Type type)
{
if (type.IsValueType)
{
generator.Emit(OpCodes.Box, type);
}
else
{
generator.Emit(OpCodes.Castclass, type);
}
}
public static void UnboxIfNeeded(this ILGenerator generator, Type type)
{
if (type.IsValueType)
{
generator.Emit(OpCodes.Unbox_Any, type);
}
else
{
generator.Emit(OpCodes.Castclass, type);
}
}
public static void CallMethod(this ILGenerator generator, MethodInfo methodInfo)
{
if (methodInfo.IsFinal || !methodInfo.IsVirtual)
{
generator.Emit(OpCodes.Call, methodInfo);
}
else
{
generator.Emit(OpCodes.Callvirt, methodInfo);
}
}
public static void Return(this ILGenerator generator)
{
generator.Emit(OpCodes.Ret);
}
}
}