21

我正在开发一个应用程序,我需要调用泛型类的方法并且我不关心实例的实际类型。类似于以下 Java 代码:

public class Item<T>{
  private T item;

  public doSomething(){...}
}

...
public void processItems(Item<?>[] items){
 for(Item<?> item : items)
   item.doSomething();
}

当时我很着急,所以我通过定义一个带有我需要调用的方法的接口并让泛型类实现它来解决我的问题。

public interface IItem  
{
   void doSomething();
}

public class Item<T> : IItem {
  private T item;

  public void doSomething(){...}
}

...
public void processItems(IItem[] items)
{
 foreach(IItem item in items)
   item.doSomething();
}

这种解决方法效果很好,但我想知道实现相同行为的正确方法是什么。

编辑:

我忘了提到调用者processItems不知道实际类型。实际上这个想法是作为参数传递给的数组processItems可以包含混合类型。由于在 .Net 中不可能有这样的数组,因此使用非泛型基类或接口似乎是唯一的方法。

4

5 回答 5

25

执行此操作的正常方法是使方法通用:

public void ProcessItems<T>(Item<T>[] items) {
  foreach(Item<T> item in items)
    item.DoSomething();
}

假设调用者知道类型,类型推断应该意味着他们不必显式指定它。例如:

Item<int> items = new Item<int>(); // And then populate...
processor.ProcessItems(items);

话虽如此,创建一个指定与类型无关的操作的非泛型接口也很有用。这在很大程度上取决于您的确切用例。

于 2009-02-09T11:24:34.983 回答
3

我看到你只想调用一些没有参数的方法......已经有一个合同:Action

public void processItems(IEnumerable<Action> actions)
{
  foreach(Action t in actions)
    t();
}

客户:

List<Animal> zoo = GetZoo();
List<Action> thingsToDo = new List<Action>();
//
thingsToDo.AddRange(zoo
  .OfType<Elephant>()
  .Select<Elephant, Action>(e => e.Trumpet));
thingsToDo.AddRange(zoo
  .OfType<Lion>()
  .Select<Lion, Action>(l => l.Roar));
thingsToDo.AddRange(zoo
  .OfType<Monkey>()
  .Select<Monkey, Action>(m => m.ThrowPoo));
//
processItems(thingsToDo);
于 2009-02-09T14:53:11.473 回答
1

在 .NET 通用实现中,您无法省略类型参数;这是设计使然。事实上,这只能在 Java 中实现,因为它是基于类型擦除的实现。

您只能使用基本的非通用接口(think IEnumerable<T>and IEnumerable)。

于 2009-02-09T11:22:45.020 回答
0

继乔恩的帖子。使方法通用(模板)否定了对这种功能的要求(使用 <?>)。您始终可以将类型提供给泛型类/函数,并且对于您不知道需要什么类型的情况,您也可以将有问题的方法/类设为泛型......最终用户必须在调用时提供类型这样的函数或使用泛型类,以使代码能够编译......否则你会得到一些编译器错误。

于 2009-02-09T11:34:33.150 回答
0

在从 Java 移植东西时,我一直在为同样的问题苦苦挣扎,因为我有类似的结构

if (o instanceof Collection<?>) doSoemthing((Collection<?>)o);

幸运的是,泛型 ICollection 也是非泛型 ICollection,如果有人需要将其中的元素视为纯对象,它仍然是可能的:

if (o is ICollection) DoSomething((ICollection)o);

这样,由于我们不关心集合中元素的实际类型,所以我们在这里得到的只是对象。此处需要注意:如果集合包含原始类型(例如 int 或 byte),则自动装箱可能会引入性能损失。

于 2009-12-30T16:57:35.850 回答