Codeforces C. Nezzar and Symmetric Array (#698 Div.2) (构造 / 思维)

2023-10-31 19:32

本文主要是介绍Codeforces C. Nezzar and Symmetric Array (#698 Div.2) (构造 / 思维),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

传送门

题意: 有一个数组a,由2n个不同的整数组成,对于每个a_{i}都有一个a_{j}使得 a_{i} = −a_{j} (1<=i,j<=2n)。

有一结果数组d,其 d_{i} = \sum _{j=1}^{2n}|a_{i}-a_{j}|,现在给出这个数组d,能否找到对应的数组a。

思路: 

  * 由于是找对称点,可以放在数轴上来讨论。a_{i}为数轴上的点,d_{i}为其与其他所有点的距离和。

  *  不难发现,d值大的点在两侧,d值小的点在中间。由于是对称的,我们讨论一边即可。

  *  不难发现:a_{i-1}a_{i}间的距离 x_{i} = |a_{i-1}-a_{i}|。

              d_{i} = ( (a_{i-1}点到其左侧所有点的距离和) + (n+i-1)*x_{i} ) + (a_{i}到其右侧所有点的距离和)。

              d_{i-1} = (a_{i-1}点到其左侧所有点的距离和) + ( (n-i+1)x_{i} + (a_{i}到其右侧所有点的距离和) )。

              因此  d_{i} = (2*n-2)* x_{i},若求得的 x_{i} 不为整数,则不存在数组a。

  *   现在我们确定了所有点都能找到对称点了,开始判断它们的对称距离是否为偶数,即是否能正好关于0点对称。

      即得满足 x_{1} = 2*a_{1},而 d_{1} = n*x_{1} + 2*(a_{1}到其右侧所有点的距离和),求得的 x_{1} 必须为大于0的偶数。

  *  满足以上所有条件,则能够构成合法的a数组,思路有参考大佬博客。

代码实现:

#include<bits/stdc++.h>
#define endl '\n'
#define null NULL
#define ll long long
#define int long long
#define pii pair<int, int>
#define lowbit(x) (x &(-x))
#define ls(x) x<<1
#define rs(x) (x<<1+1)
#define me(ar) memset(ar, 0, sizeof ar)
#define mem(ar,num) memset(ar, num, sizeof ar)
#define rp(i, n) for(int i = 0, i < n; i ++)
#define rep(i, a, n) for(int i = a; i <= n; i ++)
#define pre(i, n, a) for(int i = n; i >= a; i --)
#define IOS ios::sync_with_stdio(0); cin.tie(0);cout.tie(0);
const int way[4][2] = {{1, 0}, {-1, 0}, {0, 1}, {0, -1}};
using namespace std;
const int  inf = 0x3f3f3f3f;
const double PI = acos(-1.0);
const double eps = 1e-6;
const ll   mod = 1e9 + 7;
const int  N = 2e5 + 5;inline void read(int &x){char t=getchar();while(!isdigit(t)) t=getchar();for(x=t^48,t=getchar();isdigit(t);t=getchar()) x=x*10+(t^48);
}int t, n;
int d[N], st[N]; //用来存放a1-an的到其他点的距离和
map<int,int> mp;signed main ()
{IOS;cin >> t;while(t --){mp.clear();cin >> n;int cnt = 0, m = n<<1;for(int i = 0; i < m; i ++){cin >> d[i];mp[d[i]] ++;if(mp[d[i]]==1) st[++cnt] = d[i];}sort(st+1, st+cnt+1);int flag = 0;for(int i = 1; i <= cnt; i++){if(mp[st[i]]!=2){ //确保每个点a[i]都能有个和它对称的点a[j]flag = 1;     //但目前还不能确定对称距离是否为偶数break;        //即不能确定是 -1 1 这样的对称, 还是 -1 2这样的对称}}int sum = 0; //用来求a1到右边所有点的距离和if(!flag){for(int i = cnt; i > 1; i --){int k = 2*i-2;int s = st[i]-st[i-1];if(s%k){  //xi不是整数flag = 1;break;}else sum += (cnt-i+1)*(s/k);}}   //实际d1太小      //x1不是整数       //x1为奇数,不能关于0点对称if((st[1]-2*sum)<=0 || (st[1]-2*sum)%n || (st[1]-2*sum)/n%2) flag = 1; //判断x1是否能存在cout << (flag ? "NO":"YES") << endl;}return 0;
}

 

这篇关于Codeforces C. Nezzar and Symmetric Array (#698 Div.2) (构造 / 思维)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/317627

相关文章

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

Codeforces Round #261 (Div. 2)小记

A  XX注意最后输出满足条件,我也不知道为什么写的这么长。 #define X first#define Y secondvector<pair<int , int> > a ;int can(pair<int , int> c){return -1000 <= c.X && c.X <= 1000&& -1000 <= c.Y && c.Y <= 1000 ;}int m

Codeforces Beta Round #47 C凸包 (最终写法)

题意慢慢看。 typedef long long LL ;int cmp(double x){if(fabs(x) < 1e-8) return 0 ;return x > 0 ? 1 : -1 ;}struct point{double x , y ;point(){}point(double _x , double _y):x(_x) , y(_y){}point op

Codeforces Round #113 (Div. 2) B 判断多边形是否在凸包内

题目点击打开链接 凸多边形A, 多边形B, 判断B是否严格在A内。  注意AB有重点 。  将A,B上的点合在一起求凸包,如果凸包上的点是B的某个点,则B肯定不在A内。 或者说B上的某点在凸包的边上则也说明B不严格在A里面。 这个处理有个巧妙的方法,只需在求凸包的时候, <=  改成< 也就是说凸包一条边上的所有点都重复点都记录在凸包里面了。 另外不能去重点。 int

Codeforces 482B 线段树

求是否存在这样的n个数; m次操作,每次操作就是三个数 l ,r,val          a[l] & a[l+1] &......&a[r] = val 就是区间l---r上的与的值为val 。 也就是意味着区间[L , R] 每个数要执行 | val 操作  最后判断  a[l] & a[l+1] &......&a[r] 是否= val import ja

leetcode105 从前序与中序遍历序列构造二叉树

根据一棵树的前序遍历与中序遍历构造二叉树。 注意: 你可以假设树中没有重复的元素。 例如,给出 前序遍历 preorder = [3,9,20,15,7]中序遍历 inorder = [9,3,15,20,7] 返回如下的二叉树: 3/ \9 20/ \15 7   class Solution {public TreeNode buildTree(int[] pr

颠覆你的开发模式:敏捷思维带来的无限可能

敏捷软件开发作为现代软件工程的重要方法论,强调快速响应变化和持续交付价值。通过灵活的开发模式和高效的团队协作,敏捷方法在应对动态变化和不确定性方面表现出色。本文将结合学习和分析,探讨系统变化对敏捷开发的影响、业务与技术的对齐以及敏捷方法如何在产品开发过程中处理持续变化和迭代。 系统变化对敏捷软件开发的影响 在敏捷软件开发中,系统变化的管理至关重要。系统变化可以是需求的改变、技术的升级、

C++中类的构造函数调用顺序

当建立一个对象时,首先调用基类的构造函数,然后调用下一个派生类的 构造函数,依次类推,直至到达派生类次数最多的派生次数最多的类的构造函数为止。 简而言之,对象是由“底层向上”开始构造的。因为,构造函数一开始构造时,总是 要调用它的基类的构造函数,然后才开始执行其构造函数体,调用直接基类构造函数时, 如果无专门说明,就调用直接基类的默认构造函数。在对象析构时,其顺序正好相反。

Codeforces Round 971 (Div. 4) (A~G1)

A、B题太简单,不做解释 C 对于 x y 两个方向,每一个方向至少需要 x / k 向上取整的步数,取最大值。 由于 x 方向先移动,假如 x 方向需要的步数多于 y 方向的步数,那么最后 y 方向的那一步就不需要了,答案减 1 代码 #include <iostream>#include <algorithm>#include <vector>#include <string>

Java构造和解析Json数据的两种方法(json-lib构造和解析Json数据, org.json构造和解析Json数据)

在www.json.org上公布了很多JAVA下的json构造和解析工具,其中org.json和json-lib比较简单,两者使用上差不多但还是有些区别。下面首先介绍用json-lib构造和解析Json数据的方法示例。 一、介绍       JSON-lib包是一个beans,collections,maps,java arrays 和XML和JSON互相转换的包,主要就是用来解析Json