• AtCoder Beginner Contest 254 Ex. Multiply or Divide by 2(思维题/优先队列)


    题目

    初始有两个长为n(n<=1e5)的可重集合a和b,第i个数分别是ai,bi(0<=ai<=1e9,0<=bi=1e9)

    你可以执行以下两种操作若干次:

    1. 从A集合中选一个数x出来,令x变为2x后,再放回A集合

    2. 从A集合中选一个数x出来,令x变为x/2向下取整后,再放回A集合

    问让A和B两个集合完全相同的最小操作次数,如果不能实现输出-1

    思路来源

    AtCoder Beginner Contest 254 A-Ex - 知乎

    题解

    典中典问题

    cf好像后来也出过一个,只是好像要输出具体方案,但是思路是一致的

    如果A和B集合能完全相同,a和b一定存在一组一一映射关系,

    不妨ai最后映射上的数是bj,且ai执行过第一种乘2操作,

    那么,可以视为bj执行了一次除以2操作,前提是bj需要是偶数

    将两种新操作改写为,

    1. 从B集合中选一个偶数x出来,令x变为x/2,再放回B集合

    2. 从A集合中选一个数x出来,令x变为x/2向下取整后,再放回A集合

    这样两种操作就是对称的,并且数是只会变小的,所以就可以贪心地从最大数开始考虑了

    使两个集合完全相同,等价于一对一对消去,能将两个集合消除空

    记a集合中的最大数为mxa,b集合中的最大数为mxb

    ①若mxa=mxb,成对消除

    ②若mxb>mxa

    (1)若mxb为奇数,不能操作,输出-1

    (2)若mxb为偶数,mxb/2后放回,记一次操作

    ③若mxa>mxb,mxa/2后放回,记一次操作

    代码

    1. #include
    2. using namespace std;
    3. const int N=1e5+10;
    4. int n,a[N],b[N],cnt;
    5. priority_queue<int>pa,pb;
    6. int main(){
    7. scanf("%d",&n);
    8. for(int i=1;i<=n;++i){
    9. scanf("%d",&a[i]);
    10. pa.push(a[i]);
    11. }
    12. for(int i=1;i<=n;++i){
    13. scanf("%d",&b[i]);
    14. pb.push(b[i]);
    15. }
    16. while(!pa.empty()){
    17. if(pa.top()==pb.top()){
    18. pa.pop();pb.pop();
    19. }
    20. else if(pb.top()>pa.top()){
    21. if(pb.top()&1){
    22. puts("-1");
    23. return 0;
    24. }
    25. int x=pb.top();pb.pop();
    26. pb.push(x>>1);cnt++;
    27. }
    28. else if(pa.top()>pb.top()){
    29. int x=pa.top();pa.pop();
    30. pa.push(x>>1);cnt++;
    31. }
    32. }
    33. printf("%d\n",cnt);
    34. return 0;
    35. }

  • 相关阅读:
    Spring的Ordered
    不理解路径问题的大坑记录
    多级式多传感器信息融合中的状态估计(Matlab代码实现)
    第三十七章 持久对象和SQL
    sql server笔记1(表的定义)
    Redis:send of 37 bytes failed with errno=10054
    书籍数组中的最长连续序列(4)0716
    PHP非对称与对称双向加密解密的方式
    【高速数字化仪应用案例系列】虹科数字化仪在大型物理实验领域的应用
    Classloader整理
  • 原文地址:https://blog.csdn.net/Code92007/article/details/128066646