• E2. Salyg1n and Array (hard version) Codeforces Round 897 (Div. 2)


    Problem - E2 - Codeforces

    题目大意:有一个隐藏的长度为n的数组a,给出一个整数k,可以询问i,评测机返回[i,i+k-1]区间内的数的异或和,并将这一区间内的数的顺序翻转,要求在57次询问内得到整个数组的异或和

    1<=k<=n<=k*k<=2500;n,k都是偶数

    思路:因为询问一个区间后,这个区间会进行翻转,所以如果我们询问两个相邻的重合区间,那么区间重合部分被异或了两遍也就相当于没异或,其余两端的数都异或了一遍,然后这个重合区间又被反转到了最后,如果再以这个重合区间的左端点作为询问区间的左端点,那么这个重合区间的数都被异或了3遍,也就是异或了一遍,其他数也是一遍,这样就求出了这一段数的异或和,也就是我们可以用三次询问求出一段数量为(k,2k)的区间的异或和

    拿k=6,n=10举例,初始询问[1,6],数组变为[a[6],a[5],a[4],a[3],a[2],a[1],a[7],a[8],a[9],a[10]],接着因为还剩两次操作和4个数没查,所以我们询问1+(n%k)/2,即[3,8],数组变为[a[6],a[5],a[8],a[7],a[1],a[2],a[3],a[4],a[9],a[10]],最后再询问1+n%k,即[5,10],期间,这样我们a[1]~a[4]询问了三遍,其余的询问了1遍,异或起来都是一次贡献,即求得了a[1]~a[10]的异或和

    我们知道了(k,2k)的区间怎么求,其他的区间直接永不重合的长度为k的区间去求即可,总询问次数不超过53次

    1. //#include<__msvc_all_public_headers.hpp>
    2. #include
    3. using namespace std;
    4. const int N = 3e4 + 5;
    5. typedef long long ll;
    6. int n;
    7. void solve()
    8. {
    9. int k;
    10. cin >> n >> k;
    11. ll ans = 0;
    12. for (int i = 1; i + k - 1 <= n; i += k)
    13. {//询问不重合的区间
    14. cout << "? " << i << endl;
    15. cout.flush();
    16. ll x;
    17. cin >> x;
    18. ans ^= x;
    19. }
    20. if (n % k)
    21. {//必须要有重合区间
    22. cout << "? " << n - k - (n % k) / 2 + 1 << endl;
    23. cout.flush();
    24. ll x;
    25. cin >> x;
    26. ans ^= x;
    27. cout << "? " << n - k + 1 << endl;
    28. cout.flush();
    29. cin >> x;
    30. ans ^= x;
    31. }
    32. cout << "! " << ans << endl;
    33. cout.flush();
    34. }
    35. int main()
    36. {
    37. ios::sync_with_stdio(false);
    38. cin.tie(0);
    39. int t;
    40. cin >> t;
    41. while (t--)
    42. {
    43. solve();
    44. }
    45. return 0;
    46. }

  • 相关阅读:
    Map和Set
    Jmeter接口测试——使用教程(下)
    刮腻子避坑全屋挂网
    python自动化之——获取钉钉群所有人的昵称
    Redis为什么快?
    牛客竞赛每日俩题 - Day6
    一个示例学习C语言到汇编层面
    人工智能术语翻译(五)
    redis五种数据类型
    docker centos host setting
  • 原文地址:https://blog.csdn.net/ashbringer233/article/details/132876064