题面

一道比较神仙的构造题。

首先确定 \(1\) 到 \(n\) 的路径长度,不妨设其长为 \(m\) 。

通过观察发现,\(m\) 就是 \(\min_{1<i<n}\{dist_{1,i} + dist_{i,n}\}\)。

如果所有的 \(| dist_{1, i} - dist_{i, n} |\) 都相等,那么可以特判一下:

  • 首先将 \(1\) 与 \(n\) 连接起来,路径长度为 \(| dist_{1, i} - dist_{i, n} |\) ;
  • 对于每一个 \(\forall 1<i<n\) :
    • 如果\(dist_{1,i}>dist_{i,n}\),那么将 \(i\) 与 \(1\) 连接,长度为 \(dist_{1,i}\) ;
    • 如果\(dist_{1,i} \leq dist_{i,n}\) ,那么将 \(i\) 与 \(n\)连接,长度为 \(dist_{i, n}\) 。

若点 \(i\) 满足 \(dist_{1,i} + dist_{i,n} = m\),则 \(i\) 在 \(1\) 到 \(n\) 的路径上。

否则可以求出 \(i\) 到 \((1, n)\) 路径的距离和 \(i\) 在路径上的“投影”的

位置,查找位置可以使用二分查找。

将 \(i\) 挂在路径上对应的点上即可。

如果找不到可以挂起的位置或者 \(1\) 到 \(n\) 的路径上有几个点重复就无解。

注意特判 \(n=2\) 的情况。

代码如下(刚好 \(100\) 行…):

#include <bits/stdc++.h>
#define DEBUG fprintf(stderr, "Passing [%s] line %d\n", __FUNCTION__, __LINE__)
#define itn int
#define gI gi
#define mk make_pair using namespace std; inline int gi()
{
int f = 1, x = 0; char c = getchar();
while (c < '0' || c > '9') {if (c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return f * x;
} const int maxn = 500003; int sz, ddd, len, n, m = 1000000007, d1[maxn], dn[maxn], dis[maxn], jlx[maxn]; struct Node
{
int id, dis;
} a[maxn];
struct Ans
{
int u, v, w;
} ans[maxn]; inline bool cmp(Node x, Node y) {return x.dis < y.dis;} inline void sub1_getans()
{
puts("TAK");
printf("1 %d %d\n", n, len);
for (int i = 2; i < n; i+=1)
{
if (d1[i] < dn[i]) printf("1 %d %d\n", i, d1[i]);
else printf("%d %d %d\n", n, i, dn[i]);
}
return;
} inline int Binary_Search(int x)
{
int l = 1, r = sz;
while (l <= r)
{
int mid = (l + r) >> 1;
if (a[mid].dis == x) return mid;
else if (a[mid].dis < x) l = mid + 1;
else r = mid - 1;
}
return -1;
} int main()
{
n = gi();
if (n == 2) //特判
{
puts("TAK\n1 2 1");
return 0;
}
d1[1] = dn[n] = 0;
for (int i = 2; i < n; i+=1) d1[i] = gi();
for (int i = 2; i < n; i+=1) dn[i] = gi(), jlx[i] = abs(d1[i] - dn[i]);
bool fl = true;
for (int i = 3; i < n; i+=1) if (jlx[i] != jlx[i - 1]) {fl = false; break;}
if (fl) {len = jlx[2]; sub1_getans(); return 0;} //特判
for (int i = 2; i < n; i+=1) m = min(m, d1[i] + dn[i]);
a[++sz] = (Node){1, 0};
d1[n] = dn[1] = m;
for (int i = 2; i < n; i+=1)
if (d1[i] + dn[i] == m) a[++sz] = (Node){i, d1[i]};
a[++sz] = (Node){n, d1[n]};
sort(a + 1, a + 1 + sz, cmp);
for (int i = 2; i < n; i+=1)
{
dis[i] = d1[i] - (d1[i] + dn[i] - m) / 2;
}
for (int i = 1; i < sz; i+=1)
{
if (a[i].dis == a[i + 1].dis) {puts("NIE"); return 0;}
ans[++ddd] = (Ans){a[i].id, a[i + 1].id, a[i + 1].dis - a[i].dis};
}
for (int i = 2; i < n; i+=1)
{
int nowlen = d1[i] + dn[i];
if (nowlen != m)
{
int find_jl = Binary_Search(dis[i]); //二分挂起的位置
if (((d1[i] + dn[i] - m) & 1) || find_jl == -1) {puts("NIE"); return 0;}
ans[++ddd] = (Ans){a[find_jl].id, i, (d1[i] + dn[i] - m) / 2};
}
}
puts("TAK");
for (int i = 1; i <= ddd; i+=1) printf("%d %d %d\n", ans[i].u, ans[i].v, ans[i].w);
return 0;
}

另:这份代码在洛谷上 AC 了, 但是在 BZOJ 上会 WA, 不知道为什么……

最新文章

  1. 魅族mx4 pro连电脑,adb无法获取devices信息解决
  2. Linux进程间通信(一): 信号 signal()、sigaction()
  3. Dubbo系列(3)_官方Demo说明
  4. RHEL7软件包管理
  5. 零散知识记录-一个MQ问题
  6. 基础面试题——HTML/CSS
  7. iblog语法高亮示例
  8. Codeforces 559A 第六周 O题
  9. textarea 的最大高度以及最小高度
  10. xp的停止更新对我们有什么影响?
  11. 二、Windows基础数据类型
  12. 49、html基础认识&amp;常用标签(1)
  13. 201521123070 《JAVA程序设计》第3周学习总结
  14. python3 dict(字典)
  15. ubuntu 终端作死体验
  16. IntelliJ IDEA生成live template(代码模板)
  17. P1282 多米诺骨牌 dp
  18. 利用ueditor保存到mysql数据库时出现乱码
  19. np.cumsum()函数和正则表达式的含义
  20. [转]OpenBLAS项目与矩阵乘法优化

热门文章

  1. HAProxy 使用小记
  2. java Socket通信,客户端与服务端相互发消息
  3. C#序列化与反序列化学习笔记
  4. Android实战项目:房贷计算器
  5. 嵊州D5T3 指令 program 神奇的位运算
  6. cf 989C
  7. “/Reports”应用程序中的服务器错误。
  8. DataGridView只显示数据源中绑定的字段
  9. 解决linux 终端UnicodeDecodeError: &#39;utf-8&#39; codec can&#39;t decode byte 0xce in position 0: invalid continuation byte
  10. IDEA科学使用