我有一个不同的位掩码字段,我想对它执行按位AND.
PG::Error: ERROR: cannot AND bit strings of different sizes SELECT "groups".* FROM "groups" WHERE (read_roles_bitmask = B'0' OR read_roles_bitmask & B'10' > B'0')
(你需要在表中使用不同长度的位掩码来获得此错误.)
我期待按位数学看起来如下:
00010& 100000010 = 00010
我也尝试将位掩码转换为整数而没有运气.
为什么Postgresql会扼杀这个?
我应该如何重写这个查询才能很好地发挥?
我能够使用以下内容来使按位运算符工作:
LPAD(read_roles_bitmask :: VARCHAR,64,‘0’):: BIGINT
但是这仅限于64位,有更好的方法吗?
解决方法
Postgresql位和位变量类型的行为非常无用,它拒绝扩展操作的位域的方式,并且它正确地扩展它们用于强制转换而不是向左扩展它们.
Pg在AND或OR操作之前用零向左扩展较小的操作数是有意义的,而不是失败.
你不能使用强制转换来获得相同的长度,因为由于某些疯狂的原因,对位(n)的强制转换正确填充参数,使其在几乎所有情况下都无用.
您可以使用类似lpad($1 :: text,length($1),length($2)),’0′):: left变量左边 – 用零填充位字段到两个长度中的较大者.这很麻烦,但它会起作用.我建议编写包装器函数以包含混乱.
或者,考虑修改src / backend / utils / adt / varbit.c中的位支持代码,以向左扩展和左截断位字段添加函数,并使用函数进行左扩展比较.基于现有代码,它应该很简单.