Dav*_*den 7 c# linq linq-to-objects
鉴于以下课程:
public class WeekOfYear : IEquatable<WeekOfYear>, IComparable<WeekOfYear>
{
private readonly DateTime dateTime;
private readonly DayOfWeek firstDayOfWeek;
public WeekOfYear(DateTime dateTime)
: this(dateTime, DayOfWeek.Sunday)
{
}
public WeekOfYear(DateTime dateTime, DayOfWeek firstDayOfWeek)
{
this.dateTime = dateTime;
this.firstDayOfWeek = firstDayOfWeek;
}
public int Year
{
get
{
return dateTime.Year;
}
}
public int Week
{
get
{
return CultureInfo.CurrentCulture.Calendar.GetWeekOfYear(dateTime, CalendarWeekRule.FirstDay, firstDayOfWeek);
}
}
public bool Equals(WeekOfYear other)
{
return Year == other.Year && Week == other.Week;
}
public int CompareTo(WeekOfYear other)
{
if (Year > other.Year || Year == other.Year && Week > other.Week)
{
return 1;
}
if (Equals(other))
{
return 0;
}
return -1;
}
public override string ToString()
{
return String.Format("Week of {0}", dateTime.FirstDayOfWeek(firstDayOfWeek).ToString("MMMM dd, yyyy"));
}
}
public class WeekOfYearComparer : IEqualityComparer<WeekOfYear>, IComparer<WeekOfYear>
{
public bool Equals(WeekOfYear x, WeekOfYear y)
{
return x.Equals(y);
}
public int GetHashCode(WeekOfYear weekOfYear)
{
return weekOfYear.GetHashCode();
}
public int Compare(WeekOfYear x, WeekOfYear y)
{
return x.CompareTo(y);
}
}
Run Code Online (Sandbox Code Playgroud)
此测试失败(意外):
[Test]
public void Fails()
{
var dates = new List<DateTime>
{
new DateTime(2012, 1, 1),
new DateTime(2012, 2, 1),
new DateTime(2012, 1, 1)
};
IEnumerable<IGrouping<WeekOfYear, DateTime>> groups = dates.GroupBy(date => new WeekOfYear(date), new WeekOfYearComparer());
Assert.That(groups.Count(), Is.EqualTo(2)); // count is 3
}
Run Code Online (Sandbox Code Playgroud)
这个测试通过(预期):
[Test]
public void Works()
{
var dates = new List<DateTime>
{
new DateTime(2012, 1, 1),
new DateTime(2012, 2, 1),
new DateTime(2012, 1, 1)
};
var groups = dates.GroupBy(
date =>
{
var weekOfYear = new WeekOfYear(date);
return new { weekOfYear.Year, weekOfYear.Week };
});
Assert.That(groups.Count(), Is.EqualTo(2));
}
Run Code Online (Sandbox Code Playgroud)
为什么第一次测试结果为3?
Mar*_*ell 10
相等性检查的第一部分是通过哈希码完成的; 您必须提供有效的哈希代码实现(为什么,请参阅为什么在重写Equals方法时重写GetHashCode很重要?).您的比较器可以执行此操作,但它遵循对象:
public int GetHashCode(WeekOfYear weekOfYear)
{
return weekOfYear.GetHashCode();
}
Run Code Online (Sandbox Code Playgroud)
并且该对象不提供有效的哈希码.内部合适的实现WeekOfYear类似于:
public bool Equals(WeekOfYear other)
{
return other != null && Year == other.Year && Week == other.Week;
}
public override bool Equals(object obj)
{
return Equals(obj as WeekOfYear);
}
public override int GetHashCode()
{ // exploit number of weeks in year
return (Year.GetHashCode()*52) + Week.GetHashCode();
}
Run Code Online (Sandbox Code Playgroud)
注意到我也提供了override平等.
实际上,由于您的对象提供了所有代码,因此自定义比较器没有任何好处; 你可以WeekOfYearComparer完全删除,因为默认行为是在底层类型上寻找合适的相等/比较操作:
var groups = dates.GroupBy(date => new WeekOfYear(date));
Run Code Online (Sandbox Code Playgroud)