无法创建变量类型“Item”的实例,因为它没有new()约束

时间:2013-02-04 22:27:21

标签: c# templates generics compiler-errors

我正在尝试测试方法 - 并收到错误:

Cannot create an instance of the variable type 'Item' because it does not have the new() constraint

以下所需信息:

public interface IHasRect
{
 Rectangle Rectangle { get; }
}

助手类:

class Item : IHasRect
{
  public Item(Point p, int size)
  {
     m_size = size;
     m_rectangle = new Rectangle(p.X, p.Y, m_size, m_size); 
  }
}

对于要测试的函数,我需要实例化一个对象......

public class SomeClass<T> where T : IHasRect

测试:

public void CountTestHelper<Item>()
  where Item : IHasRect
  {
    Rectangle rectangle = new Rectangle(0, 0, 100, 100); 
    SomeClass<Item> target = new SomeClass<Item>(rectangle);            
    Point p = new Point(10,10);
    Item i = new Item(p, 10);      // error here        
    ...
  }
[TestMethod()]
public void CountTest()
{
  CountTestHelper<Item>();
}   

我试图通过阅读http://msdn.microsoft.com/en-us/library/d5x73970.aspxhttp://msdn.microsoft.com/en-us/library/x3y47hd4.aspx来了解此错误的含义或解决方法,但它无济于事。

我不明白这个错误 - 我已经限制了&#34; SomeClass&#34;是类型的。我不能约束整个Test类(由Visual Studio生成的单元测试类,其中包含所有测试) - 否则我将得到许多其他错误。 Item类没有任何模板......

请帮我修复此错误。谢谢。

3 个答案:

答案 0 :(得分:106)

除非使用new关键字将其标记为实现默认构造函数,否则无法初始化Generic类型对象:

public void CountTestHelper<Item>() where Item : IHasRect, new()
 {
    Rectangle rectangle = new Rectangle(0, 0, 100, 100); 
    SomeClass<Item> target = new SomeClass<Item>(rectangle);            
    Point p = new Point(10,10);
    Item i = new Item();    // constructor has to be parameterless!
    ...
 }

另一方面,如果您尝试在应用程序中的其他位置定义初始化Item类型对象,请尝试使用命名空间:

MyAppNamespace.Item i = new MyAppNamespace.Item(p, 10);

答案 1 :(得分:27)

因为很多人通过问题来到这里(这是非常通用的并且与编译器消息相匹配)让我给出关于编译错误的更详细的答案。

您在方法中使用泛型。编译器不知道它将接收哪种类型,因此不保证您的类型具有无参数构造器。例如:

class A {
    A(int i){ ... }
}

class B { ... }

public void MyMethod<T>(){
    T t = new T(); //This would be fine if you use 'MyMethod<B>' but you would have a problem calling 'MyMethod<A>' (because A doesn´t have a parameterless construtor;
}

要解决此问题,您可以告诉编译器您的通用参数有一个无参数的construtor。这是通过定义约束来完成的:

public void MyMethod<T>()  where T: new(){
    T t = new T(); //Now it's ok because compiler will ensure that you only call generic method using a type with parameterless construtor;
}

有关构造函数约束的更多信息,请访问: https://msdn.microsoft.com/en-us/library/bb384067.aspx

答案 2 :(得分:12)

行中的Item

Item i = new Item(p, 10);

指的是Item方法的泛型类型参数CountTestHelper,而不是类Item。更改通用参数名称,例如

public void CountTestHelper<TItem>() where TItem : IHasRect
{
    Rectangle rectangle = new Rectangle(0, 0, 100, 100); 
    SomeClass<TItem> target = new SomeClass<TItem>(rectangle);            
    Point p = new Point(10,10);
    Item i = new Item(p, 10);    
    ...
}

或者,您可以完全限定要创建的Item类的名称:

public void CountTestHelper<Item>() where Item : IHasRect
{
    Rectangle rectangle = new Rectangle(0, 0, 100, 100); 
    SomeClass<Item> target = new SomeClass<Item>(rectangle);            
    Point p = new Point(10,10);
    SomeNamespace.Item i = new SomeNamespace.Item(p, 10);  
}