gpt4 book ai didi

parameter-passing - AutoMapper - 将参数传递给自定义解析器的奇怪行为

转载 作者:行者123 更新时间:2023-12-04 23:00:44 27 4
gpt4 key购买 nike

虽然我对 AutoMapper 比较陌生,但我在我正在开发的一个小项目中使用它。我以前从未遇到过使用它的问题,但现在我在将参数传递给自定义解析器时遇到了一些奇怪的行为。

场景如下:我从存储库中获取消息列表,然后将它们映射到它的前端友好版本。没什么特别的,只是对象之间的一些正常映射。我在该前端对象中有一个字段,它告诉某个用户是否已经为该消息投票,这就是我使用自定义解析器的目的(这是第二个“ForMember”):

    public List<SupportMessageUi> GetAllVisible(string userId)
{
Mapper.CreateMap<SupportMessage, SupportMessageUi>()
.ForMember(dest => dest.Votes,
opt => opt.ResolveUsing<SupportMessageVotesResolver>())
.ForMember(dest => dest.UserVoted,
opt => opt.ResolveUsing<SupportMessagesUserVotedResolver>()
.ConstructedBy(() => new SupportMessagesUserVotedResolver(userId)));

var messages = _unitOfWork.MessagesRepository.Get(m => m.Visible);

var messagesUi = Mapper.Map<List<SupportMessageUi>>(messages);

return messagesUi;
}

我在 Web 服务上调用此方法,问题是:我第一次调用 Web 服务(使用 Web 服务控制台)时,它都运行良好。例如,如果我将 '555' 作为 userId 传递,我将使用正确的值访问此方法:

enter image description here

在自定义解析器中,值被正确传递给构造函数:
enter image description here

返回的结果是正确的。问题来了。我第二次调用服务时,传递了一个不同的参数(这次是'666'),到达自定义解析器的构造函数的参数是旧的参数('555')。这就是我的意思:

在映射对象之前,我们可以看到传递给构造函数的值是正确的('666'):
enter image description here

但是当它到达解析器的构造函数时,值是错误的,并且是旧的('555'):
enter image description here

对服务的所有后续调用都使用自定义解析器构造函数 ('555') 中的原始值,与我传递给服务的值无关(如果我从另一个浏览器进行调用,也会发生这种情况)。如果我关闭服务器并重新启动它,我可以传递一个新参数(将在所有其他调用中使用,直到我再次关闭它)。

知道为什么会这样吗?

最佳答案

这是因为 AutoMapper.CreateMap是静态方法,只需要调用一次。与 CreateMap您的网络方法中的代码,您每次在网络服务上调用该方法时都试图调用它。自网络服务器 流程 在通话之间保持事件状态(除非你重新启动它,就像你说的那样)然后静态映射保持在原位。因此,调用 AutoMapper.Reset 的必要性,正如您在回答中所说。

但建议您将映射创建放在 AppStartGlobal或静态构造函数或其他任何东西,所以你只调用一次。有一些方法可以调用 Map 允许您传入值,因此您无需尝试使用 ValueResolver 的构造函数来处理事情。 .

这是一个使用 ValueResolver 的示例(注意实现 IValueResolver 而不是继承 ValueResolver<TSource, TDestination> 的更改):

[Test]
public void ValueTranslator_ExtraMapParameters()
{
const int multiplier = 2;
ValueTranslator translator = new ValueTranslator();
Mapper.AssertConfigurationIsValid();

ValueSource source = new ValueSource { Value = 4 };
ValueDest dest = translator.Translate(source, multiplier);
Assert.That(dest.Value, Is.EqualTo(8));

source = new ValueSource { Value = 5 };
dest = translator.Translate(source, multiplier);
Assert.That(dest.Value, Is.EqualTo(10));
}

private class ValueTranslator
{
static ValueTranslator()
{
Mapper.CreateMap<ValueSource, ValueDest>()
.ForMember(dest => dest.Value, opt => opt.ResolveUsing<ValueResolver>().FromMember(src => src.Value));
}

public ValueDest Translate(ValueSource source, int multiplier)
{
return Mapper.Map<ValueDest>(source, opt => opt.Items.Add("multiplier", multiplier));
}

private class ValueResolver : IValueResolver
{
public ResolutionResult Resolve(ResolutionResult source)
{
return source.New((int)source.Value * (int)source.Context.Options.Items["multiplier"]);
}
}
}

private class ValueSource { public int Value { get; set; } }
private class ValueDest { public int Value { get; set; } }

这是一个使用 TypeConverter 的示例:

[Test]
public void TypeTranslator_ExtraMapParameters()
{
const int multiplier = 3;
TypeTranslator translator = new TypeTranslator();
Mapper.AssertConfigurationIsValid();

TypeSource source = new TypeSource { Value = 10 };
TypeDest dest = translator.Translate(source, multiplier);
Assert.That(dest.Value, Is.EqualTo(30));

source = new TypeSource { Value = 15 };
dest = translator.Translate(source, multiplier);
Assert.That(dest.Value, Is.EqualTo(45));
}

private class TypeTranslator
{
static TypeTranslator()
{
Mapper.CreateMap<TypeSource, TypeDest>()
.ConvertUsing<TypeConverter>();
}

public TypeDest Translate(TypeSource source, int multiplier)
{
return Mapper.Map<TypeDest>(source, opt => opt.Items.Add("multiplier", multiplier));
}

private class TypeConverter : ITypeConverter<TypeSource, TypeDest>
{
public TypeDest Convert(ResolutionContext context)
{
TypeSource source = (TypeSource)context.SourceValue;
int multiplier = (int)context.Options.Items["multiplier"];

return new TypeDest { Value = source.Value * multiplier };
}
}
}

private class TypeSource { public int Value { get; set; } }
private class TypeDest { public int Value { get; set; } }

关于parameter-passing - AutoMapper - 将参数传递给自定义解析器的奇怪行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25896276/

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