位运算-入门

语法

位运算符

位运算符作用于位,并逐位执行操作。&、 | 和 ^ 的真值表如下所示:

p

q

p & q

p | q

p ^ q

0

0

0

0

0

0

1

0

1

1

1

1

1

1

0

1

0

0

1

1

假设如果 A = 60,且 B = 13,现在以二进制格式表示,它们如下所示:

A = 0011 1100

B = 0000 1101

-----------------

A&B= 0000 1100

A|B= 0011 1101

A^B= 0011 0001

~A = 1100 0011

下表显示了 C++ 支持的位运算符。假设变量 A 的值为 60,变量 B 的值为 13,则:

运算符

描述

实例

&

如果同时存在于两个操作数中,二进制 AND 运算符复制一位到结果中。

(A & B) 将得到 12,即为 0000 1100

|

如果存在于任一操作数中,二进制 OR 运算符复制一位到结果中

(A | B) 将得到 61,即为 0011 1101

^

如果存在于其中一个操作数中但不同时存在于两个操作数中,二进制异或运算符复制一位到结果中。

(A ^ B) 将得到 49,即为 0011 0001

~

二进制补码运算符是一元运算符,具有"翻转"位效果,即0变成1,1变成0。

(~A ) 将得到 -61,即为 1100 0011,一个有符号二进制数的补码形式。

二进制左移运算符。左操作数的值向左移动右操作数指定的位数。

A

>>

二进制右移运算符。左操作数的值向右移动右操作数指定的位数。

A >> 2 将得到 15,即为 0000 1111

数字转型

a*1ll (第一个是数字1 第二第三个是字母L)相当于(long long) a;

快速冥

对一个数取冥 会出现多余的运算

如 :2的1000次方 (b)

2*2*2*2*2*2*2*.....

不妨写成 2 * 2^2 * 2^4 * 2^8...... a

这样只需要判断b的二进制形式的末尾 是0还是1 , 如果是1 就 乘上结果 如果是0则继续

每次还需要对a进行迭代 第一次是2 第二次是 4 第三次是16......

时间复杂度就成了logn ;

最开始所说的多余运算 就是 运算子项 4*2 (及2的三次方)不如跳开不求 直接求4*4(及2的四次方),数小的时候 可能只是减少了一次运算,但是平方运算不是线性的 次方越大 减少的就越多。

当然还有其他的划分原则 竟然用c++ 不妨直接利用二进制

最短Hamilton路径

 

最开始想法是DFS ,看N的范围20就放弃了一下 ,后面想DFS加上剪枝, 其实剪枝改变不来最坏情况的运算。

DP+状态压缩。

状态方程:

f[ allPoint][now point]= f[ allPoint -now point][ upper point] +wi[upper point][now point];

用一维表示走过点的集合,二维表示落脚点。

及只考虑走过了哪些点 以即最后落在了哪个点。所有的状态是n*n。

大概相当于一种局部的剪枝,DFS不能剪掉局部,最后只会记录一个值。

例如

1->2->3->...4 ==18

1->3->2->...4 ==20

中间省略的东西一样 剪纸的判断是 cnt>ans;

这样的判断并不能有效的剪掉,还是会计算到数值末尾,而且这种情况普遍存在。

用状态转移方程,将问题的思考角度改变成

每次取到局部最优 最优的再加上下一次,有效的剪枝。

对于状态方程的迭代 还要考虑两个判断条件。

从方程中不难看出 每次循环要判断 now point 和 upper point 是否已经在allPoint集合中 如果不在集合中则要跳过循环。

压缩状态:

采用二进制的思维 0不存在1存在 再加上位运算符 。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值