Visual Studio 2015 扩展方法调用作为方法组
Visual Studio 2015 extension method call as method group
我有一个像
这样的扩展方法
public static void RemoveDetail<TMaster, TChild>(this TMaster master, TChild child)
where TMaster : class, IMaster<TChild>
where TChild : class, IDetail<TMaster>;
我有两个类
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
}
和
public class Permission : IDetail<Principal>
我从方法 public static void Foreach<T>(this IEnumerable<T> source, Action<T> action);
:
接受的操作调用 RemoveDetail
aPrincipal.Permissions.Foreach(x => aPrincipal.RemoveDetail(x));
ReSharper 建议我用
这样的方法组替换此调用
aPrincipal.Permissions.Foreach(aPrincipal.RemoveDetail);
这在 VS2013 和以前的版本中运行良好,但在 VS2015 中编译失败
'Principal' does not contain a definition for 'RemoveDetail' and no extension method 'RemoveDetail' accepting a first argument of type 'Principal' could be found (are you missing a using directive or an assembly reference?)
有人有什么建议吗?
我是否必须更新所有用法并制作 ReSharper 以避免这种替换?
我不明白它在 VS2013 中是如何工作的。 RemoveDetail
是 TMethod
的一种方法,而不是你的操作 MyClass
。
应该是这样的
z.ForEach(x => master.RemoveDetail(x);
我能够重现你的问题
public interface IDetail<T>
{
}
public interface IMaster<T>
{
}
public class MyClass
{
public void method()
{
Principal aPrincipal = new Principal();
aPrincipal.Permissions.Foreach(x => aPrincipal.RemoveDetail(x)); // No suggestion from resharper
}
}
public class Permission : IDetail<Principal>
{
}
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
}
public static class Class
{
public static void RemoveDetail<TMaster, TChild>(this TMaster master, TChild child)
where TMaster : class, IMaster<TChild>
where TChild : class, IDetail<TMaster>
{
}
public static void Foreach<T>(this IEnumerable<T> source, Action<T> action)
{
}
}
resharper 没有提出任何建议。所以可能你必须将你的 Resharper 更新到更新的版本。可能是一个错误,现在已修复。
如果这与您的做法不一样。那你得把更多的信息。
当我尝试将其转换为方法组时,编译器也给出了同样的错误。所以这个问题可能仍然存在:Why Compiler cant do this?
编辑:
要解决此问题,您必须从 Principal 内部调用 RemoveDetail
方法。所以让你的校长 class 像这样。
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
public void RemoveDetail(Permission p)
{
Class.RemoveDetail(this, p);
}
}
我认为编译器内部存在某种歧义(可能是错误)无法识别 RemoveDetail
方法。编译器如何尝试在 Principal
中找到它。所以你可以通过在 Principal
中创建 RemoveDetail
并从那里调用静态 RemoveDetail
来修复它。
编辑 2:
问题是泛型类型约束。
where TMaster : class, IMaster<TChild>
这使得编译器查看 class,它实现了 IMaster<TChild>
,那是 Principal
。如果您删除此 where 子句,它将解决问题。否则编译器期望它应该是 Principal
.
如果你使用 lambda,你就消除了这种歧义。
aPrincipal.RemoveDetail // compiler expects property/field/method in Principal
// but compiler forgets method group and static generic method!
x => aPrincipal.RemoveDetail(x) // this is no more like property or field
//but a method that is considered static generic method!
所以这是一个 C#6 错误
我于 2015 年 9 月 3 日在此处向 Microsoft 提供了有关此主题的反馈:
https://connect.microsoft.com/VisualStudio/feedback/details/1747835/visual-studio-2015-extension-method-call-as-method-group
今天我收到了 Neal [MSFT] 的反馈:
This is fixed in VS2015 Update 1.
It is tracked here: https://github.com/dotnet/roslyn/issues/4970
我有一个像
这样的扩展方法public static void RemoveDetail<TMaster, TChild>(this TMaster master, TChild child)
where TMaster : class, IMaster<TChild>
where TChild : class, IDetail<TMaster>;
我有两个类
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
}
和
public class Permission : IDetail<Principal>
我从方法 public static void Foreach<T>(this IEnumerable<T> source, Action<T> action);
:
aPrincipal.Permissions.Foreach(x => aPrincipal.RemoveDetail(x));
ReSharper 建议我用
这样的方法组替换此调用aPrincipal.Permissions.Foreach(aPrincipal.RemoveDetail);
这在 VS2013 和以前的版本中运行良好,但在 VS2015 中编译失败
'Principal' does not contain a definition for 'RemoveDetail' and no extension method 'RemoveDetail' accepting a first argument of type 'Principal' could be found (are you missing a using directive or an assembly reference?)
有人有什么建议吗? 我是否必须更新所有用法并制作 ReSharper 以避免这种替换?
我不明白它在 VS2013 中是如何工作的。 RemoveDetail
是 TMethod
的一种方法,而不是你的操作 MyClass
。
应该是这样的
z.ForEach(x => master.RemoveDetail(x);
我能够重现你的问题
public interface IDetail<T>
{
}
public interface IMaster<T>
{
}
public class MyClass
{
public void method()
{
Principal aPrincipal = new Principal();
aPrincipal.Permissions.Foreach(x => aPrincipal.RemoveDetail(x)); // No suggestion from resharper
}
}
public class Permission : IDetail<Principal>
{
}
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
}
public static class Class
{
public static void RemoveDetail<TMaster, TChild>(this TMaster master, TChild child)
where TMaster : class, IMaster<TChild>
where TChild : class, IDetail<TMaster>
{
}
public static void Foreach<T>(this IEnumerable<T> source, Action<T> action)
{
}
}
resharper 没有提出任何建议。所以可能你必须将你的 Resharper 更新到更新的版本。可能是一个错误,现在已修复。
如果这与您的做法不一样。那你得把更多的信息。
当我尝试将其转换为方法组时,编译器也给出了同样的错误。所以这个问题可能仍然存在:Why Compiler cant do this?
编辑:
要解决此问题,您必须从 Principal 内部调用 RemoveDetail
方法。所以让你的校长 class 像这样。
public class Principal : IMaster<Permission>
{
public virtual IEnumerable<Permission> Permissions { get; }
public void RemoveDetail(Permission p)
{
Class.RemoveDetail(this, p);
}
}
我认为编译器内部存在某种歧义(可能是错误)无法识别 RemoveDetail
方法。编译器如何尝试在 Principal
中找到它。所以你可以通过在 Principal
中创建 RemoveDetail
并从那里调用静态 RemoveDetail
来修复它。
编辑 2:
问题是泛型类型约束。
where TMaster : class, IMaster<TChild>
这使得编译器查看 class,它实现了 IMaster<TChild>
,那是 Principal
。如果您删除此 where 子句,它将解决问题。否则编译器期望它应该是 Principal
.
如果你使用 lambda,你就消除了这种歧义。
aPrincipal.RemoveDetail // compiler expects property/field/method in Principal
// but compiler forgets method group and static generic method!
x => aPrincipal.RemoveDetail(x) // this is no more like property or field
//but a method that is considered static generic method!
所以这是一个 C#6 错误
我于 2015 年 9 月 3 日在此处向 Microsoft 提供了有关此主题的反馈:
https://connect.microsoft.com/VisualStudio/feedback/details/1747835/visual-studio-2015-extension-method-call-as-method-group
今天我收到了 Neal [MSFT] 的反馈:
This is fixed in VS2015 Update 1.
It is tracked here: https://github.com/dotnet/roslyn/issues/4970