Ищете общий способ реализации функции в базовом классе - PullRequest
3 голосов
/ 19 августа 2010

Я пытаюсь сделать простую реализацию шаблона спецификации в слое моего домена.

Если у меня есть статический класс, полный спецификаций, подобных этому:

public static class FooSpecifications
{
  public static Func<Foo, bool> IsSuperhuman
  {
    get
    {
      return foo => foo.CanShootLasersOutOfItsEyes && foo.CanFly;
    }
  }
}

Тогда яможет делать изумительные вещи, как это:

IEnumerable<Foo> foos = GetAllMyFoos();
var superFoos = foos.Where(FooSpecifications.IsSuperhuman);

Я также могу добавить метод bool в Foo, чтобы определить, соответствует ли конкретный экземпляр спецификации:

public bool Meets(Func<Foo, bool> specification)
{
  return specification.Invoke(this);
}

Учитывая, что Foo, как и все моидоменные сущности, расширяет DomainObject, есть ли способ, которым я могу поместить общую реализацию Meets () в DomainObject, чтобы избавить меня от реализации Meets () отдельно в каждой сущности?

1 Ответ

0 голосов
/ 19 августа 2010

Примерно так ...

    public abstract class DomainObj<T> // T - derived type
        where T : DomainObj<T>
    {
        public bool Meets(Func<T, bool> specification)
        {
            return specification.Invoke((T) this);
        }
    }

    public class Foo : DomainObj<Foo> {}

    public class Bar : DomainObj<Bar> {}       

        Func<Foo, bool> foospec = x => true;
        Func<Bar, bool> barspec = x => true;

        var foo = new Foo();
        var bar = new Bar();
        foo.Meets(foospec);
        foo.Meets(barspec); // won't compile because of mismatched types of spec and object instance

РЕДАКТИРОВАТЬ

Может быть, будет лучше перевести метод Meet в расширение.Это устранит необходимость в параметре типа.

    public abstract class DomainObj
    {
    }

    public static class DomainObjExtensions
    {
        public static bool Meets<T>(this T obj, Func<T, bool> f)
            where T : DomainObj
        {
            return f(obj);
        }
    }

    public class Foo : DomainObj {}

    public class Bar : DomainObj {}

    Func<Foo, bool> foospec = x => true;
    Func<Bar, bool> barspec = x => true;

    var foo = new Foo();
    var bar = new Bar();
    foo.Meets(foospec);
    foo.Meets(barspec); // error
...