题目描述
我们知道从n个非负整数中任取两个相加共有n*(n-1)/2个和,现在已知这n*(n-1)/2个和值,要求n个非负整数。
输入格式:
输入文件有若干行,每行一组数据,包含n*(n-1)/2+1个空格隔开的非负整数,其中第一个数表示n(2
输出格式:
输出文件若干行,对应每一个输入,该行按从小到大的次序依次输出一组满足要求的n个非负整数,相邻两个整数之间用一个空格隔开;若问题无解则输出“Impossible”。
输入输出样例
输入样例
3 1269 1160 1663
输出样例
383 777 886
思路
首先,它输入的和是没有标号的,所以我们自行标号
我们按从小到大的顺序给要求的数编号
所以先把和从小到大排一下序
最小的和一定是最小数与第2小数的和
第二小的和一定是最小数与第3小数的和
所以我们枚举最小数,就可求得第2小和第3小数,同时查询在和中是否有第二小数加第三小数
若有就继续查询,剩下的和中最小的就是最小数和第4小数的和
按照这种方法一直查询下去
若没有就返回,继续枚举
所以,最终方法为:
(1)枚举最小的数;
(2)根据最小的数和最小的和,得到次小的数;
(3)由前k小的数推出第k+1小的数。
算法的时间复杂度是O(Cn),其中C表示数值的大小。
从本题的解题过程中,我们看到其中最关键的一步就是以最小的和作为突破口。在解决数学问题的时候,很多情况下从为数不多的信息中寻找突破口是至关重要的。找到了突破口,问题本身也就迎刃而解了。
代码
#include <bits/stdc++.h>
using namespace std;
#define ll long long
inline int read(){
int ret=