【BZOJ 3107】【CQOI 2013】二进制a+b

本文介绍了一种不依赖动态规划的构造方法来解决与三个整数a、b、c的二进制表示中1的数量相关的特定位运算问题。该方法根据不同情况分为三类进行处理,并给出了具体实现细节。
摘要由CSDN通过智能技术生成

网上的写法都是dp,然后发现一个构造写法,太稳了ORZ
http://blog.csdn.net/PoPoQQQ/article/details/48006557
具体的证明可以看这个博客,我这里就只写构造方法了。

首先答案只和a、b、c二进制中1的数量有关,不妨设为x、y、z且x>=y。
分成三种情况(几种特殊情况也能包括进去):

1<=z<=y

0–0000–11111111–111111
0–1111–00000000–111111
1–0000–00000000–111110
从左到右分别有1位、y-z位、x-z位、z位。

res = 1 << (x + y - z);
res = res + (1 << z) - 2;

y<z<=x

0–11111–1111111–11111
0–00000–1111111–00000
1–00000–1111110–11111
从左到右分别有1位、x-z位、y位、z-y位。
实际上去掉最后z-y位就和上面的一模一样了。

res = 1 << x;
res = res + (((1 << y) - 2) << (z - y));
res = res + (1 << (z - y)) - 1;

x<z<=x+y

0–11111–11111111–0000
0–11111–00000000–1111
1–11110–11111111–1111
从左到右分别有1位、x+y-z位、z-y位、z-x位。

res = (1 << (z + 1)) - 1;
res = res - (1 << (z + z - x - y));

有两种情况是无解的:
1、z>x+y,这不废话么。。。
2、答案所需位数小于位数上限,也就是说答案太长了放不下。

#include<cmath>
#include<cstdio>
#include<vector>
#include<queue>
#include<cstring>
#include<iomanip>
#include<stdlib.h>
#include<iostream>
#include<algorithm>
#define ll long long
#define inf 1000000000
#define mod 1000000007
#define N 100000
#define fo(i,a,b) for(i=a;i<=b;i++)
#define fd(i,a,b) for(i=a;i>=b;i--)
using namespace std;
int a,b,c,x,y,z,res,lim;
int calc(int x)
{
    int res = 0;
    while (x) {if (x&1) res++; x = x >> 1;}
    return res;
}
int dgt(int x)
{
    int res = 0;
    while (x) {res++; x = x >> 1;}
    return res;
}
int main()
{
    scanf("%d%d%d",&a,&b,&c);
    lim = max(dgt(a),max(dgt(b),dgt(c)));
    x = calc(a); y = calc(b); z = calc(c);
    if (x < y) swap(x,y);
    if (z <= y)
        {
            res = 1 << (x + y - z);
            res = res + (1 << z) - 2;
        } else
    if (z <= x)
        {
            res = 1 << x;
            res = res + (((1 << y) - 2) << (z - y));
            res = res + (1 << (z - y)) - 1;
        } else
    if (z <= x + y)
        {
            res = (1 << (z + 1)) - 1;
            res = res - (1 << (z + z - x - y));
        } else {printf("-1\n"); return 0;}
    if (dgt(res) > lim) printf("-1\n"); else printf("%d\n",res);
    return 0;
}
内容概要:该题库专为研究生入学考试计算机组成原理科目设计,涵盖名校考研真题、经典教材课后习题、章节题库和模拟试题四大核心模块。名校考研真题精选多所知名高校的计算机组成原理科目及计算机联考真题,并提供详尽解析,帮助考生把握考研命题趋势与难度。经典教材课后习题包括白中英《计算机组成原理》(第5版)和唐朔飞《计算机组成原理》(第2版)的全部课后习题解答,这两部教材被众多名校列为考研指定参考书目。章节题库精选代表性考题,注重基础知识与重难点内容,帮助考生全面掌握考试大纲要求的知识点。模拟试题依据历年考研真题命题规律和热门考点,精心编制两套全真模拟试题,并附标准答案,帮助考生检验学习成果,评估应试能力。 适用人群:计划参加研究生入学考试并报考计算机组成原理科目的考生,尤其是需要系统复习和强化训练的学生。 使用场景及目标:①通过研读名校考研真题,考生可以准确把握考研命题趋势与难度,有效评估复习成效;②通过经典教材课后习题的练习,考生可以巩固基础知识,掌握解题技巧;③通过章节题库的系统练习,考生可以全面掌握考试大纲要求的各个知识点,为备考打下坚实基础;④通过模拟试题的测试,考生可以检验学习成果,评估应试能力,为正式考试做好充分准备。 其他说明:该题库不仅提供详细的题目解析,还涵盖了计算机组成原理的各个方面,包括计算机系统概述、数据表示与运算、存储器分层、指令系统、中央处理器、总线系统和输入输出系统等。考生在使用过程中应结合理论学习与实践操作,注重理解与应用,以提高应试能力和专业知识水平。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值