6

我正在使用ValueInjecter将属性从域模型映射到通过服务层提供的 DTO。有问题的服务也接受更新......因此传入更新的 DTO,然后将其注入域对象并保存。

    // Domain
    public class Member 
    {
      public Country Country { get; set; }
    }

    public class Country 
    {
      public string Code { get; set; }
      public string Name { get; set; }
    }

    //Dto
    public class MemberDto 
    {
       public string CountryCode { get; set; }
    }

    //Transformation Method attempt 1
    public Member InjectFromDto (MemberDto dto, Member source)
    {
       source = source.InjectFrom<UnflatLoopValueInjection>(dto);
       return source;
    }

现在上面的代码所做的就是更新 Property Member.Country.Code 这显然不是我需要它做的。

所以从文档中,我想我需要创建一个覆盖并得到这个:

public class CountryLookup: UnflatLoopValueInjection<string, Country>
    {
        protected override Country SetValue(string sourcePropertyValue)
        {
            return countryService.LookupCode(sourcePropertyValue);
        }
    }


 //revised transformation call
 //Transformation Method attempt 2
    public Member InjectFromDto (MemberDto dto, Member source)
    {
       source = source.InjectFrom<UnflatLoopValueInjection>(dto)
                      .InjectFrom<CountryLookup>(dto);
       return source;
    }

我的问题是在调试过程中, CountryLookup 永远不会被调用。

我能想到的可能原因:

  • Nhibernate Proxy 类导致值注入器与 Country 类型不匹配?但这没有意义,因为它在展平期间起作用。
  • 也许由于某种原因,unflattening 没有触发。即 Dto 是 CountryCode 和 Domain 是 Country.Code

我需要使用 Dto 上的 CountryCode 属性来调用 countryService.LookupCode 以返回要在更新注入期间使用的正确对象。

4

3 回答 3

3

unflattening 将是这样做:

entity.Country.Code <- dto.CountryCode

你需要的是:

entity.Country <- dto.CountryCode

因此,您的解决方案是继承一个 ExactValueInjection,您将从 CountryCode 转到 Country。

建议你做的是和我在我的另一个项目http://awesome.codeplex.com的现场演示中所做的一样

我有这样的东西:

    public class Entity
    {
       public int Id{get;set;}
    }
    public class Member : Entity
    {
        public Country Country{get;set;}
    }
    public class MemberDto : DtoWithId
    {
        public int? Country {get;set;}
    }

并使用这些注入从实体到 dto 并返回

    public class NullIntToEntity : LoopValueInjection
        {
            protected override bool TypesMatch(Type sourceType, Type targetType)
            {
                return sourceType == typeof(int?) && targetType.IsSubclassOf(typeof(Entity));
            }

            protected override object SetValue(object sourcePropertyValue)
            {
                if (sourcePropertyValue == null) return null;
                var id = ((int?) sourcePropertyValue).Value;

                dynamic repo = IoC.Resolve(typeof(IRepo<>).MakeGenericType(TargetPropType));

                return repo.Get(id);
            }
        }
//(you also need to have a generic repository, notice IRepo<>)    
    public class EntityToNullInt : LoopValueInjection
        {
            protected override bool TypesMatch(Type sourceType, Type targetType)
            {
                return sourceType.IsSubclassOf(typeof (Entity)) && targetType == typeof (int?); 
            }

            protected override object SetValue(object o)
            {
                if (o == null) return null;
                return (o as Entity).Id;
            }
        }

这些注入不仅可以处理来自 int 的处理吗?到Country和 back 以及任何其他继承Entity的类型

于 2011-01-24T18:20:25.850 回答
3

使用 Omu 的建议/参考,这是问题的具体代码。

 public class CountryLookup : ExactValueInjection
    {
        private ICountryService countryservice;

        public CountryLookup(ICountryService countryService)
        {
           this.countryService = countryService; 
        }

        protected override bool TypesMatch(Type s, Type t)
        {
            return (s == typeof(string)) && (t == typeof (Country));

        }
        protected override Object SetValue(object v)
        {
            if (v == null) 
                return null;

            var country = countryService.LookupCode((string) v);
            return country;
        }

        public override string SourceName()
        {
            return "CountryCode";
        }

        public override string TargetName()
        {
            return "Country";
        }    
    }

public Member InjectFromDto (MemberDto dto, Member source)
{
   source = source.InjectFrom<UnflatLoopValueInjection>(dto)
                  .InjectFrom<CountryLookup>(dto);
   return source;
}
于 2011-01-26T04:07:26.713 回答
0

是调用setter方法的框架吗?在大多数 DI 框架中,标准是 setMethod() 中的小写“s”。只是一个初步的建议。

于 2011-01-24T14:53:53.503 回答