我在项目中添加了一个协变接口:
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参数,不是接口 或代表。
这究竟是什么问题?我的实施是否有失败,分别是。怎么解决这个问题?不能,我们一旦使用协变类型就必须避免访问者?是否可以阻止为某些类型创建访问器来解决此问题?
答案 0 :(得分:14)
这是Visual Studio 2010中的一个错误。它已向Microsoft Connect报告,但已关闭,显然无法修复。
根据Bruce Taimana的博客文章,the private accessor feature的开发已停止,可能会在Visual Studio的未来版本中删除。列出的可能替代方案是:
使用
Microsoft.VisualStudio.TestTools.UnitTesting.PrivateObject
类来协助访问代码中的内部和私有API。这可以在Microsoft.VisualStudio.QualityTools.UnitTestFramework.dll
程序集中找到。创建一个反射框架,能够反映您的代码以访问内部或私有API。
- 醇>
如果您尝试访问的代码是内部代码,则可以使用
InternalsVisibleToAttribute
访问API,以便您的测试代码可以访问内部API。
答案 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警告参数可以是协变的,则需要忽略此警告。