Codeforces 1311 E. Construct the Binary Tree(构造二叉树)

Description:

You are given two integers n n n and d d d. You need to construct a rooted binary tree consisting of n vertices with a root at the vertex 1 1 1 and the sum of depths of all vertices equals to d d d.

A tree is a connected graph without cycles. A rooted tree has a special vertex called the root. A parent of a vertex v v v is the last different from v vertex on the path from the root to the vertex v. The depth of the vertex v v v is the length of the path from the root to the vertex v v v. Children of vertex v are all vertices for which v is the parent. The binary tree is such a tree that no vertex has more than 2 2 2 children.

You have to answer t independent test cases.

Input

The first line of the input contains one integer t ( 1 ≤ t ≤ 1000 ) t (1≤t≤1000) t(1t1000) — the number of test cases.

The only line of each test case contains two integers n n n and d ( 2 ≤ n , d ≤ 5000 ) d (2≤n,d≤5000) d(2n,d5000)— the number of vertices in the tree and the required sum of depths of all vertices.

It is guaranteed that the sum of n n n and the sum of d d d both does not exceed 5000 5000 5000 ( ∑ n ≤ 5000 , ∑ d ≤ 5000 ) (∑n≤5000,∑d≤5000) (n5000,d5000).

Output

For each test case, print the answer.

If it is impossible to construct such a tree, print “ N O NO NO” (without quotes) in the first line. Otherwise, print “{ Y E S YES YES}” in the first line. Then print n − 1 n−1 n1 integers p 2 , p 3 , … , p n p_2,p_3,…,p_n p2,p3,,pn in the second line, where p i p_i pi is the parent of the vertex i. Note that the sequence of parents you print should describe some binary tree.

Example

input

3
5 7
10 19
10 18

output

YES
1 2 1 3 
YES
1 2 3 3 9 9 2 1 6 
NO

Note

Pictures corresponding to the first and the second test cases of the example:

在这里插入图片描述

解题思路:

给出 n n n 个节点深度和为 d d d 问你是否可以构成一个二叉树,可以的话输出全部节点深度。
全部节点组成一条链的话深度和肯定最大,我们就先组成一条链,然后每次移动最下面那个节点,看看移动到哪里合适,然后判断是否可以组成深度和为 d d d

AC代码:

const int N = 5e3 + 5;
int n, d;
int ans[N];
int ch[N];
int fa[N];
int c, now, mx;
int cnt, pos;
int get_dep()
{
    int ans = now;
    c++;
    if (c == mx)
        now++, mx *= 2, c = 1;
    return ans;
}

int main()
{
    int t;
    sd(t);
    while (t--)
    {
        mem(ch, 0);
        sdd(n, d);
        cnt = 0;
        rep(i, 1, n - 1)
        {
            ans[i] = i;
            cnt += i;
        }
        pos = n - 1;
        now = 1, c = 1, mx = 2;
        if (cnt < d)
        {
            puts("NO");
            continue;
        }
        while (cnt > d)
        {
            int dep = get_dep();
            if (dep >= pos)
                break;
            int tmp = cnt - d;
            if (tmp >= (pos - dep))
            {
                cnt -= pos - dep;
                ans[pos] = dep;
            }
            else
            {
                ans[pos] -= tmp;
                cnt = d;
            }
            pos--;
        }
        if (cnt > d)
            puts("NO");
        else
        {
            puts("YES");
            ans[0] = 0;
            rep(i, 1, n - 1)
            {
                rep(j, 0, n - 1) if (ans[j] == ans[i] - 1 && ch[j] < 2)
                {
                    ch[j]++;
                    fa[i] = j;
                    break;
                }
            }
            rep(i, 1, n - 1)
            {
                printf("%d%c", fa[i] + 1, i == n - 1 ? '\n' : ' ');
            }
        }
    }
    return 0;
}
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值