我正在 C# 中构建一个自定义的用户定义类型以用于 SQL CLR。请参阅此参考。
底层数据可以使用常规有符号 32 位整数以 4 个字节表示。这是实现:
[Serializable]
[SqlUserDefinedType(Format.UserDefined, IsByteOrdered = true,
IsFixedLength = true, MaxByteSize = 4)]
public struct MyUDT : INullable, IBinarySerialize
{
private int _value;
public int Value
{
get { return _value; }
}
public MyUDT(int value)
: this()
{
_value = value;
}
public override string ToString()
{
return _value.ToString(CultureInfo.InvariantCulture);
}
[SqlMethod(OnNullCall = false)]
public static MyUDT Parse(SqlString s)
{
int i = int.Parse(s.Value);
return new MyUDT(i);
}
public bool IsNull { get; private set; }
public static MyUDT Null
{
get { return new MyUDT { IsNull = true }; }
}
public void Read(BinaryReader r)
{
var bytes = r.ReadBytes(4);
if (BitConverter.IsLittleEndian)
Array.Reverse(bytes);
int i = BitConverter.ToInt32(bytes, 0);
_value = i ^ unchecked((int) 0x80000000);
}
public void Write(BinaryWriter w)
{
int i = _value ^ unchecked((int) 0x80000000);
var bytes = BitConverter.GetBytes(i);
if (BitConverter.IsLittleEndian)
Array.Reverse(bytes);
w.Write(bytes);
}
}
当然,我会添加其他方法,使其更有用,但这是编译和发布的最低要求。
我需要确保当这种类型用于比较时,例如当它是表中的一列时,它是正确排序的。将有正值和负值。
如您所见,我不得不颠倒字节顺序,因为字节序是倒序的。我还必须调整字节,使零表示为0x80000000
,而 -1 变为0x7FFFFFFF
。否则,这些值将无法正确排序。
但是,这不是SQL Server 整数的正常工作方式。例如:
SELECT convert(varbinary, -16) -- 0xFFFFFFF0
SELECT convert(varbinary, 16) -- 0x00000010
然而这些就好了!所以我只能得出结论,除了原始字节顺序之外,还有其他排序类型。但是我在我的 UDT 中找不到任何方法来控制它。可能吗?
我发现的所有示例都显示如下:
public void Read(BinaryReader r)
{
_value = r.ReadInt32();
}
public void Write(BinaryWriter w)
{
w.Write(_value);
}
但那是完全不同的格式。它类似于 SQL Server 的格式,但字节顺序是相反的(1 是0x01000000
,-16 是0xF0FFFFFF
)。所以事情完全错了。那么这些例子有错吗?某处有错误吗?
我还考虑过使用Format.Native
和省略自定义序列化。那种工作,但我最终得到 5 个字节的值,其中 0 看起来像0x8000000000
-1 看起来像0x7FFFFFFF00
. 它像我一样自动转换,但它为IsNull
布尔字段存储了一个额外的字节。它不会接受将其标记为[NonSerialized]
,并且不会让我将其删除。可空性是一项要求,因此它必须存在。
肯定有比必须进行位操作更简单的方法吗?
另一个问题是我的类型不能直接转换为int
内部 SQL。虽然,我不确定我是否需要它。
另请参阅DBA.SE 上的相关(但不同)问题。