codeforces 879c
2 seconds
256 megabytes
standard input
standard output
Petya learned a new programming language CALPAS. A program in this language always takes one non-negative integer and returns one non-negative integer as well.
In the language, there are only three commands: apply a bitwise operation AND, OR or XOR with a given constant to the current integer. A program can contain an arbitrary sequence of these operations with arbitrary constants from 0 to 1023. When the program is run, all operations are applied (in the given order) to the argument and in the end the result integer is returned.
Petya wrote a program in this language, but it turned out to be too long. Write a program in CALPAS that does the same thing as the Petya's program, and consists of no more than 5 lines. Your program should return the same integer as Petya's program for all arguments from 0 to 1023.
The first line contains an integer n (1 ≤ n ≤ 5·105) — the number of lines.
Next n lines contain commands. A command consists of a character that represents the operation ("&", "|" or "^" for AND, OR or XOR respectively), and the constant xi 0 ≤ xi ≤ 1023.
Output an integer k (0 ≤ k ≤ 5) — the length of your program.
Next k lines must contain commands in the same format as in the input.
3
| 3
^ 2
| 1
2
| 3
^ 2
3
& 1
& 3
& 5
1
& 1
3
^ 1
^ 2
^ 3
0
You can read about bitwise operations in https://en.wikipedia.org/wiki/Bitwise_operation.
Second sample:
Let x be an input of the Petya's program. It's output is ((x&1)&3)&5 = x&(1&3&5) = x&1. So these two programs always give the same outputs.
这个题怎么讲呢,网上很多题解都用的一种方法,这就很没有意思,这时候我先讲一下我用的方法,可能比较蠢,实现起来也挺麻烦,不过好歹是a了
题意:这是一个程序,只会三种操作&、|、^, 先输入一个n,说明有n个操作,
直接看数据
3
| 3
^ 2
| 1 这个就意味着某个数x进行了三次操作,分别是x|3之后再x^2,在x|1,现在让你根据结果进行转换成至多5部操作的集合,
就是说上面的一系列步骤,我们可以化成x|3再x^2的两步操作 当然这个操作可能会有很多,就是让你转化成5步以内的操作 好了讲思路了,首先这个题要一位一位的算,这个是我首先想到的,一位一位的操作,最后进行组合,就是我的思路;
首先看这个东西 x的某一位 或0 和 与1 都不会改变这一位 对吧
再看 x的某一位 或1 和 与0 都会使这一位必定发生改变 对吧
当这一位必定发生改变的时候,我之前在这一位上的所有操作都没了意义对吧,因为前面无论我怎么操作,我最后只要在这一位上|1或&0,那么这位的值就是确定的 若是我在这一位上进行|0和&1操作,这一位的值不会发生改变,那么我们就可以跳过这个操作 也就是说我只要纪录下异或的操作就行了,每当我遇见&0和|1的时候我就清空我纪录的异或操作 拿这个例子
3
| 3
^ 2
| 1 看到我的操作思路首先看 这是个或,或的是0000000011(二进制),那么就是说我的第一位和第二位必定会发生改变,我就把我的第一位和第二位更新成1,
再看第二步 异或0000000010 把0存进第1位的数组里面(装的是异或的操作 就是说第1位第1次异或0),
把1存进第2位的数组里面(就是说第2位第1次异或1)............
接下来看第三步 或0000000001 第一位是1就是说我的第一位必定会变成1,于是第一位前面的异或操作都可以清除了,因为这一位必定变成1,
其他位是0就是说我的其他位保持不变,
这时候我们的x第一位是1,异或操作的集合为空,第二位是1,异或操作的集合是{1},第三位...........
首先我们肯定是先进行或操作和与操作,还是那句话,确定了这一位是|1还是&0,前面的所有操作都没了意义,而|0和&1我又可以忽略,因为他们不改变这一位上的值
所以先进性或和与操作,最后才轮到异或操作
先看这一位上是1还是0,如果确定是1那么这一位就|1,就是说假设我现在知道第一位和第三位是1,那么我就|5就可以了,这样的话第一位和第三位就变成了1,而其他的位不会发生改变
若是这一位上是0,那么我就进行&操作,就是说我现在确定了第一位和第三位是0,那么我就可以&1111111010这样就把第一位和第三位变成了0,其他位不发生改变
最后处理每一位的异或操作集合,假设第一位的异或集合是{0,1,0}(这里说一下,异或操作是符合交换律和结合律的),那么第一位最后就是异或上1(0^1^0 = 1),若是这位的
假设第二位的异或集合是{0,0},那么第二位最后异或0(0^0 = 0)
最后把每一位异或1的加起来:假设第一位异或1,第三位异或1,其他位异或0,那么我最后异或5即可 丑陋的代码:
#include <iostream>
#include <cstdio>
#include <queue>
using namespace std;
int arr[16];//纪录某一位上的值 这一位上是-1代表从来没发生过&0和|1的操作
vector<int> p[16];//纪录某一位上的异或操作,每当遇见|1和&0时清空
int pp[16];//纪录p[16]异或操作的结果
int main()
{
int n,i,j,k,x;
int res = 0;
char a = '&',b = '|',c = '^';
char ch;
scanf("%d",&n);
for(i = 0; i < 16; ++i)
arr[i] = -1;
//赋上初值
for(int c = 0; c < n; ++c)
{
getchar();
ch = getchar();
scanf("%d",&k);
for(i = 1,j = 1; i < 1024; (i <<= 1),++j)
{
if((i&k) && ch == '|')
{
//|1时进行处理
arr[j] = 1;
p[j].clear();
}
else if((i&k) == 0 && ch == '&')
{
//&0时进行处理
arr[j] = 0;
p[j].clear();
}
else if(ch == '^')
{
//插入异或操作
if((i&k))
x = 1;
else
x = 0;
p[j].push_back(x);
}
}
}
for(i = 1; i < 16; ++i)
{
int d = 0;
for(j = 0; j < p[i].size(); ++j)
d ^= p[i][j];
pp[i] = d;
//求每一位异或操作的最后结果
}
printf("3\n");
for(i = 1; i < 16; ++i)
{
//判断哪一位被强制转换成了1 这一位|1即可
if(arr[i] == -1)
continue;
if(arr[i] == 1)
res += 1<<(i-1);
}
printf("| %d\n",res);
res = 0;
for(i = 1; i < 16; ++i)
{
//判断哪一位被强制转换成了0 这一位&0即可
if(arr[i] == -1)
continue;
if(arr[i] == 0)
{
res += 1<<(i-1);
}
}
res = 1023^res;
printf("& %d\n",res);
res = 0;
for(i = 1; i < 16; ++i)
{
//最后输出每一位异或1的和
//就是第一位和第三位是1,其他位是0,输出^5
if(pp[i] == 1)
{
res += 1<<(i-1);
}
}
printf("^ %d\n",res);
}
-------------------------------------------------------------------------
之后我在看看别人的那个解法,想明白了今后再补上
codeforces 879c的更多相关文章
- Codeforces 879C/878A - Short Program
传送门:http://codeforces.com/contest/879/problem/C 本题是一个位运算问题——位运算的等价变换. 假设位运算符“&”“|”“^”是左结合的,且优先级相 ...
- 2021.1.23--vj补题
B - B CodeForces - 879B n people are standing in a line to play table tennis. At first, the first tw ...
- python爬虫学习(5) —— 扒一下codeforces题面
上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...
- 【Codeforces 738D】Sea Battle(贪心)
http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...
- 【Codeforces 738C】Road to Cinema
http://codeforces.com/contest/738/problem/C Vasya is currently at a car rental service, and he wants ...
- 【Codeforces 738A】Interview with Oleg
http://codeforces.com/contest/738/problem/A Polycarp has interviewed Oleg and has written the interv ...
- CodeForces - 662A Gambling Nim
http://codeforces.com/problemset/problem/662/A 题目大意: 给定n(n <= 500000)张卡片,每张卡片的两个面都写有数字,每个面都有0.5的概 ...
- CodeForces - 274B Zero Tree
http://codeforces.com/problemset/problem/274/B 题目大意: 给定你一颗树,每个点上有权值. 现在你每次取出这颗树的一颗子树(即点集和边集均是原图的子集的连 ...
- CodeForces - 261B Maxim and Restaurant
http://codeforces.com/problemset/problem/261/B 题目大意:给定n个数a1-an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a ...
随机推荐
- jQuery Dom对象操作 增、删、改、复制、包裹
1. 增(插入) 内部插入 //向每个匹配的元素内部追加内容,为最后一个子元素$('.violet').append('<div></div>'); //把所有匹配的元素追加到 ...
- linux工具介绍
http://linuxtools-rst.readthedocs.io/zh_CN/latest/tool/index.html 工具参考篇 1. gdb 调试利器 2. ldd 查看程序依赖库 3 ...
- JoyOI1035 棋盘覆盖
原题链接 对棋盘染色,坐标和为奇数的染黑,偶数为白.这时会发现对于相同颜色的格子,是无法放置骨牌的,这样我们就将所有格子分成两类,然后根据能否放置骨牌连边,最后就是求二分图最大匹配了. 这里我是用的匈 ...
- BZOJ1912或洛谷3629 [APIO2010]巡逻
一道树的直径 BZOJ原题链接 洛谷原题链接 显然在原图上路线的总长为\(2(n-1)\). 添加第一条边时,显然会形成一个环,而这条环上的所有边全部只需要走一遍.所以为了使添加的边的贡献最大化,我们 ...
- redis的五种存储类型的具体用法
String 类型操作 string是redis最基本的类型,而且string类型是二进制安全的.意思是redis的string可以包含任何数据.比如jpg图片或者序列化的对象 $redis-> ...
- flask学习视频
https://study.163.com/course/courseMain.htm?courseId=1004091002 主要 https://www.cnblogs.com/senlinyan ...
- .Net直接将Web页面table导出到Excel
项目管理系统有个统计表需要导出到Excel表中.常用的方法是在后台C#代码查询数据再写入Excel表中最后保存在目标路径. 为减轻数据库服务器的压力和保持页面的样式,能否直接将页面的表格直接导出到Ex ...
- Flex的Number和Text
今天要说的问题不是Number和String转换的问题.而是使用时容易出的一些错误: public static function ToFixed(value:Number, digits:uint ...
- 【RabbitMQ】 RabbitMQ安装
MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法.应用程序通过读写出入队列的消息(针对应用程序的数据)来通信,而无需专用连接来链接它们.消息传递指的是程序之间 ...
- 【Redis】安装 Redis接口时异常 ,系统ruby版本过低
场景 操作系统Linux CentOS 7.2,安装Redis接口时,使用命令:gem install redis ,用于系统ruby版本过低,报错“redis requires Ruby versi ...