63

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

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

以下所需资料:

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 - 但它没有帮助。

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

4

3 回答 3

154

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

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);
于 2013-02-04T22:32:29.653 回答
39

因为很多人是通过问题tilte(非常通用并且与编译器消息匹配)到达这里的,所以让我对编译错误itef给出更详细的答案。

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

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;
}

为了解决这个问题,你可以告诉编译器你的泛型参数有一个无参数的构造器。这是通过定义约束来完成的:

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

于 2015-03-30T11:40:30.480 回答
17

Item行:

Item i = new Item(p, 10);

指的是方法的泛型类型参数ItemCountTestHelper而不是类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);  
}
于 2013-02-04T22:32:43.307 回答