洛谷 B3930 [GESP202312 五级] 烹饪问题 题解

B3930 GESP202312 五级 烹饪问题

PART1题目

题目传送门👉https://www.luogu.com.cn/problem/B3930

题目背景

对应的选择、判断题:https://ti.luogu.com.cn/problemset/1137

题目描述

有 \(N\) 种食材,编号从 \(1\) 至 \(N\),其中第 \(i\) 种食材的美味度为 \(a_i\)。

不同食材之间的组合可能产生奇妙的化学反应。具体来说,如果两种食材的美味度分别为 \(x\) 和 \(y\) ,那么它们的契合度为 x\\ \\text{and}\\ y

其中,\(\text{and}\) 运算为按位与运算,需要先将两个运算数转换为二进制,然后在高位补足 ,再逐位进行与运算。例如,\(12\) 与 \(6\) 的二进制表示分别为 \(1100\) 和 \(0110\) ,将它们逐位进行与运算,得到 \(0100\) ,转换为十进制得到 4,因此 \(12\ \text{and}\ 6 = 4\)。在 C++ 或 Python 中,可以直接使用 & 运算符表示与运算。

现在,请你找到契合度最高的两种食材,并输出它们的契合度。

输入格式

第一行一个整数 \(N\),表示食材的种数。

接下来一行 \(N\) 个用空格隔开的整数,依次为 \(a_1,\cdots,a_N\),表示各种食材的美味度。

输出格式

输出一行一个整数,表示最高的契合度。

输入输出样例 #1

输入 #1

复制代码
3
1 2 3

输出 #1

复制代码
2

输入输出样例 #2

输入 #2

复制代码
5
5 6 2 10 13

输出 #2

复制代码
8

说明/提示

样例解释 1

可以编号为 \(2,3\) 的食材之间的契合度为 \(2\ \text{and} \ 3=2\),是所有食材两两之间最高的契合度。

样例解释 2

可以编号为 \(4,5\) 的食材之间的契合度为 \(10\ \text{and}\ 13=8\),是所有食材两两之间最高的契合度。

数据范围

对于 \(40\%\) 的测试点,保证 \(N \le 1,000\);

对于所有测试点,保证 \(N \le 10^6\),\(0\le a_i \le 2,147,483,647\)。

PART2解题思路

2.1题目解释

本题要求从\(N\)个数中选出两个,是他们按位与运算结果最大。

2.2思路

  • 因为 \(N \le 10^6\) ,所以不可以直接两两枚举(时间复杂度\(O(N^{2})\) )
  • 可以注意到 \(0\le a_i \le 2,147,483,647\),所以二进制位最多只有\(31\)位
  • 二进制中高位权值大于低位
  • 所以应该优先保证高位为1,再考虑低位
  • 所以可以采用贪心策略
  • 从高位到低位进行枚举确定答案

PART3算法步骤

1.初始化答案为0

2.从高位开始枚举

  • 尝试将第一位设为1
  • 统计有多少个数满足条件
  • 如果满足的数不少于 两个,更新\(ans\)

3.输出答案

(时间复杂度:\(O(31×N)\),空间复杂度:\(O(N)\) )

PART4代码

4.1AC记录

网址:https://www.luogu.com.cn/record/277491769

4.2AC代码

复制代码
#include<bits/stdc++.h>
using namespace std;
int main() {
    unsigned long long n;cin>>n;//输入食材数量
    unsigned long long a[n+1];
    for(unsigned long long i=1;i<=n;i++) cin>>a[i];//读入
    unsigned long long ans=0;//存储最大契合度
    for(long long i=30;i>=0;i--) {//从高位到低位进行枚举
        unsigned long long f=ans|(1ull<<i);//尝试将此为设为1
        int cnt=0;
        for(unsigned long long i2=1;i2<=n;i2++) {//遍历全部食材
            unsigned long long j=a[i2];
            unsigned long long f2=j&f;
            if(f2==f) {
                cnt++;
                if(cnt>=2) {//至少两个数满足条件,则此位可以取1
                    ans=f;//更新ans
                    break;//因只需要寻找两个,所以可以直接跳出循环
                }
            }
        }
    }
    cout<<ans;//输出答案
    return 0;
}

4.3注意事项

unsigned long long

因为题目中的最大值可以达到 2,147,483,647,这个数刚好是 2^31 - 1,用 int(32位有符号整数)能存下。但在代码中进行了 1ull << i 这样的位移操作,当 i=30 时结果是 2^30,在 int 范围内没问题,但 unsigned long long 可以确保:

  • 位移时不溢出:如果将来数据范围变大或用更高的位数,unsigned long long 更安全

  • 统一数据类型:a_i 是非负整数,用无符号类型更符合语义

  • 避免符号位干扰:有符号整数的最高位是符号位,按位运算时可能产生意外结果,而 unsigned long long 所有位都是数值位,位运算更纯粹

相关推荐
玖玥拾5 小时前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
один but you6 小时前
constexpr函数
c++
凡人叶枫7 小时前
Effective C++ 条款41:了解隐式接口和编译期多态
java·开发语言·c++·effective c++
凡人叶枫7 小时前
Effective C++ 条款42:了解 typename 的双重意义
java·linux·服务器·c++
小胖xiaopangss7 小时前
BRpc使用
c++·rpc
-森屿安年-7 小时前
63. 不同路径 II
c++·算法·动态规划
chase_my_dream7 小时前
Cartographer详细讲解
c++·人工智能·自动驾驶
森G7 小时前
75、服务器源码解析---------云视频服务项目
linux·服务器·网络·c++·qt
碧海蓝天20227 小时前
C++法则24:在标准 C++ 中,没有任何可移植的方式判断指针 T* pt 指向的内存位置是否已经 构造了对象,程序员必须手动跟踪哪些元素已构造。
java·开发语言·c++