找数组的i,j(j>i)使得a[j] - a[i]的值最大(算法)

第一种方法:

用两重循环对每对点都试一下,然后取最大值即可,时间复杂度为O(n2)

#include 

#include

usingnamespacestd;

intmaxIndexDiff(inta[],intn)

{

intmaxDiff = -1;

for(inti =0; i <  n; ++i)

{

for(intj =n-1; j > i ; --j)

{

if(a[j]>a[i]) maxDiff = max(maxDiff,j-i);

}

}returnmaxDiff;

}

int

main()

{

inta[]={9,2,3,4,5,6,7,8,18,0};

intn =sizeof(a)/sizeof(a[0]);

intmaxDiff =maxIndexDiff(a,n);

cout<

}

第二种方法:

首先对数组按照高度排序从小到大排序,如果高度相等的话,按照索引从小到大排序。

此时只需要在右边找一个索引值j,在左边找一个索引值i,使j-i最大即可。

可以建立一个rightMax数组,记录下每个索引右边的最大值,注意从右边往左边扫描,计算右边最大值简单些。

然后将当前的rightMax与排序后原始索引作差取最大值即可。时间复杂度O(nlogn)

#include #include#includeusingnamespacestd;structnode{intindex;intheight;

node(intidx =0,inth =0):index(idx),height(h){}booloperator<(constnode& a)const{if(height!=a.height)returnheight

}

};intmaxIndexDiff(inta[],intn){

vectorb(n);for(inti =0; i < n ; ++ i) {b[i].index = i;b[i].height =a[i];}

sort(b.begin(),b.end());

vector rightMax(n,b[n-1].index);

rightMax[n-1]=b[n-1].index;for(inti = n -2; i>=0; --i){if(b[i].index > rightMax[i+1]) rightMax[i] =b[i].index;elserightMax[i]=rightMax[i+1];

}intmaxDiff = -1;for(inti =0; i< n ; ++i){

maxDiff= max(maxDiff,rightMax[i]-b[i].index);

}returnmaxDiff;

}intmain(){inta[]={9,2,3,4,5,6,7,8,18,0};intn =sizeof(a)/sizeof(a[0]);intmaxDiff =maxIndexDiff(a,n);

cout<

}

还有一种方法是利用二分查找,注意j-i的值必定在0~n-1之间(索引是从0开始的)

取中间一个值mid=(0+n-1)/2,

如果存在a[i+mid]>a[i],则必然j-i至少是mid,继续向上二分查找

否则,j-i不超过mid,则向下二分查找

#include #include#includeusingnamespacestd;boolexist(inta[],intn,intk){for(inti =0; i+k< n; ++i){if(a[i] < a[i+k])returntrue;

}returnfalse;

}intmaxIndexDiff(inta[],intn){intleft =0, right = n-1;while(left <=right){intmid = (left+right)/2;if(exist(a,n,mid)) left=mid+1;elseright = mid-1;

}returnright;

}intmain(){inta[]={9,2,3,4,5,6,7,8,18,0};intn =sizeof(a)/sizeof(a[0]);intmaxDiff =maxIndexDiff(a,n);

cout<

}

第三种方法:

从左向右扫描一遍,记录每个索引左边的最小值(包括自己),leftMin[0..n-1]

从右向左扫描一遍,记录每个索引右边的最大值(包括自己),rightMax[0..n-1]

要注意的是:

对于leftMin[i],其左边的leftMin[0..i-1]都大于等于leftMin[i],其右边的left[i+1..n-1]都小于leftMin[i]

对于rightMax[j],其左边的rightMax[0..j-1]都大于等于rightMax[j],其右边的rightMax[j+1..n-1]都小于rightMax[j]

对于leftMin[i] 和rightMax[j]

如果leftMin[i]leftMin,记录当前的j-i,使++j

否则leftMin[i]>=rightMax[j], 则i的左边肯定都比rightMax[j]大,要增大++i,

#include #include#include#includeusingnamespacestd;intmaxIndexDiff(inta[],intn){

vector leftMin(n,a[0]),rightMax(n,a[n-1]);for(inti =1; i < n; ++i ) leftMin[i] = min(a[i],leftMin[i-1]);for(inti = n-2; i>=0; -- i) rightMax[i] = max(a[i],rightMax[i+1]);inti =0, j=0, maxDiff = -1;while(i

maxDiff= max(maxDiff,j-i);

j++;

}else++i;

}returnmaxDiff;

}intmain(){inta[]={9,2,3,4,5,6,7,8,18,0};intn =sizeof(a)/sizeof(a[0]);intmaxDiff =maxIndexDiff(a,n);

cout<

}

其中数组a[n]是无序的,求a[j]-a[i]的最大值,且i

第一种方法:

从左往右求下标0到 k - 1 的最小值MIN

从右往左求 下标k到n -1 的最大值MAX

对于每个k都有一个MAX - MIN的值,最后求这个值的最大值即可。

例如数组:4 5 2 6 3 1

K:1 2 3 4 5

MIN: 4 4 2 2 2

MAX:6 6 6 3 1

MAX - MIN,最大的值为6 - 2 = 4, 即为结果

第二种方法:

令b[j] = a[j + 1] - a[j],

那么a[j] - a[i]=(a[i+1]-a[i])+(a[i+2]-a[i+1])+...+(a[j]-a[i-1])

= b[i] +b[i+1]+ ...+ b[j - 1],

即将问题转化成求一个数组子序列的最大值。这个过程的算法是有O(n)的算法的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容