2

由于我在如何使用“$update_mask”字段时遇到了很大的困难(出于性能原因,我不想调用 fn_cdc_is_bit_set),这里是 C# 代码:

  • updateMask是 CDC“fn_cdc_get_all_changes_[table]”查询中的“__$update_mask”字段的内容
  • colOrdinal是要从中获取更改标志的列的序号(使用“n_cdc_get_column_ordinal”检索此值)

结果是“ hasChanged ”标志。如果字段已在更新中更改,则设置为 true。

请注意,这适用于 SQL Server 2008 和 2012,但可能不适用于未来的版本。

byte[] updateMask = this.UpdateMask;
        unchecked
        {
            byte relevantByte = updateMask[(updateMask.Length - 1) - ((colOrdinal - 1) / 8)];
            int bitMask = 1 << ((colOrdinal - 1) % 8);
            var hasChanged = (relevantByte & bitMask) != 0;

            return hasChanged;
        }
4

3 回答 3

1

只需使用BitArray来获取更改的列的序号索引。反转掩码并将其添加到BitArray,所有数组值设置为true相应索引处的更改列。

byte[] updateMask = this.UpdateMask;  
Array.Reverse(updateMask);
BitArray columnBits = new BitArray(bitmask);

// To match your code sample, you can just check if the column bit is set at the ordinal.
bool hasChanged = columnBits[colOrdinal];

// You could loop and get the column indexes like this as well.
for (int i = 0; i < columnBits.Length; i++)
{
  // If the bit is set to true then the column at the same index was modified.
  if (columnBits[i])
  {
    // Do something with modified column at index 'i'.
  }      
}

我试图使代码示例尽可能简单易读。请记住首先检查数组索引是否存在等。

于 2013-11-07T16:48:07.607 回答
0

我对此不是 100% 确定的,但最有可能的值$update_mask是在COLUMNS_UPDATED()触发器内返回的值。您可以在此处获取有关如何解释该值的详细信息:http: //msdn.microsoft.com/en-us/library/ms186329.aspx

于 2013-01-30T15:36:07.247 回答
0

我有同样的问题。我有多次调用函数 sys.fn_cdc_is_bit_set 的过程,甚至编译该过程也需要几分钟,而且执行速度也非常慢。当我用手动掩码解析替换函数 sys.fn_cdc_is_bit_set 时,它的工作速度要快得多。当 substring([ $update_mask],len([ $update_mask]) - (([ColumnOrder]-1)/8),1) & power(2,([ColumnOrder]-1)%8) > 0 然后 1 的情况否则 0 结束

于 2015-03-04T08:21:11.770 回答