- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
从Artech's blog看然后我们在评论中进行了讨论。由于那个博客只有中文,所以我在这里做一个简单的解释。重现代码:
[AttributeUsage(AttributeTargets.Class, Inherited = true, AllowMultiple = true)]
public abstract class BaseAttribute : Attribute
{
public string Name { get; set; }
}
public class FooAttribute : BaseAttribute { }
[Foo(Name = "A")]
[Foo(Name = "B")]
[Foo(Name = "C")]
public class Bar { }
//Main method
var attributes = typeof(Bar).GetCustomAttributes(true).OfType<FooAttribute>().ToList<FooAttribute>();
var getC = attributes.First(item => item.Name == "C");
attributes.Remove(getC);
attributes.ForEach(a => Console.WriteLine(a.Name));
该代码获取所有的 FooAttribute
并删除名称为“C”的那个。明明输出的是“A”和“B”?如果一切顺利,你就不会看到这个问题。事实上,理论上你会得到“AC”“BC”甚至更正“AB”(我的机器上有 AC,博客作者有 BC)。问题是由 System.Attribute 中 GetHashCode/Equals 的实现引起的。实现的一个片段:
[SecuritySafeCritical]
public override int GetHashCode()
{
Type type = base.GetType();
//*****NOTICE*****
FieldInfo[] fields = type.GetFields(BindingFlags.NonPublic
| BindingFlags.Public
| BindingFlags.Instance);
object obj2 = null;
for (int i = 0; i < fields.Length; i++)
{
object obj3 = ((RtFieldInfo) fields[i]).InternalGetValue(this, false, false);
if ((obj3 != null) && !obj3.GetType().IsArray)
{
obj2 = obj3;
}
if (obj2 != null)
{
break;
}
}
if (obj2 != null)
{
return obj2.GetHashCode();
}
return type.GetHashCode();
}
它使用 Type.GetFields
所以从基类继承的属性被忽略,因此 FooAttribute
的三个实例是等价的(然后是 Remove
方法随机)。那么问题来了:执行上有什么特殊原因吗?或者这只是一个错误?
最佳答案
一个明显的错误,不。一个好主意,也许是也可能不是。
一件事等于另一件事是什么意思?如果我们真的想的话,我们可以变得非常哲学。
只是有点哲学,有几件事必须坚持:
x.Equals(x)
必须持有。x.Equals(y)
然后y.Equals(x)
如果!x.Equals(y)
然后!y.Equals(x)
.x.Equals(y)
和 y.Equals(z)
然后x.Equals(z)
.还有一些其他的,虽然只有这些可以直接反射(reflect)在 Equals()
的代码中。一个人。
如果实现重写 object.Equals(object)
, IEquatable<T>.Equals(T)
, IEqualityComparer.Equals(object, object)
, IEqualityComparer<T>.Equals(T, T)
, ==
或 !=
不满足以上,就是明显的bug。
.NET 中反射(reflect)平等的其他方法是 object.GetHashCode()
, IEqualityComparer.GetHashCode(object)
和 IEqualityComparer<T>.GetHashCode(T)
.这里有一个简单的规则:
如果a.Equals(b)
那么它必须持有 a.GetHashCode() == b.GetHashCode()
.等价于 IEqualityComparer
和 IEqualityComparer<T>
.
如果那不成立,那么我们又遇到了一个错误。
除此之外,没有关于平等必须意味着什么的全面规则。取决于自身提供的类的语义Equals()
覆盖或由相等比较器强加给它的那些。当然,这些语义应该非常明显,或者记录在类或相等比较器中。
总而言之,Equals
是如何实现的?和/或 GetHashCode
有一个错误:
GetHashCode
之间的关系和 Equals
和上面的不一样。覆盖了 Attribute
, equals 确实具有自反、对称和传递属性,它是 GetHashCode
确实匹配它,它的文档是 Equals
覆盖是:
This API supports the .NET Framework infrastructure and is not intended to be used directly from your code.
你真的不能说你的例子反驳了这一点!
由于您提示的代码在这些方面都没有失败,所以它不是错误。
这段代码中有一个错误:
var attributes = typeof(Bar).GetCustomAttributes(true).OfType<FooAttribute>().ToList<FooAttribute>();
var getC = attributes.First(item => item.Name == "C");
attributes.Remove(getC);
你首先要求一个满足条件的项目,然后要求删除一个等于它的项目。没有理由不检查所讨论类型的相等性语义就期望 getC
将被删除。
你应该做的是:
bool calledAlready;
attributes.RemoveAll(item => {
if(!calledAlready && item.Name == "C")
{
return calledAlready = true;
}
});
也就是说,我们使用一个谓词来匹配第一个属性为Name == "C"
的谓词。没有别的。
关于c# - GetHashCode 和 Equals 在 System.Attribute 中实现不正确?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8839445/
我有一个带有帮助页面的 Web API 2 项目,该项目在本地运行良好,但当我将其推送到 Azure 时抛出此错误: Method not found: 'System.String System.S
我有两台服务器,但通常运行相同的设置 - IIS、SQL Server 等。一台给我这个错误,另一台没有。我从 Visual Studio 向两者发布相同的代码。 它们都在运行 .NET CLR Ve
System.out声明为 public static final PrintStream out。 但是你可以调用System.setOut()重新分配它。 嗯?如果它是 final,这怎么可能?
System.out被声明为 public static final PrintStream out。 但是您可以调用System.setOut()重新分配它。 嗯?如果是 final,这怎么可能?
我有这个 linq 查询: private void GetReceivedInvoiceTasks(User user, List tasks) { var areaIds = user.A
我有一个 MonoTouch 应用程序,当我为设备编译它时,出现以下错误: Error MT2002: Can not resolve reference: System.Boolean System
您好,我有一个名为 DailyVisitReport 的 View 。在该 View 中,我有两个名为 FromDate 和 toDate 的字段。如果我选择 FromDate 和 ToDate 取决
是否可以从 ObjectContext 对象中读取元组列表? 我在存储过程中有类似这样的数据库查询 SELECT T.Id as Item1, -- this is guid T.Wo
我正在尝试创建 Odata 端点,但每当我尝试执行任何涉及日期的查询时都会收到此错误。 我在下面的非常简单示例中重新创建了它。 数据库表 EDMX(片段)
我正在尝试创建 Odata 端点,但每当我尝试执行任何涉及日期的查询时都会收到此错误。 我在下面的非常简单示例中重新创建了它。 数据库表 EDMX(片段)
我有一个方法可以从数据读取器的数据中生成类类型列表。 if (datareader != null && datareader .HasRows) { Dictionary pDict= GetP
我有一些旧的 C++ 代码,它们使用 stdio 进行输入和输出。该代码还通过 fork 生成新进程。它将 stdio 重新映射到每个新进程,以便每个 session 获取其各自的数据。 我正在考虑使
我的应用程序可以很好地构建/链接/部署到模拟器,但我只是第一次尝试将应用程序构建/部署到真实设备,并且链接器失败。 我不使用 System.Console或 ConsoleColor在我的应用程序的任
主要是我很好奇。 我们有一个名为 Unit 的对象在我们的代码库中 - 代表桥梁或道路的组件。在我们的例子中,看到带有 Unit 的 ReactiveUI 命令可能会模棱两可。作为声明中的泛型之一。
我试图将Object变量转换为StreamWriter。但是,它不起作用。有什么错? StreamWriter file = (StreamWriter) myObject; 最佳答案 myObjec
为什么以下不编译? using System; using System.Linq; using System.Linq.Expressions; public static class Extens
我正在使用 Visual Studio Community 2015 开发面向 .NET 4.5 的 Visual Basic 应用程序.我没有编写应用程序,所以我使用 NuGet 添加了所有缺失的依
我刚刚开始使用 powershell,我正在制作一个非常简单的加密功能。我想获取字符串中的每个字符,将其转换为 int 并添加一个选定的数字,然后将其转换回一个字符。 这工作正常: function
一些使用我的应用程序的人似乎变得越来越 System.MissingMethodException: Method not found: 'System.Object System.Windows.T
我是 C# 和实体的新手 我想知道是否有人在这里帮助我。我选择了哪个返回我的 customerid,所以我想将它作为参数传递给我的构造函数,我的构造函数参数类型是 guid 但我的选择类型不同,我不知
我是一名优秀的程序员,十分优秀!