5

我已经解决了几个小时的问题,我想我已经接近了。我正在开发一个应用程序,我们可以有 50-100 种以相同方式执行的类型。因此,我没有创建 50-100 个类,而是尝试使其通用,这就是我所拥有的:

这是基类:

public class RavenWriterBase<T> : IRavenWriter<T> where T : class, IDataEntity

这是界面:

public interface IRavenWriter<T>
{
    int ExecutionIntervalInSeconds { get; }
    void Execute(object stateInfo);
    void Initialize(int executionIntervalInSeconds, Expression<Func<T, DateTime>> timeOrderByFunc);
}

这就是我使用它的方式:

private static void StartWriters()
{
    Assembly assembly = typeof(IDataEntity).Assembly;
    List<IDataEntity> dataEntities = ReflectionUtility.GetObjectsForAnInterface<IDataEntity>(assembly);

    foreach (IDataEntity dataEntity in dataEntities)
    {
        Type dataEntityType = dataEntity.GetType();
        Type ravenWriterType = typeof(RavenWriterBase<>).MakeGenericType(dataEntityType);

        Expression<Func<IDataEntity, DateTime>> func = x => x.CicReadTime;

        // This is where I'm stuck. How do I activate this as RavenWriterBase<T>?
        var ravenWriter = Activator.CreateInstance(ravenWriterType);

        //ravenWriter.Initialize(60, func);  // I can't do this until I cast.

        // More functionality here (not part of this issue)
    }
}

我从上面被困在这条线上:

var ravenWriter = Activator.CreateInstance(ravenWriterType);

这是我的问题:
如何将它用作 RavenWriterBase 或 IRavenWriter?就像是:

ravenWriter.Initialize(60, func);

我认为它需要是这样的,但我需要为 IRavenWriter<> 指定一个类型,但我还不知道:

var ravenWriter = Activator.CreateInstance(ravenWriterType) as IRavenWriter<>;

如果我将鼠标悬停在 ravenWriter 上,我成功地拥有了我的对象:

在此处输入图像描述

但现在我需要能够以通用方式使用它。我怎样才能做到这一点?

更新:

我只是想使用 dynamic 关键字,这很有效:

dynamic ravenWriter = Activator.CreateInstance(ravenWriterType);
ravenWriter.Initialize(60);

我有点作弊,因为我意识到每个 IDataEntity 的 Func 都是相同的,因此没有必要将其作为参数传递给 Initialize()。但是,至少现在我可以调用 Initialize()。但是现在 Func 是相同的,我也不应该需要通用接口。

4

2 回答 2

2

我的解决方案是:

  • 创建一个非通用接口IRavenWriter
  • 使IRavenWriter<T>继承自IRavenWriter
  • 保持ExecuteExecutionIntervalInSeconds_IRavenWriter
  • 在你的作家中IRavenWriter拥有Func<DateTime>并使用它
  • 移至Initialize_IRavenWriter<T>
  • 使用工厂根据类型和表达式初始化 Func:

例如:

public class MyDateTime
{
    public DateTime This { get; set; }
}

public static Func<DateTime> GetFunk<T>(Expression<Func<T, DateTime>> timeOrderByFunc, T t)
{
    return () => timeOrderByFunc.Compile()(t);
}

你使用:

GetFunk<MyDateTime>(x => x.This, new MyDateTime(){This = DateTime.Now});
于 2012-04-13T15:54:20.823 回答
1

Type将运行时转换为编译时通用类型参数并不难。只需引入用于创建/初始化对象的新接口:


interface IRawenWriterFactory
{
  object Create();
}

class RawenWriterFactory<T> : IRawenWriterFactory
{
  public object Create()
  {
    Expression<Func<IDataEntity, DateTime>> func = x => x.CicReadTime;

    var ravenWriter = new RavenWriterBase<T>();
    ravenWriter.Initialize(60, func);

    return ravenWriter;
  }
}

现在只需RawenWriterFactory像您创建的那样使用 dataEntityType 创建ravenWriter并通过非通用 IRavenWriterFactory 接口使用它。

但是,如果您要更改设计,可能会有更简单的解决方案。例如,如果您将 Initialize 方法转换为构造函数,您将能够将 func 作为 Activator.CreateInstance 参数传递,并且您根本不需要使用通用接口进行初始化。

于 2012-04-16T09:03:30.190 回答