7-4 三足鼎立 (25 分)(每日一练)

2023-11-11 09:31
文章标签 每日 25 三足鼎立

本文主要是介绍7-4 三足鼎立 (25 分)(每日一练),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

7-4 三足鼎立 (25 分)

目录

输入格式:

输出格式:

输入样例:

输出样例:

样例解释:

题意理解:

二分法查找原理

具体思路:

代码


当三个国家中的任何两国实力之和都大于第三国的时候,这三个国家互相结盟就呈“三足鼎立”之势,这种状态是最稳定的。

现已知本国的实力值,又给出 n 个其他国家的实力值。我们需要从这 n 个国家中找 2 个结盟,以成三足鼎立。有多少种选择呢?

输入格式:

输入首先在第一行给出 2 个正整数 n(2≤n≤105)和 P(≤109),分别为其他国家的个数、以及本国的实力值。随后一行给出 n 个正整数,表示n 个其他国家的实力值。每个数值不超过 109,数字间以空格分隔。

输出格式:

在一行中输出本国结盟选择的个数。

输入样例:

7 30
42 16 2 51 92 27 35

结尾无空行

输出样例:

9

结尾无空行

样例解释:

能联合的另外 2 个国家的 9 种选择分别为: 

{16, 27}, {16, 35}, {16, 42}, {27, 35}, {27, 42}, {27, 51}, {35, 42}, {35, 51}, {42, 51}

1.题意理解:任意两个数之和大于第三个数---->联想三角形的三边关系
         也就是说只要这三个数满足是三角形的三边关系就一定满足任意两个数之和大于第三个数
        而满足任意两个数之和大于第三个数就一定满足为三角形
        判断三条边是否能组成三角形的所有理由,一个是:三角形内任意2边之和大于第3边,能满足这个就够了.
        但:(三角形内任意2边之差小于第3边)这个也算,
  
  2.做题思路:一开始我用暴力破解:即一个循环输入,再来一个嵌套循环遍历数组找出符合的其他两个国家,
                 但是本题数据过大,运行会超时。代码如下:

#include <stdio.h>
int main() {long long n,p,i,j,sum=0;scanf("%lld %lld",&n,&p);if(n>=2&&n<=100000) {long long a[n];for(i=0; i<n; i++) {scanf("%lld",&a[i]);}for(i=0; i<=p-2; i++) {for(j=i+1; j<n; j++) {if(a[i]+a[j]>p&&a[i]+p>a[j]&&a[j]+p>a[i]) {sum++;}}}printf("%lld",sum);}return 0;
} 


于是整个题就变成了如何降低时间复杂度的问题,我们可以不用暴力法,
既然我一开始是一个一个查找并且就是这里在数据大时耗时  所以想到改用二分法查找
既然涉及到二分查找那就讲一下原理: 


二分法查找原理: 1.它和排序是绑定在一起的(升序) 
                      2.如找到等于10的数 :

              
                    由以上可只二份查找有两个关键点:
                一是:查找的数组一定是排序的
             二是: 它是通过移动左右两个边界进行缩小查找范围。
         (在很大的数据时一个一个遍历是很耗时的 ) 
           三是:通过左右之间的中间数进行比较。 
                              


具体思路:
方法  1:

(思想来源:本文链接:7-8 三足鼎立 (25分)__努力努力再努力_的博客-CSDN博客) 
(1)找规律(画图解析):

1.先将数组整个排序,按照从大到小的顺序,然后遍历,
                  2. 将这个数固定为选的一条边,结合自已输入的(本国的数)这样就有同时的两条边,
                   现在只需要确定第三条边就行
                   
    3.你可能会问16不也满足吗?是的,但在遍历到16时就已经把27的情况考虑啦
    所以每次都是从i+1到n中选。               
        4.可以看出围绕r-l+1来做 ; 
        5.现在我们需要做的就是二分法找到大于最小边界的数和小于最大边界的数即可,
        6.然后下标相减就是所要的(这只是一次的结果) */


#include <stdio.h>
#include <math.h>
#include <stdlib.h>
int cmp(const void *a,const void *b){return *(int *)a-*(int *)b;
}
int main() 
{long long n,p,y,i,h,sum=0;scanf("%lld %lld",&n,&p);y=n+1;//方便从1开始标记 long long a[y];a[0]=0;for(i=1;i<=n;i++)scanf("%lld",&a[i]);//排序(升序) qsort(a,y,sizeof(long long),cmp);//开始用二分法long long left,right;long long m;//存中间位的下标long long num1,num2;//存每次大于最小值的边界下标和每次小于最大值的边界下标 for(i=1;i<=n;i++){left=i+1;right=n;long long max,min;//每次要满足的a[x]<max(两边之和),a[x]>min(两边这差)max=a[i]+p;min=fabs(a[i]-p);//接下来确定大于最小值的边界while(left<=right){//这个一般都是作位二分查找的循环条件 m=(right+left)/2;//每次它都会变一定要放在里面,注意中间下标是加号不是减号!!!//找左位关键是a[m]小于min即min在a[m]和right之间才会移动左位 //在这里右位不是关键 if(a[m]<=min){left=m+1;} else{right=m-1;}} num1=left;left=i+1;right=n; //找小于最大值的边界的小标while(left<=right){m=(right+left)/2;if(a[m]>=max){right=m-1;} else{left=m+1;}}num2=right;h=(num2-num1)+1;//h只是一次循环的情况sum=sum+h;//sum是总和} printf("%lld",sum);return 0;
}

这篇关于7-4 三足鼎立 (25 分)(每日一练)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

五大特性引领创新! 深度操作系统 deepin 25 Preview预览版发布

《五大特性引领创新!深度操作系统deepin25Preview预览版发布》今日,深度操作系统正式推出deepin25Preview版本,该版本集成了五大核心特性:磐石系统、全新DDE、Tr... 深度操作系统今日发布了 deepin 25 Preview,新版本囊括五大特性:磐石系统、全新 DDE、Tree

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数

每日一练7:简写单词(含链接)

1.链接 简写单词_牛客题霸_牛客网 2.题目 3.代码1(错误经验) #include <iostream>#include <string>using namespace std;int main() {string s;string ret;int count = 0;while(cin >> s)for(auto a : s){if(count == 0){if( a <=

【每日刷题】Day113

【每日刷题】Day113 🥕个人主页:开敲🍉 🔥所属专栏:每日刷题🍍 🌼文章目录🌼 1. 91. 解码方法 - 力扣(LeetCode) 2. LCR 098. 不同路径 - 力扣(LeetCode) 3. 63. 不同路径 II - 力扣(LeetCode) 1. 91. 解码方法 - 力扣(LeetCode) //思路:动态规划。 cl

【JavaScript】LeetCode:21-25

文章目录 21 最大子数组和22 合并区间23 轮转数组24 除自身以外数组的乘积25 缺失的第一个正数 21 最大子数组和 贪心 / 动态规划贪心:连续和(count)< 0时,放弃当前起点的连续和,将下一个数作为新起点,这里提供使用贪心算法解决本题的代码。动态规划:dp[i]:以nums[i]为结尾的最长连续子序列(子数组)和。 dp[i] = max(dp[i - 1]

力扣 739. 每日温度【经典单调栈题目】

1. 题目 理解题意: 1.1. 给一个温度集合, 要返回一个对应长度的结果集合, 这个结果集合里面的元素 i 是 当前 i 位置的元素的下一个更高温度的元素的位置和当前 i 位置的距离之差, 若是当前元素不存在下一个更高温度的元素, 则这个位置用0代替; 2. 思路 本题用单调栈来求解;单调栈就适用于来求当前元素左边或者右边第一个比当前元素大或者小的元素;【单调栈:让栈中的元素保持单调

每日一题——第八十一题

打印如下图案: #include<stdio.h>int main() {int i, j;char ch = 'A';for (i = 1; i < 5; i++, ch++){for (j = 0; j < 5 - i; j++){printf(" ");//控制空格输出}for (j = 1; j < 2 * i; j++)//条件j < 2 * i{printf("%c", ch

每日一题,力扣leetcode Hot100之238.除自身以外数组的乘积

乍一看这个题很简单,但是不能用除法,并且在O(N)时间复杂度完成或许有点难度。 考虑到不能用除法,如果我们要计算输出结果位置i的值,我们就要获取这个位置左边的乘积和右边的乘积,那么我新设立两个数组L和R。 对于L来说,由于表达的是位置i左边的数的乘积,那么L[0]=1,因为第一个数字左边没数那么为了不影响乘积初始值就设置为1,那么L[1]=L[0]*nums[0],那么L[i]=L[i-1

英语每日一段 195

Promising economic indicators won’t instantly reverse the lingering impact of hard times for millions of families, workplace culture expert Jessica Kriegel said. “Perception and reality are sometimes