3

晚上好,

我正在构建一种缓存,比较两个数据表(最后一个读取和我需要写入的那个)并仅检索修改过的行。我使用 DataRowComparer.Default 作为平等比较器,但它的目标是错误的比较行与许多字段。

它与具有短文本值的 3 列表格完美配合,但比较长文本描述时,即使我更改单个字符,它也无法返回整个表格。

代码非常简单:

var diffDs = ds.Tables[0].AsEnumerable().Except(cachedTable.AsEnumerable(), DataRowComparer.Default);

想法?谢谢!

更新: 手动调试我能够比较 ds.Tables[0].AsEnumerable() 行与 cachedtable.AsEnumerable() 行:绝对等于但 DataRowComparer 除外。我删除了任何 DateTime 列,试图避免格式差异而没有成功。交叉点也不行。

更新 2: 除了不适用于空/空字段。对于 IEqualityComparer,它们似乎有所不同。

4

1 回答 1

0

我找到了一个解决方案:

DataRowComparer 代码的工作方式如下:

internal static bool AreEqual(object a, object b)
    {
        if (Object.ReferenceEquals(a, b))
        {   // same reference or (null, null) or (DBNull.Value, DBNull.Value)
            return true;
        }
        if (Object.ReferenceEquals(a, null) || Object.ReferenceEquals(a, DBNull.Value) ||
            Object.ReferenceEquals(b, null) || Object.ReferenceEquals(b, DBNull.Value))
        {   // (null, non-null) or (null, DBNull.Value) or vice versa
            return false;
        }
        return (a.Equals(b) || (a.GetType().IsArray && CompareArray((Array)a, b as Array)));
    }

如您所见,空值和 DBNull 的考虑方式不同,并且 null==DBNull 始终为假。显然,一个空字符串看起来像空字段,但它不是 DBNull。因此,由于这个空值,两个明显相似的表是不同的。我的解决方案在从 xml 源构建时解析 ds.Tables[0](这是具有空字符串和空值的解决方案)并使用 System.Convert.DBNull 进行替换

现在比较完美。

如果您从 sql 查询或从 xml 源构建数据表,则 PS \n 和 \r\n 也不同。易于解决假设上面的 null/DBNull 解决方案。

于 2012-08-09T08:03:30.573 回答