Codeforces878 A. Short Program

题目类型:位运算

传送门:>Here<

题意:给出\(N\)个位运算操作,要求简化操作数量,使之结果不受影响(数据在1023之内)

解题思路

我们发现数字的每一位是独立的。也就是说,每一个操作可以看成是对数字的二进制的每一位进行操作。而二进制只有0或1,因此如果我们能够确定每一位有没有变,是怎么变的,就可以确定当前这一位发生了什么操作。因此,很显然我们可以把操作数量压缩成3步:&,|,^。

每一位的变化无非四种情况:对于初始状态和最终状态,不过时0->0,0->1,1->0,1->1。

因此如果我们知道每一位初始状态为0时的最终状态,初始状态为1的最终状态,就可以确定是怎么变的了

好了,那么我们设定两个数。一个是000000000(B),一个是111111111(B)。然后计算最终状态。然后将这两个数的同一位置的变化做一个比较

依然分类讨论。

如果0->0且1->0,那么等价于在这一位&0了。

如果0->0且1->1,等价于没有操作

如果0->1且1->0,等价于^1

如果0->1且1->1,等价于|1

注意&的那个数刚开始也是111111111(B)

反思

将一个数的每一位割裂开来看似乎很自然,却很难想。

对分类讨论思想依然不是很敏感。分类讨论思想在OI中是非常重要的,要多多去想分几类,如何分类。

Code

/*By DennyQi 2018*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
const int MAXN = 10010;
const int MAXM = 20010;
const int INF = 1061109567;
inline int Max(const int a, const int b){ return (a > b) ? a : b; }
inline int Min(const int a, const int b){ return (a < b) ? a : b; }
inline int read(){
    int x = 0; int w = 1; register char c = getchar();
    for(; c ^ '-' && (c < '0' || c > '9'); c = getchar());
    if(c == '-') w = -1, c = getchar();
    for(; c >= '0' && c <= '9'; c = getchar()) x = (x<<3) + (x<<1) + c - '0'; return x * w;
}
int N,x,y,a,b,c,t,b1,b2;
char opt[5];
int main(){
    scanf("%d", &N);
    x = 0, y = (1<<10)-1;
    a = y;
    for(int i = 1; i <= N; ++i){
        scanf("%s %d", opt, &t);
        if(opt[0] == '&'){
            x &= t, y &= t;
        }
        if(opt[0] == '|'){
            x |= t, y |= t;
        }
        if(opt[0] == '^'){
            x ^= t, y ^= t;
        }
    }
    for(int i = 0; i < 10; ++i){
        b1 = (1<<i) & y;
        b2 = (1<<i) & x;
        if(b1 && !b2) continue;
        if(!b1 && !b2){
            a -= (1<<i);
        }
        if(!b1 && b2){
            c += (1<<i);    
        } 
        if(b1 && b2){
            b += (1<<i);
        }
    }
    printf("3\n");
    printf("& %d\n| %d\n^ %d",a,b,c);
    return 0;
}

转载于:https://www.cnblogs.com/qixingzhi/p/9744105.html

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值