可见的山峰对数量

可见的山峰对数量

题目描述

一个不含有负数数组可以代表一圈环形山,每个位置的值代表山的高度。比如,{3,1,2,4,5},{4,5,3,1,2}或{1,2,4,5,3}都代表同样结构的环形山。3->1->2->4->5->3 方向叫作 next 方向(逆时针),3->5->4->2->1->3 方向叫作 last 方向(顺时针)。

山峰 A 和 山峰 B 能够相互看见的条件为:

  1. 如果 A 和 B 是同一座山,认为不能相互看见。

  2. 如果 A 和 B 是不同的山,并且在环中相邻,认为可以相互看见。

  3. 如果 A 和 B 是不同的山,并且在环中不相邻,假设两座山高度的最小值为 min。如果 A 通过 next 方向到 B 的途中没有高度比 min 大的山峰,或者 A 通过 last 方向到 B 的途中没有高度比 min 大的山峰,认为 A 和 B 可以相互看见。

问题如下:

给定一个不含有负数且没有重复值的数组 arr,请问有多少对山峰能够相互看见?

输入描述:

第一行一个整数 T,表示测试数据的组数。

每组数据的第一行有三个数字 n, p, m,其中 n 表示 山峰的数量,

山峰的高度数组等于 1 - p 的 p! 个全排列按字典序排序后的第 m 个全排列的前 n 项。

输出描述:

输出一行一个整数表示答案。

示例
输入
1
5 5 2
输出
7
说明
1-5 的全排列排序后的第二个排列 为 1 2 3 5 4
备注:

1 ≤ T ≤ 10000 1 \le T \le 10000 1T10000
1 ≤ n ≤ p ≤ 1000000 1 \le n \le p \le 1000000 1np1000000
1 ≤ m ≤ p ! 1 \le m \le p! 1mp!


题解:

找规律。。。。结果为 2 ∗ n − 3 2*n-3 2n3

我们假设寻找的规则是 “小找大” ,这样保证不会漏掉某种情况。

  • 若 n = 1,结果为 0
  • 若 n = 2,结果为 1
  • 若 n >= 3,因为每个元素都不相同,所以肯定存在最大值 max1 和次大值 max2 ,对于非 {max1, max2} 的山,其往左右两边能分别找到最近的比它高的山,这样的话,每座山的贡献 2 对,所以对于非 {max1, max2} 山,总贡献为 2 ∗ ( n − 2 ) 2*(n-2) 2(n2),再加上 max1 和 max2 形成的一对,所以最终结果为 2 ∗ n − 3 2*n-3 2n3
代码:
#include <cstdio>

using namespace std;

int main(void) {
    int t, n, p, m;
    scanf("%d", &t);
    while ( t-- ) {
        scanf("%d%d%d", &n, &p, &m);
        printf("%d\n", n == 1 ? 0 : (n << 1) - 3);
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值