起床困难综合症(0x01位运算)

简介: 笔记

起床困难综合症


题意

drd 的防御战线由 n 扇防御门组成。


每扇防御门包括一个运算 op 和一个参数 t,其中运算一定是 OR,XOR,AND 中的一种,参数则一定为非负整数。


如果还未通过防御门时攻击力为 x,则其通过这扇防御门后攻击力将变为 x op t。


最终 drd 受到的伤害为对方初始攻击力 x 依次经过所有 n 扇防御门后转变得到的攻击力。


由于 atm 水平有限,他的初始攻击力只能为 0 到 m 之间的一个整数(即他的初始攻击力只能在 0,1,…,m 中任选,但在通过防御门之后的攻击力不受 m 的限制)。


为了节省体力,他希望通过选择合适的初始攻击力使得他的攻击能让 drd 受到最大的伤害,请你帮他计算一下,他的一次攻击最多能使 drd 受到多少伤害。


思路

这道题的前提是 位运算的主要特点之一是在二进制表示下不进位


所以假设我们选择的初始值为x 那么我们可以分别考虑 x 的每一位应该选择 0 还是 1


最后组合到一起,就得到了最终的 x


那么第 k 位 选 1 还是 0 呢?


对于下面两种情况 选择 1 更优


已经填好的更高位的数值加上 1 << k 小于等于 m。

初始值为 1 得到的最终值比初始值为 0 得到的最终值大

若不满足上面两个条件,那么肯定是选 0 更优


每一位都确定好之后,就可以得到答案


代码

#include<bits/stdc++.h>
// #define int long long
#define INF 0x3f3f3f3f
#define mod 1000000007
#define rep(i, st, ed) for (int (i) = (st); (i) <= (ed);++(i))
#define pre(i, ed, st) for (int (i) = (ed); (i) >= (st);--(i))
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
template<typename T> inline T gcd(T a, T b) { return b ? gcd(b, a % b) : a; }
template<typename T> inline T lowbit(T x) { return x & -x; }
const int N = 1e5 + 10;
int n, m;
pair<string, int>a[N];
int cal(int bit, int x) {
  for (int i = 1; i <= n; ++i) {
    int k = a[i].second >> bit & 1;
    if (a[i].first == "AND") x &= k;
    else if (a[i].first == "OR") x |= k;
    else x ^= k;
  }
  return x;
}
void solve() {
  cin >> n >> m;
  for (int i = 1; i <= n; ++i) {
    string str;
    int x;
    cin >> str >> x;
    a[i] = { str,x };
  }
  int res = 0, val = 0;
  for (int i = 29; i >= 0; --i) {
    int res0 = cal(i, 0);
    int res1 = cal(i, 1);
    if (val + (1 << i) <= m && res0 < res1) {
      val += 1 << i;
      res += res1 << i;
    }
    else res += res0 << i;
  }
  cout << res << endl;
}
signed main() {
  // int t; cin >> t;
  // while (t--)
    solve();
  return 0;
}


目录
相关文章
|
算法 C++
【洛谷算法题】P5707-上学迟到【入门1顺序结构】
【洛谷算法题】P5707-上学迟到【入门1顺序结构】
|
5月前
|
算法 关系型数据库 程序员
第一周算法设计与分析:B : 如何溜的最快
这篇文章提供了解决算法问题"如何溜的最快"的方法,即计算从原点(0,0)到任意点(x,y)所需的最短步数,每步长度固定为R,通过特判和计算总距离除以步长向上取整来确定步数。
|
8月前
【编程题-错题集】kotori和气球(组合数学)
【编程题-错题集】kotori和气球(组合数学)
C++——“甜蜜的谎言”之循环嵌套
C++——“甜蜜的谎言”之循环嵌套
四道好题分享(看似简单,但是棘手)
四道好题分享(看似简单,但是棘手)
123 0
(拯救选择困难症)随机选择今天中午吃啥
(拯救选择困难症)随机选择今天中午吃啥
(拯救选择困难症)随机选择今天中午吃啥
|
算法 JavaScript 前端开发
日拱算法:解两道“杨辉三角”题
什么是“杨辉三角”,想必大家并不陌生~~ 在「杨辉三角」中,每个数是它左上方和右上方的数的和。