How to solve this HARD issue

1. Problem:

A non-empty zero-indexed array A consisting of N integers is given.

peak is an array element which is larger than its neighbours. More precisely, it is an index P such that 0 < P < N − 1 and A[P − 1] < A[P] > A[P + 1].

For example, the following array A:

    A[0] = 1
A[1] = 5
A[2] = 3
A[3] = 4
A[4] = 3
A[5] = 4
A[6] = 1
A[7] = 2
A[8] = 3
A[9] = 4
A[10] = 6
A[11] = 2

has exactly four peaks: elements 1, 3, 5 and 10.

You are going on a trip to a range of mountains whose relative heights are represented by array A, as shown in a figure below. You have to choose how many flags you should take with you. The goal is to set the maximum number of flags on the peaks, according to certain rules.

Flags can only be set on peaks. What's more, if you take K flags, then the distance between any two flags should be greater than or equal to K. The distance between indices P and Q is the absolute value |P − Q|.

For example, given the mountain range represented by array A, above, with N = 12, if you take:

  • two flags, you can set them on peaks 1 and 5;
  • three flags, you can set them on peaks 1, 5 and 10;
  • four flags, you can set only three flags, on peaks 1, 5 and 10.

You can therefore set a maximum of three flags in this case.

Write a function:

int solution(int A[], int N);

that, given a non-empty zero-indexed array A of N integers, returns the maximum number of flags that can be set on the peaks of the array.

For example, the following array A:

    A[0] = 1
A[1] = 5
A[2] = 3
A[3] = 4
A[4] = 3
A[5] = 4
A[6] = 1
A[7] = 2
A[8] = 3
A[9] = 4
A[10] = 6
A[11] = 2

the function should return 3, as explained above.

Assume that:

  • N is an integer within the range [1..200,000];
  • each element of array A is an integer within the range [0..1,000,000,000].

Complexity:

  • expected worst-case time complexity is O(N);
  • expected worst-case space complexity is O(N), beyond input storage (not counting the storage required for input arguments).

Elements of input arrays can be modified.

Copyright 2009–2015 by Codility Limited. All Rights Reserved. Unauthorized copying, publication or disclosure prohibited.
 
2. Dig into this issue
    最初我认为这个问题非常复杂,需要牵涉到例如两点间最小距离啊,删除节点啊,对peak间的间距进行排序啊等等问题。
    不过在我看到时间复杂度要求为O(N)时,我觉得我还是想多了。
 
    通过数学我们可以得出,如果dis = 最后一个peak与第一个peak之间的距离,那么(k-1)*k<dis;
    于是我们就可以算出最大可能的k,而k与dis成√的关系,于是如果按照k的scale进行循环,那么我们就成功的减少了运算的时间复杂度。
 
    对于每一个k来说,最容易实现的方法肯定是从最左边的peak向右查找(因为如果从下一个开始,dis减少,k也会变少)。所以我们从最左边peak开始
   ,加上一个k,得到下一步,在这个点上寻找下一个peak,在依次向下搜索,直到找到的peak数等于k,完成,或者找不到下一个peak了(k过大),那么k-1,重复
   上层操作。
    
     这个操作在时间复杂度上还有一个技术障碍,就是在i位置寻找下一个peak需要O(N)级别的操作,我们如何将它变为O(1)级别的操作呢?
     用一个数组即可。
     我们首先遍历这个A,找出所有peak所在位置。然后定义一个数组nextpeak[],对于nextpeak[i],代表从i位置往后(包括i位置),所找到的第一个peak。
     通过O(N)的空间,我们就可以将找nextpeak的操作降为O(1)级别。
 
     这样通过我们上述的循环算法,总能找到k解。而且通过具体的验证,这个算法的时间复杂度是O(N)级别的。
 3.结果:
    
 
4.源代码为:

// you can write to stdout for debugging purposes, e.g.
// printf("this is a debug message\n"); int solution(int A[], int N) {
// write your code in C99
int i = ;
// 每一个节点是否为peak
int isPeak[N];
isPeak[]=;
isPeak[N-]=;
// peak个数
int count = ;
for(i=;i<N-;i++)
{
if(A[i]>A[i-]&&A[i]>A[i+])
{
isPeak[i]=;
count++;
}
else
{
isPeak[i]=;
}
}
//如果peak为0,那么直接退出没商量
if(count == )
{
return ;
}
//放入相应peak的位置。
int peak[count]; int j=;
for(i=;i<N;i++)
{
if(isPeak[i]==)
{
peak[j]=i;
j++;
}
} int dis = peak[count-]-peak[]; //最大可能k
int maxk =;
while((maxk-)*maxk<dis)
{
maxk++;
}
if((maxk-)*maxk!=dis)
maxk--; // 存入在i节点处下一个peak的位置,如果不存在下一个peak,为-1;
int nextpeak[N]; j=count-;
int temp = -;
for(i=N-;i>;i--)
{
if(i>peak[j])
{
nextpeak[i]=temp;
}
else
{
temp = peak[j];
j--;
nextpeak[i]=temp;
}
// printf("%d ",nextpeak[i]);
} //从 maxk,向下搜索,直到找出一个i(k)满足条件
int start = peak[];
int nodes = ;
for(i=maxk;i>;i--)
{
while(nodes<i)
{
start = start+i;
if(start > N-)
{
break;
}
start = nextpeak[start];
// printf("\n%d ",start);
if(start == -)
{
break;
}
else
{
nodes++;
}
}
if(nodes == i)
{
return i;
}
else
{
nodes = ;
start = peak[];
}
} return nodes;
}

codility flags solution的更多相关文章

  1. Codility NumberSolitaire Solution

    1.题目: A game for one player is played on a board consisting of N consecutive squares, numbered from ...

  2. Solution of NumberOfDiscIntersections by Codility

    question:https://codility.com/programmers/lessons/4 this question is seem like line intersections qu ...

  3. Solution to Triangle by Codility

    question: https://codility.com/programmers/lessons/4 we need two parts to prove our solution. on one ...

  4. the solution of CountNonDivisible by Codility

    question:https://codility.com/programmers/lessons/9 To solve this question , I get each element's di ...

  5. GenomicRangeQuery /codility/ preFix sums

    首先上题目: A DNA sequence can be represented as a string consisting of the letters A, C, G and T, which ...

  6. *[codility]Peaks

    https://codility.com/demo/take-sample-test/peaks http://blog.csdn.net/caopengcs/article/details/1749 ...

  7. *[codility]Country network

    https://codility.com/programmers/challenges/fluorum2014 http://www.51nod.com/onlineJudge/questionCod ...

  8. *[codility]AscendingPaths

    https://codility.com/programmers/challenges/magnesium2014 图形上的DP,先按照路径长度排序,然后依次遍历,状态是使用到当前路径为止的情况:每个 ...

  9. *[codility]MaxDoubleSliceSum

    https://codility.com/demo/take-sample-test/max_double_slice_sum 两个最大子段和相拼接,从前和从后都扫一遍.注意其中一段可以为0.还有最后 ...

随机推荐

  1. linux命令-文件命令

    1.解压.tar文件 tar -vxf *.tar 2.把一个文件夹下的内容复制到另一个文件夹 将aaa内所有内容复制到bbb cp -a aaa/* /bbb/  * 3.复制文件时不改变文件的时间 ...

  2. Linux服务器安全登录设置记录

    在日常运维工作中,对加固服务器的安全设置是一个机器重要的环境.比较推荐的做法是:1)严格限制ssh登陆(参考:Linux系统下的ssh使用(依据个人经验总结)):     修改ssh默认监听端口    ...

  3. js变量问题

    this指向问题,谁调用它,它就指谁!!! 1.var foo = 1; function bar() { foo = 10; return; function foo() {} } bar(); a ...

  4. 商城项目:装nginx时碰到的各种问题

    因为项目需要,我们要在linux上nginx.碰到了各种问题.在这里一一记录下来. 首先我要开启我的两个虚拟机,开起来之后.用主机的SeureCRT去连接.都是好的. 但是我在虚拟机机上去ping I ...

  5. 微信H5中的一些坑

    最近在写微信公众号H5页面 遇到了一些坑,在这里记录一下 记录一下signature的计算 // 首先找到hex_sha1的加密算法,ticket 是后端提供的 var url_local = loc ...

  6. [LeetCode] Largest Rectangle in Histogram 直方图中最大的矩形

    Given n non-negative integers representing the histogram's bar height where the width of each bar is ...

  7. [LeetCode] Longest Common Prefix 最长共同前缀

    Write a function to find the longest common prefix string amongst an array of strings. 这道题让我们求一系列字符串 ...

  8. 【MySQL】花10分钟阅读下MySQL数据库优化总结

    1.花10分钟阅读下MySQL数据库优化总结http://www.kuqin.com2.扩展阅读:数据库三范式http://www.cnblogs.com3.my.ini--->C:\Progr ...

  9. Android----消息弹出框

    关于Android的知识,自从工作了就没有什么时间去总结学习过的知识,我个人比较喜欢学习后总结,今天就写一下关于android中消息弹出框的几种方式的简单示例,按照自己的思路写了一段,希望对和我一样在 ...

  10. 通过一个实例重新认识引用类型,值类型,数组,堆栈,ref

    昨天在写代码时候遇到了一个问题,百思不得其解,感觉颠覆了自己对C#基础知识的认知,因为具体的情境涉及公司代码不便放出,我在这里举个例子,先上整个测试所有的代码,然后一一讲解我的思考过程: using ...