使用包含协变类型的项目的访问器在单元测试项目中构建失败

时间:2010-09-08 09:16:16

标签: c# unit-testing covariance accessor

我在项目中添加了一个协变接口:

interface IView
{
}

interface IPresenter<out TView> where TView : IView
{
    TView View { get; }
}

我创建了一些类,实现了这些接口:

class TestView : IView
{
}

class TestPresenter : IPresenter<TestView>
{
  public TestView View
  {
    get { return something; }
  }

  private void DoSomething()
  {
  }
}

我可以毫无问题地使用它:

IPresenter<IView> presenter = new TestPresenter();

所以一切似乎都正确,所以我认为我的协方差用法是正确的。不幸的是,我们的单元测试项目包含位于同一项目中的某些类型的私有访问器,如协变接口,这会导致构建失败。

  

无法加载类型   'GenericInheritanceTest.IPresenter_Impl`1'   从装配   “GenericInheritanceTest_Accessor,   版本= 0.0.0.0,文化=中立,   PublicKeyToken = null'因为它   声明一个协变或逆变   type参数,不是接口   或代表。

这究竟是什么问题?我的实施是否有失败,分别是。怎么解决这个问题?不能,我们一旦使用协变类型就必须避免访问者?是否可以阻止为某些类型创建访问器来解决此问题?

3 个答案:

答案 0 :(得分:14)

这是Visual Studio 2010中的一个错误。它已向Microsoft Connect报告,但已关闭,显然无法修复。

根据Bruce Taimana的博客文章,the private accessor feature的开发已停止,可能会在Visual Studio的未来版本中删除。列出的可能替代方案是:

  
      
  1. 使用Microsoft.VisualStudio.TestTools.UnitTesting.PrivateObject类来协助访问代码中的内部和私有API。这可以在Microsoft.VisualStudio.QualityTools.UnitTestFramework.dll程序集中找到。

  2.   
  3. 创建一个反射框架,能够反映您的代码以访问内部或私有API。

  4.   
  5. 如果您尝试访问的代码是内部代码,则可以使用InternalsVisibleToAttribute访问API,以便您的测试代码可以访问内部API。

  6.   

答案 1 :(得分:1)

我正在努力解决这个错误,但也遇到了错误:“BuildShadowTask”意外失败。在我尝试解决它时,摆脱错误的唯一方法是从泛型中删除out关键字,即协变接口。

实际上,当Resharper告诉我类型参数可能是协变的时候我遇到了这个错误:

public class VerticalSeekBar extends SeekBar {

    public VerticalSeekBar(Context context){
        super(context);
    }

    public VerticalSeekBar(Context context, AttributeSet attrs, int defStyle){
        super(context, attrs, defStyle);
    }

    public VerticalSeekBar(Context context, AttributeSet attrs){
        super(context, attrs);
    }

    public void onSizeChanged(int w, int h, int oldw, int oldh){
        super.onSizeChanged(h, w, oldh, oldw);
    }

    @Override
    protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec){
        super.onMeasure(heightMeasureSpec, widthMeasureSpec);
        setMeasuredDimension(getMeasuredHeight(), getMeasuredWidth());
    }

    protected void onDraw(Canvas c){
        c.rotate(90);
        c.translate(0, - getWidth());
        super.onDraw(c);
    }

    private OnSeekBarChangeListener onSeekBarChangeListener = new OnSeekBarChangeListener() {
        @Override
        public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser){

        }

        @Override
        public void onStartTrackingTouch(SeekBar seekBar){

        }

        @Override
        public void onStopTrackingTouch(SeekBar seekBar){

        }
    };

    @Override
    public void setOnSeekBarChangeListener(OnSeekBarChangeListener onSeekBarChangeListener){
        this.onSeekBarChangeListener = onSeekBarChangeListener;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (!isEnabled()) {
            return false;
        }

        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                onSeekBarChangeListener.onStartTrackingTouch(this);
                setPressed(true);
                setSelected(true);
                applyProgress(event);
                break;
            case MotionEvent.ACTION_MOVE:
                super.onTouchEvent(event);
                setPressed(true);
                setSelected(true);
                applyProgress(event);
                break;
            case MotionEvent.ACTION_UP:
                onSeekBarChangeListener.onStopTrackingTouch(this);
                setPressed(false);
                setSelected(false);
                applyProgress(event);
                break;
            case MotionEvent.ACTION_CANCEL:
                super.onTouchEvent(event);
                setPressed(false);
                setSelected(false);
                break;
        }
        return true;
    }

    private void applyProgress(MotionEvent event){
        int progress = (int) (getMax() * event.getY() / getHeight()) - 0;
        setProgress((int) (getMax() * event.getY() / getHeight()) - 0);
        onSizeChanged(getWidth(), getHeight(), 0, 0);
    }
}

更改为:

private delegate TResult Action<TResult>(); 

可悲的是,我不得不再次更改它并在评论中禁用Resharper警告:

private delegate TResult Action<out TResult>();

因此,一种策略可以是搜索:

// ReSharper disable once TypeParameterCanBeVariant 
private delegate TResult Action<TResult>();
在项目中

并删除out关键字,只是为了让它编译。 微软也不是很优雅,但也不是很优雅,因为 这是五年前通过Microsoft Connect和他们支持的 我们选择了解决这个问题。问题出在单元测试项目中。它没有帮助 从Visual Studio单元测试框架更改为NUnit测试框架。

答案 2 :(得分:0)

如果您使用Unity Interception,并且您的参数标记为out,则Unity Interception将导致此错误。原因是拦截必须能够读取参数列表。因此,与上述情况类似,如果Resharper警告参数可以是协变的,则需要忽略此警告。