【分类讨论】【解析几何】【 数学】【推荐】1330. 翻转子数组得到最大的数组值

本文主要是介绍【分类讨论】【解析几何】【 数学】【推荐】1330. 翻转子数组得到最大的数组值,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者推荐

视频算法专题

本文涉及知识点

分类讨论 解析几何

LeetCode1330. 翻转子数组得到最大的数组值

给你一个整数数组 nums 。「数组值」定义为所有满足 0 <= i < nums.length-1 的 |nums[i]-nums[i+1]| 的和。
你可以选择给定数组的任意子数组,并将该子数组翻转。但你只能执行这个操作 一次 。
请你找到可行的最大 数组值 。
示例 1:
输入:nums = [2,3,1,5,4]
输出:10
解释:通过翻转子数组 [3,1,5] ,数组变成 [2,5,1,3,4] ,数组值为 10 。
示例 2:
输入:nums = [2,4,9,24,2,1,10]
输出:68
提示:
1 <= nums.length <= 3*104
-105 <= nums[i] <= 105

分类讨论

n = nums.length
一,不翻转。
二,翻转nums[0,i2)。i2 ∈ \in [0,n)。没必要翻转两个数组。
三,翻转nums[i1,n)。i1 ∈ \in [0,n)。
四,翻转nums[i1,i2),i1不为0,i2不为n,i1和i2是合法下标,i1<i2。nums[i1-1]和nums[i2]都会发生变化。
翻转之前:|a-b|+|c-d| 翻转之后:|a-c|+|b-d|。
翻转后的数组值变大 ⟺ \iff f(a,b,c,d) = |a-c|+|b-d| -|a-b|+|c-d| > 0。
令这4个数,下标从小大到大为a(nums[i1-1]),b(nums[i1]),c,d。
两点简化:
简化一,只需要考虑a <= c。
简化二,只需要考虑a <= b。

简化一证明

某组数a1,b1,c1,d1 ,a1 < c1 。
某外一组数{c1,d1,a1,b1} = f(c1,d1,a1,b1)=|c1-a1|+|d1-b1| - |c1-d1|-[a1-b1|
第一项第二项取反:|a1-c1|+|b1-d1|
第三项第四项互换:-|a1-b1|-|c1-d1|。
故:f(c1,d1,a1,b1) = f(a1,b1,c1,d1)

简化二证明:

某组数a1,b1,c1,d1 a1 > b1。
另外一组数{b1,a1,d1,c1} , f(b1,a1,d1,c1)=|b1-d1|+[a1-c1| -|b1-a1|-|d1-c1|
交换第一项和第二项目:|a1-c1|+|b1-d1|
第三项、第四项目,取反:-|a1-b1| - |c1,d1|
故:f(b1,a1,d1,c1) = f(a1,b1,c1,d1)。

等效问题

有一维四点a,b,c,d。 a <=b,a<=c。
线段ac的长度+线段bd的长度,能否大于线段ab的长度+线段cd的长度。
一,假定 b < c。
在这里插入图片描述
第一二种情况,两者有重合部分,长度不变。
第三四中情况,两者无重合部分,长度增加 两条线段的间隙 × \times × 2
二,b > c。
在这里插入图片描述
全部变短或不变,全部部分重合或全部重合。

结论

a<b a<c ,两条线段有重叠部分,则变短或不变。 没有重叠部分,变长间隙长度 × \times × 2。
上文已经证明了 两种简化的结果一样,现在来证明两个简化的判断条件一致。

简化一证明

某组数a1,b1,c1,d1 ,a1 < c1 。
简化前:线段a1b1和c1d1是否重叠。
简化后:线程c1d1和a1b1是否重叠。
两种显然等效。

简化二证明:

某组数a1,b1,c1,d1 a1 > b1。
线段a1b1就是b1a1,c1d1就是d1c1,所以无需证明。

重新简化

a<b 且b < c 且c<d。
ac的长度+bd的长度== ad的长度+bc的长度,都是ad的长度+bc的长度。
故调整a、b,使得a<b。调整c、d,使得c<d。
调整后增加的长度就是:(c-b)*2,结果为负数忽略。

代码

template<class ELE,class ELE2>
void MinSelf(ELE* seft, const ELE2& other)
{*seft = min(*seft,(ELE) other);
}template<class ELE>
void MaxSelf(ELE* seft, const ELE& other)
{*seft = max(*seft, other);
}#define MacEnumMask(mask,maskMax) for (int mask = maskMax; mask; mask = (mask - 1) & maskMax) class Solution {
public:int maxValueAfterReverse(vector<int>& nums) {m_c = nums.size();long long ans = 0;for (int i = 0; i + 1 < m_c; i++){ans += abs(nums[i] - nums[i + 1]);}int iAdd = 0;for (int i = 1; i < m_c; i++){//交换nums[i,m_c)MaxSelf(&iAdd, abs(nums.back() - nums[i - 1]) - abs(nums[i] - nums[i - 1]));}for (int i = 1; i < m_c; i++){//交换nums[0,i)MaxSelf(&iAdd, abs(nums.front() - nums[i]) - abs(nums[i] - nums[i - 1]));}{//a,b在前 ,c,d在后int  b = max(nums[0], nums[1]);for (int j = 2; j < m_c; j++){int c = min(nums[j - 1], nums[j]);int d = max(nums[j - 1], nums[j]);MaxSelf(&iAdd, 2 * c - 2 * b);MinSelf(&b, d);}}{//c,d在前 ,a,b在后int b = max(nums[m_c-1], nums[m_c-2]);for (int j = m_c-2; j >0 ; j--){int c = min(nums[j - 1], nums[j]);int d = max(nums[j - 1], nums[j]);MaxSelf(&iAdd, 2 * c - 2 * b);MinSelf(&b, d);}}return ans + iAdd;}int m_c;
};

测试用例

template<class T,class T2>
void Assert(const T& t1, const T2& t2)
{assert(t1 == t2);
}template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{if (v1.size() != v2.size()){assert(false);return;}for (int i = 0; i < v1.size(); i++){Assert(v1[i], v2[i]);}}int main()
{vector<int> nums;{Solution sln;nums = { 2,4,9,24,2,1,10 };auto res = sln.maxValueAfterReverse(nums);Assert(68, res);}{Solution sln;nums = { 2, 3, 1, 5, 4 };auto res = sln.maxValueAfterReverse(nums);Assert(10, res);}}

2023年5月

class Solution {
public:
int maxValueAfterReverse(vector& nums) {
m_c = nums.size();
int iTotal = 0;
for (int i = 0; i + 1 < m_c; i++)
{
iTotal += abs(nums[i] - nums[i + 1]);
}
//长度为1的子数组,结构不变。翻转整个数组结果不变
int iMaxAdd = 0;
//处理以索引0开头或m_c-1结尾的子数组
for (int i = 0; i + 1 < m_c; i++)
{
//翻转[0,i]
const int iLeft = abs(nums[i + 1] - nums[0]) - abs(nums[i + 1] - nums[i]);
//翻转[i+1,m_c)
const int iRight = abs(nums.back() - nums[i]) - abs(nums[i + 1] - nums[i]);
iMaxAdd = max(iMaxAdd, iLeft);
iMaxAdd = max(iMaxAdd, iRight);
}
int iMaxC = INT_MIN, iMinB = INT_MAX;
for (int i = 0; i + 1 < m_c; i++)
{
const int& x = nums[i];
const int& y = nums[i + 1];
iMaxC = max(iMaxC, min(x, y));
iMinB = min(iMinB, max(x, y));
}
iMaxAdd = max(iMaxAdd, 2 * (iMaxC - iMinB));
return iTotal + iMaxAdd;
}
int m_c;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关

下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

这篇关于【分类讨论】【解析几何】【 数学】【推荐】1330. 翻转子数组得到最大的数组值的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

uva 10014 Simple calculations(数学推导)

直接按照题意来推导最后的结果就行了。 开始的时候只做到了第一个推导,第二次没有继续下去。 代码: #include<stdio.h>int main(){int T, n, i;double a, aa, sum, temp, ans;scanf("%d", &T);while(T--){scanf("%d", &n);scanf("%lf", &first);scanf

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

uva 10025 The ? 1 ? 2 ? ... ? n = k problem(数学)

题意是    ?  1  ?  2  ?  ...  ?  n = k 式子中给k,? 处可以填 + 也可以填 - ,问最小满足条件的n。 e.g k = 12  - 1 + 2 + 3 + 4 + 5 + 6 - 7 = 12 with n = 7。 先给证明,令 S(n) = 1 + 2 + 3 + 4 + 5 + .... + n 暴搜n,搜出当 S(n) >=

uva 11044 Searching for Nessy(小学数学)

题意是给出一个n*m的格子,求出里面有多少个不重合的九宫格。 (rows / 3) * (columns / 3) K.o 代码: #include <stdio.h>int main(){int ncase;scanf("%d", &ncase);while (ncase--){int rows, columns;scanf("%d%d", &rows, &col

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言