NC15052 求最值

题目

题目描述

给你一个长为 \(n\) 的序列 \(a\)

定义 \(f(i,j)=(i-j)^2+g(i,j)^2\)

\(g\) 是这样的一个函数

求最小的 \(f(i,j)\) 的值,\(i!=j\)

输入描述

第一行一个数 \(n\)

之后一行 \(n\) 个数表示序列 \(a\)

输出描述

输出一行一个数表示答案

示例1

输入

4
1 0 0 -1

输出

1

备注

对于 \(100\%\) 的数据,\(2 \leq n \leq 100000 , |ai| \leq 10000\)

题解

思路

方法一

知识点:分治,排序。

观察到, \(g(i,j) = (i-j)^2 + (sum(j)-sum(i))^2\) ,其中 \(sum(i)\) 指 \([1,i]\) 的前缀和。

将其转换为平面上的点 \((i,sum(i)),(j,sum(j))\) 则函数 \(g\) 为两点之间距离的平方,问题转换为平面最近点对问题。

这里一开始处理完已经是按 \(x\) 排序好的,直接开始分治。设区间左右端点为 \(l,r\) ,若 \(l == r\) ,则直接返回无穷大;若 \(l+1 == r\) ,则返回两点距离;若其他情况则先求得对半左右两个区间的临时最小值 \(ans\) ,用这个 \(ans\) 作为阈值找到 \(x\) 轴坐标与区间分界坐标 \(mid\) 差的平方小于 \(ans\) 的点对,然后用朴素算法找到他们之间的最小值,过程中不断更新 \(ans\) 。在最后一步,有个关键优化,先按 \(y\) 轴排序,这样对于一个定点找另一点时,若 \(y\) 轴差值平方大于等于目前 \(ans\) 可以直接跳过,可以证明大部分情况可以将 \(O(n^2)\) 降到 \(O(n)\) 。

时间复杂度 平均: \(O(n\log^2n)\) 最差:\(O(n^2 \log n)\)

空间复杂度 \(O(n)\)

方法二

知识点:分治,排序。

方法二实际上是方法一的优化。通过 \([l,mid]\) 和 \([mid+1,r]\) 的 \(d^2\) 最小值,返回合并区间的 \([l,r]\) 中 \(d^2\) 最小值,在合并时用归并把当前区间的区间按 \(y\) 排序,方便查找合并区间的最小值,是 \(O(n\log n)\) 。如果每次不排序,直接取点 \(sort\) 按 \(y\) 排序,会重复排序,最差会 \(O(n^2)\) 。

要注意的是,如果采用这种方法我们就必须是先排序好再确定可行下标,但是确定下标需要原来 \(mid\) 处的 \(x\) 坐标,因此需要额外设置变量保存。

时间复杂度 \(O(n\log^2n)\)

空间复杂度 \(O(n)\)

代码

方法一

#include <bits/stdc++.h>
#define ll long long using namespace std; struct Point {
ll x, y;
}a[100007];
int c[100007]; ll sqr(ll x) { return x * x; } ll dist2(Point a, Point b) { return sqr(a.x - b.x) + sqr(a.y - b.y); } ll solve(int l, int r) {
if (l == r) return ~(1LL << 63);
if (l + 1 == r) return dist2(a[l], a[r]);
int mid = l + r >> 1;
ll ans = min(solve(l, mid), solve(mid + 1, r));
int cnt = 0;
for (int i = l;i <= r;i++) if (sqr(a[i].x - a[mid].x) < ans) c[cnt++] = i;
sort(c, c + cnt, [&](int p, int q) {return a[p].y < a[q].y;});
for (int i = 0;i < cnt;i++) {
for (int j = i + 1;j < cnt;j++) {
if (sqr(a[c[i]].y - a[c[j]].y) >= ans) break;
ans = min(ans, dist2(a[i], a[j]));
}
}
return ans;
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n;
cin >> n;
for (int i = 1;i <= n;i++) a[i].x = i, cin >> a[i].y, a[i].y += a[i - 1].y;
cout << solve(1, n) << '\n';
return 0;
}

方法二

#include <bits/stdc++.h>
#define ll long long using namespace std; struct Point {
ll x, y;
}a[100007], b[100007];
ll c[100007]; ll sqr(ll x) { return x * x; } ll dist2(Point a, Point b) { return sqr(a.x - b.x) + sqr(a.y - b.y); } ll solve(int l, int r) {
if (l == r) return ~(1LL << 63);
if (l + 1 == r) return dist2(a[l], a[r]);
int mid = l + r >> 1;
ll midx = a[mid].x;
ll ans = min(solve(l, mid), solve(mid + 1, r));
merge(a + l, a + mid + 1, a + mid + 1, a + r + 1, b + l, [&](Point a, Point b) {return a.y < b.y;});
for (int i = l;i <= r;i++) a[i] = b[i];
int cnt = 0;
for (int i = l;i <= r;i++) if (sqr(a[i].x - midx) < ans) c[cnt++] = i;
for (int i = 0;i < cnt;i++) {
for (int j = i + 1;j < cnt;j++) {
if (sqr(a[i].y - a[j].y) >= ans) break;
ans = min(ans, dist2(a[i], a[j]));
}
}
return ans;
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n;
cin >> n;
for (int i = 1;i <= n;i++) a[i].x = i, cin >> a[i].y, a[i].y += a[i - 1].y;
cout << solve(1, n) << '\n';
return 0;
}

NC15052 求最值的更多相关文章

  1. ACM3 求最值

    /*2*2014.11.18*求最值*描述:给定N个整数(1<=N<=100),求出这N个数中的最大值,最小值.*输入:多组数据,第一行为一个整数N,第二行为N个不超过100的正整数,用空 ...

  2. [NOI2005]维修数列 Splay tree 区间反转,修改,求和,求最值

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1500 Description Input 输入文件的第1行包含两个数N和M,N表示初始时数 ...

  3. hdu4521-小明系列问题——小明序列(线段树区间求最值)

    题意:求最长上升序列的长度(LIS),但是要求相邻的两个数距离至少为d,数据范围较大,普通dp肯定TLE.线段树搞之就可以了,或者优化后的nlogn的dp. 代码为  线段树解法. #include ...

  4. javascript之求最值

    求最值: var selections = $("#deliveryGridSalesOrGoods").datagrid('getRows'); var costPrice = ...

  5. poj3264(线段树区间求最值)

    题目连接:http://poj.org/problem?id=3264 题意:给定Q(1<=Q<=200000)个数A1,A2,```,AQ,多次求任一区间Ai-Aj中最大数和最小数的差. ...

  6. Sql示例说明如何分组后求中间值--【叶子】

    原文:Sql示例说明如何分组后求中间值--[叶子] 这里所谓的分组后求中间值是个什么概念呢? 我举个例子来说明一下: 假设我们现在有下面这样一个表: type        name price -- ...

  7. hdu 1754 I Hate It(树状数组区间求最值)2007省赛集训队练习赛(6)_linle专场

    题意: 输入一行数字,查询第i个数到第j个数之间的最大值.可以修改其中的某个数的值. 输入: 包含多组输入数据. 每组输入首行两个整数n,m.表示共有n个数,m次操作. 接下来一行包含n个整数. 接下 ...

  8. C语言 · 求arccos值

    算法提高 7-2求arccos值   时间限制:10.0s   内存限制:256.0MB      问题描述 利用标准库中的cos(x)和fabs(x)函数实现arccos(x)函数,x取值范围是[- ...

  9. 位运算求最值 学习笔记 (待补充QAQ)

    没有什么前言?直接进入正题qwq 俩俩异或 求最值: 建trie树 O(n)枚举每个数找这个数的最值,每次反走就成,还可以剪枝一波(如果在某位已经小于ans显然可以直接return? void Ins ...

随机推荐

  1. python基础练习题(题目 求输入数字的平方,如果平方运算后小于 50 则退出)

    day32 --------------------------------------------------------------- 实例046:打破循环 题目 求输入数字的平方,如果平方运算后 ...

  2. python基础练习题(题目 猴子吃桃问题:猴子第一天摘下若干个桃子,当即吃了一半,还不瘾,又多吃了一个第二天早上又将剩下的桃子吃掉一半,又多吃了一个。以后每天早上都吃了前一天剩下的一半零一个。到第10天早上想再吃时,见只剩下一个桃子了。求第一天共摘了多少)

    day13 --------------------------------------------------------------- 实例021:猴子偷桃 题目 猴子吃桃问题:猴子第一天摘下若干 ...

  3. oracle split 以及 简单json解析存储过程

    BEGIN; 由于之前工作上需要在oracle中做split功能以及json格分解.然后经过一番google和优化整合,最后整理到一个存储过程包中,易于管理,代码如下: 1.包定义: CREATE O ...

  4. QGIS 3.14插件开发——Win10系统PyCharm开发环境搭建四步走

    前言:最近实习要求做一个QGIS插件,网上关于QGIS 3.14插件开发环境搭建的文档不多,而且也不算太全面.正好实习的时候写了一个文档,在这里给大家分享一下. 因为是Word转的Markdown,可 ...

  5. 『现学现忘』Git基础 — 13、Git的基础操作

    目录 1.Git最基础的使用方式 (1)初始化本地版本库 (2)查看文件的状态 (3)把文件添加到暂存区 (4)把暂存区的内容提交到本地版本库 2.总结本文用到的Git命令 1.Git最基础的使用方式 ...

  6. Azure Terraform(十一)Azure DevOps Pipeline 内的动态临时变量的使用

    思路浅析 在我们分析的 Azure Terraform 系列文中有介绍到关于 Terraform 的状态文件远程存储的问题,我们在  Azure DevOps Pipeline 的 Task Job ...

  7. 修改本地仓库Maven,设置镜像

    修改本地仓库的位置: 先建立 文件路径: 进入文件夹目录: 就可以看到一个配置文件夹 就在里面修改本地仓库的路径,设置阿里云镜像 添加本地阿里云镜像: 如下图: 内容如下: <mirror> ...

  8. Hadoop(三)通过C#/python实现Hadoop MapReduce

    MapReduce Hadoop中将数据切分成块存在HDFS不同的DataNode中,如果想汇总,按照常规想法就是,移动数据到统计程序:先把数据读取到一个程序中,再进行汇总. 但是HDFS存的数据量非 ...

  9. NLP教程(3) | 神经网络与反向传播

    作者:韩信子@ShowMeAI 教程地址:http://www.showmeai.tech/tutorials/36 本文地址:http://www.showmeai.tech/article-det ...

  10. 新鲜出炉:appium2.0+ 单点触控和多点触控新的解决方案

    在 appium2.0 之前,在移动端设备上的触屏操作,单手指触屏和多手指触屏分别是由 TouchAction 类,Multiaction 类实现的. 在 appium2.0 之后,这 2 个方法将会 ...