x & -x 的含义

博客探讨了位操作符&和-x的结合在计算机科学中的应用。通过解释按位与操作和按位取反的原理,展示了如何利用这种操作来确定数字的奇偶性以及找到数值中最低位的1。对于奇数,x&(-x)的结果始终为1;对于偶数,结果是能整除该偶数的最大的2的幂。这种方法在计算低位比特位和树状数组中具有实用价值。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

&的含义

& 按位与操作,按二进制位进行"与"运算。运算规则:(有 0 则为 0)
0 & 0 = 0;
0 & 1 = 0;
1 & 0 = 0;
1 & 1 = 1;

-x的含义

-x 在计算机存储是用x的补码存储,就是在x的值的基础上进行按位取反(~x)之后在增加1所得, 也就是说

x & -x == x & (~x + 1)

我们都知道, 当一个奇数 + 1时, 表示的二进制数则会发生进位, 这样的话, 会产生一个连锁反应,也就是最低位的那些连续的1都会被清0, 如 :

0000 0000 0111 1111 + 1 = 0000 0000 1000 0000

x为奇数

x 为奇数时就比较简单了, 因为奇数取反后的值一定是偶数, 而偶数的值 + 1之后, 并不会影响进位,故 x 与 -x 相与的结果只有最后一位是1,其他位均为0
0000 0100 1111 , 取反后的结果就变成了 1111 1011 0000 ,这个值 + 1之后为

1111 1011 0000 + 1 = 1111 1011 0001

这个结果再与最初的值相与后,只有

0000 0100 1111 & 1111 1011 0001 = 0000 0000 0001

所以
当一个奇数与它的负值相与时时,结果为1

x为偶数

如果一个偶数, 如 0000 0100 1110 (十进制为78) ,取反后的结果就变成了 1111 1011 0001 ,而当这个值 + 1之后由于发生了进位, 即

1111 1011 0001 + 1 = 1111 1011 0010

这个结果再与最初的值相与后, 只会有一位保留为1

0000 0100 1110 & 1111 1011 0010 = 0000 0000 0010 

这个结果与最初的值又有什么关系呢?
很显然, 这个值与原值的末位0的个数是一致的,而且这个结果只有一位值是1, 其他位均是0

我们可以这么描述
当一个偶数与它的负值相与时, 结果是能整除这个偶数的最大的2的幂, 即: m = x & -x , 则 x % m = 0, 且 m = 2k
实际上就是把 x 用一个奇数左移 k 位来表示
上面的例子中,原值末尾有一个0,即 k = 1,故 x & -x = 21 = 2,刚好就是整除这个偶数的最大2次幂,因为原值78可拆成2和39的乘积,即78 = 2 × 39,39是奇数,它的因数里面没有2了,故整除78的最大2次幂是2,故78可以描述成39左移1位得到,即 78 = 39 << k = 39 << 1

再举个例子,0000 0001 1000 (十进制为24),取反后的结果就变成了 1111 1110 0111,而当这个值 + 1之后由于发生进位,即

1111 1110 0111 + 1 = 1111 1110 1000

这个结果再与最初的值相与后, 只会有一位保留为1

0000 0001 1000 & 1111 1110 1000 = 0000 0000 1000 

原值末尾有3个零,故 x & -x = 23 = 8,由于24 = 8 × 3,3是奇数,因数里面没有2了,故整除24的最大2次幂是8,也就是 x & -x 的值,即24可以由3左移3位得到,24 = 3 << k = 3 << 3

最后一种情况是当 x 为 0时,x & (-x) 的结果为0

举个例子,如果一个数是偶数,那我们就把这个数拆成一个偶数和一个奇数的相乘,比如说 48 = 16 × 3

int x = 68;
if ((x & (-x)) == 1) {
	printf("%d是奇数", x);
}
else if ((x & (-x)) == 0) {
	printf("%d是零", x);
}
else {
	int even_number = x & (-x);
	int odd_number = x / even_number;
	printf("%d = %d × %d", x, even_number, odd_number);
}

x & (-x) 的用途一般是用来获取某个二进制数的 LowBit ,在树状数组中会用到
lowbit(x)是x的二进制表达式中最低位的1所对应的值

int lowbit(int x)
{
    return x & (-x);
}
d-textsize=35.0&amp;tz=28800000&amp;d-api-lev=34&amp;d-bat-lev=100&amp;aKV=1&amp;d-nettype-raw=1&amp;d-device-screen-size=432X894&amp;u-appsecure=1&amp;d-density-dependent-screen-size=1080x2235&amp;consentObject=%7B%22gdpr_consent_available%22%3Atrue%7D&amp;d-bat-chrg=0&amp;d-manufacturer-name=motorola&amp;ik=ac%2Cbid%2Cits%2Cvtm%2Cplid%2Ccatid%2Chcd%2Chsv%2Chcv&amp;u-id-map=%7B%22GPID%22%3A%226c75affc-842e-4080-9aa7-6de5b5c97dba%22%7D&amp;d-localization=en_US&amp;d-dnd=0&amp;format=unifiedSdkJson&amp;d-language=en&amp;int-origin=im&amp;loc-consent-status=denied&amp;u-age-restricted=0&amp;d-key-lang=%5B%22en%22%5D&amp;client-request-id=5fe83769-c920-40fa-a8e8-e20ccd17e19a&amp;d-bat-sav=0&amp;im-plid=10000178081&amp;d-w-h=0&amp;mk-version=pr-SAND-10.8.0-20241113&amp;s-iso=cn&amp;d-app-set-id=859626cf-034d-9b30-235d-4dc00b931db8&amp;sData=%7B%22st%22%3A1742201389417%2C%22cnt%22%3A1%2C%22u-ret%22%3A0%2C%22dep%22%3A%5B0%2C0%2C0%2C0%5D%7D&amp;m10n_context=M10N_CONTEXT_ACTIVITY&amp;u-appbid=com.happy.painting.xcim&amp;d-media-volume=0&amp;is-unifid-service-used=false&amp;loc-granularity=none&amp;d-brand-name=motorola&amp;d-av-disk=227669&amp;c_data=%5B%5D&amp;adtype=int&amp;d-airplane-m=0&amp;sdk-flavor=row&amp;d-app-set-scope=1&amp;u-id-adt=0&amp;d-model-name=moto+g+stylus+5G+-+2024&amp;d-tot-disk=244140&amp;cct-enabled=false&amp;os-v=14&amp;d-device-screen-density=2.5&amp;d-drk-m=0&amp;d-orientation=1&amp;u-appdnm=Climb+High+Master&amp;u-appver=1.0&amp;ts=1742201399953 这段字符串 &amp; 作为分割符换成换行
最新发布
03-18
&lt;think&gt;好的,用户给了一个查询字符串,要求将其中的参数用换行分割。首先,我需要确定用户的需求是什么。看起来他们可能是在调试某个API请求,或者分析网络请求中的参数,所以需要将URL参数更清晰地展示出来。 用户之前询问过js_render_ver是什么,现在给了一个更复杂的查询字符串,可能是在继续之前的问题,或者遇到了新的调试需求。需要确保正确解析每个参数,并可能希望了解某些参数的含义,但当前主要任务是将参数分割成换行。 接下来,我需要处理这个长字符串。首先,用&amp;符号分割每个参数,然后替换成换行符。但要注意是否有转义字符,比如%22是双引号,%3A是冒号,这些在URL编码中常见,所以分割时应保留这些值不变,只是将&amp;换成换行。 然后,检查每个参数是否有需要解码的部分,比如查看是否有URL编码的值需要转换,但用户可能只需要结构化的展示,而不是解码后的内容。因此,保持原样可能更合适。 此外,用户可能希望参数排列清晰,方便查看每个键值对。所以分割后,每个参数单独一行,这样可读性更高。同时,需要确保没有遗漏任何参数,或者错误分割,比如检查是否有参数值中包含&amp;但没有被正确编码的情况,但根据提供的字符串,看起来都是正确用&amp;分隔的。 最后,可能需要添加注释或说明,但用户这次没有明确要求解释参数含义,只是要求格式转换。因此,响应应专注于正确分割,保持简洁,满足用户需求。&lt;/think&gt;以下是按换行符分隔后的参数列表(已保留原始格式,仅将 `&amp;` 替换为换行): ``` d-textsize=35.0 tz=28800000 d-api-lev=34 d-bat-lev=100 aKV=1 d-nettype-raw=1 d-device-screen-size=432X894 u-appsecure=1 d-density-dependent-screen-size=1080x2235 consentObject=%7B%22gdpr_consent_available%22%3Atrue%7D d-bat-chrg=0 d-manufacturer-name=motorola ik=ac%2Cbid%2Cits%2Cvtm%2Cplid%2Ccatid%2Chcd%2Chsv%2Chcv u-id-map=%7B%22GPID%22%3A%226c75affc-842e-4080-9aa7-6de5b5c97dba%22%7D d-localization=en_US d-dnd=0 format=unifiedSdkJson d-language=en int-origin=im loc-consent-status=denied u-age-restricted=0 d-key-lang=%5B%22en%22%5D client-request-id=5fe83769-c920-40fa-a8e8-e20ccd17e19a d-bat-sav=0 im-plid=10000178081 d-w-h=0 mk-version=pr-SAND-10.8.0-20241113 s-iso=cn d-app-set-id=859626cf-034d-9b30-235d-4dc00b931db8 sData=%7B%22st%22%3A1742201389417%2C%22cnt%22%3A1%2C%22u-ret%22%3A0%2C%22dep%22%3A%5B0%2C0%2C0%2C0%5D%7D m10n_context=M10N_CONTEXT_ACTIVITY u-appbid=com.happy.painting.xcim d-media-volume=0 is-unifid-service-used=false loc-granularity=none d-brand-name=motorola d-av-disk=227669 c_data=%5B%5D adtype=int d-airplane-m=0 sdk-flavor=row d-app-set-scope=1 u-id-adt=0 d-model-name=moto+g+stylus+5G+-+2024 d-tot-disk=244140 cct-enabled=false os-v=14 d-device-screen-density=2.5 d-drk-m=0 d-orientation=1 u-appdnm=Climb+High+Master u-appver=1.0 ts=1742201399953 ``` --- ### 关键参数解析(部分) 1. **设备信息** - `d-manufacturer-name=motorola`: 设备厂商为摩托罗拉 - `d-model-name=moto+g+stylus+5G+-+2024`: 设备型号 - `d-device-screen-size=432X894`: 屏幕分辨率 2. **应用信息** - `u-appbid=com.happy.painting.xcim`: 应用包名 - `u-appver=1.0`: 应用版本号 3. **网络与本地化** - `tz=28800000`: 时区偏移(28800000ms = 8小时,UTC+8) - `d-language=en`: 设备语言为英语 - `s-iso=cn`: 服务地区为中国 4. **隐私与权限** - `consentObject=%7B...%7D`: URL解码后为 `{&quot;gdpr_consent_available&quot;:true}`,表示 GDPR 同意状态 - `loc-consent-status=denied`: 地理位置权限被拒绝 5. **调试标识** - `client-request-id=5fe83769-...`: 唯一请求 ID - `mk-version=pr-SAND-10.8.0-20241113`: SDK 或框架版本 --- 如需具体参数的详细解释,可指出需要分析的字段!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值