算法日记33:14届蓝桥C++B冶炼金属(二分答案)

一、题目:

二、题解:

1、思路解析:

1)首先我们可以发现题目的样例数量为( n < = 1000 n<=1000 n<=1000),因此我们可以考虑 O ( n ∗ l o g n ) O(n*log^n) O(n∗logn)时间复杂度的算法

2)通过观察题目我们可以发现,要求我们求解答案的最小值与最大值,且答案拥有一定的单调性

3)因此我们的第一个思想是用二分答案求解:

2、二分答案详细过程:

1)找变量关系,判断合法区间

  • 通过画图我们发现,当转化率 V V V不断增加时, b [ i ] b[i] b[i]的变化如下,虽然一整段 b [ i ] b[i] b[i]的值单调性,但是我们可以对其进行分段
    • 二分: a [ i ] / x < = b [ i ] a[i]/x<=b[i] a[i]/x<=b[i]的区间,求解最小值(蓝色区间)
    • 二分: a [ i ] / x > = b [ i ] a[i]/x>=b[i] a[i]/x>=b[i]的区间,求解最大值(红色区间)

2)判断区间端点

1# 找最小值
  • 二分函数构建
    • 此时,我们发现其合法区间为 < = b [ i ] <=b[i] <=b[i](因为我们要使得我们想要求解的最值处于划分的区间边界)
    • 因此 我们可以通过判断 a [ i ] / x a[i]/x a[i]/x的值来判断是否合法(最后return r)
cpp 复制代码
//合法区间<=b[i]的值
bool check_min(ll x) //x表示当前的转化率
{
  for(ll i=1;i<=n;i++) 
  {
    if(a[i]/x>b[i]) return false;
  }
  return true;
}
  • 二分入口构建
    • check_min(mid)成立时,说明此时 a [ i ] / x < = b [ i ] a[i]/x<=b[i] a[i]/x<=b[i], m i d mid mid偏大,因此r=mid
    • 最后reutrn r即可(如图所示, r r r所指向的部分为答案)
cpp 复制代码
  ll l = 1, r = 1e9 + 7;
  while(l+1<r)  //二分答案找最满足条件的最小值
  {
    ll mid=(l+r)>>1;
    if(check_min(mid))   r=mid; 
    else l=mid;
  }
  v_min=r;
2#找最大值
  • 二分函数构建
    • 此时,我们发现其合法区间为 > = b [ i ] >=b[i] >=b[i](因为我们要使得我们想要求解的最值处于划分的区间边界)
    • 因此 我们可以通过判断 a [ i ] / x a[i]/x a[i]/x的值来判断是否合法(最后return L)
cpp 复制代码
bool check_max(ll x) //合法区间>=b[i]的值
{
  for(ll i=1;i<=n;i++) 
  {
    if(a[i]/x<b[i]) return false;
  }
  return true;
}
  • 二分入口构建
    • check_min(mid)成立时,说明此时 a [ i ] / x > = b [ i ] a[i]/x>=b[i] a[i]/x>=b[i], m i d mid mid偏小,因此l=mid
    • 最后reutrn l即可(如图所示, l l l所指向的部分为答案)
cpp 复制代码
 l=1,r=1e9+7;
  while(l+1<r)  //二分答案找满足条件的最大值
  {
    ll mid=(l+r)>>1;
    if(check_max(mid)) l=mid;
    else r=mid;
  }
  v_max=l;

3、完整代码解析:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

typedef long long ll;
const int N = 2e5 + 7;
ll a[N], b[N], n;

//合法区间<=b[i]的值
bool check_min(ll x) //x表示当前的转化率
{
  for(ll i=1;i<=n;i++) 
  {
    if(a[i]/x>b[i]) return false;
  }
  return true;
}

bool check_max(ll x) //合法区间>=b[i]的值
{
  for(ll i=1;i<=n;i++) 
  {
    if(a[i]/x<b[i]) return false;
  }
  return true;
}

int main() {
  cin >> n;
  ll v_min,v_max;

  for (int i = 1; i <= n; i++) 
  {
    cin >> a[i] >> b[i];
  }
  //转化率V->:不断增加        Vmin       Vmax
  //结果:>b[i] >b[i]  >b[i] =b[i] =b[i] =b[i]  <b[i] <b[i] <b[i]
  ll l = 1, r = 1e9 + 7;
  while(l+1<r)  //二分答案找最满足条件的最小值
  {
    ll mid=(l+r)>>1;
    if(check_min(mid))   r=mid; 
    else l=mid;
  }
  v_min=r;

  l=1,r=1e9+7;
  while(l+1<r)  //二分答案找满足条件的最大
  {
    ll mid=(l+r)>>1;
    if(check_max(mid)) l=mid;
    else r=mid;
  }
  v_max=l;

    
  cout << v_min << ' ' << v_max;
  return 0;
}
相关推荐
JAVA面经实录9176 小时前
Java企业级工程化·终极完整版背诵手册(无遗漏、全覆盖、面试+落地通用)
java·开发语言·面试
王老师青少年编程6 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
周杰伦fans7 小时前
AutoCAD .NET 二次开发:深入理解 EntityJig 的工作原理与正确实现
开发语言·.net
叼烟扛炮7 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说7 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove8 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung8 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了8 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL8 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
Bat U9 小时前
JavaEE|多线程初阶(七)
java·开发语言