文档章节

动态规划之合唱队形问题

一贱书生
 一贱书生
发布于 2016/11/22 10:21
字数 1126
阅读 47
收藏 0

问题描述:

  N 位同学站成一排,音乐老师要请其中的(N-K)位同学出列,而不改变其他同学的位置,使得剩下的K位同学排成合唱队形。合唱队形要求:设K位同学从左到右 依次编号为1,2…,K,他们的身高分别为T1,T2,…,TK,则他们的身高满足 T1<T2...<Ti>Ti+1>…>TK(1<=i<=K)。已知所有N位同学的身高,计算最少需要几位 同学出列,可以使得剩下的同学排成最长的合唱队形。 

问题分析:

  假设第i位同学为个子最高的同学,我们先对其左边的同学求最大上升子序列,再对其右边的同学求最大下降子序列,然后两者相加再减1(第i位同学被重复计算 了一次),便得到第i位同学为最高个时所能排成的最长合唱队形。如果我们对这N位同学都执行此操作,便可得到每位同学为最高个时所能排成的最长合唱队形, 选取其中最长的合唱队形作为最终的结果。  

 从 上述的分析可以看出,我们可以将问题分成互相不独立的子问题,只要得到子问题的最优解,便可得到整个问题的最优解。我们可以用一张表来记录所有已解决问题 的答案,从而避免了重复计算。这里的一个关键问题便是:如何得到第i位同学的最大上升子序列和最大下降子序列。假如这N位同学的身高分别 为:176,163,150,180,170,130,167,160,我们用up[i]来记录第i位同学的最大上升子序列。如果要得到180同学为最高 个时的最大上升子序列即up[4],我们只需求出前3位同学所能形成的最大上升子序列,将其加1即可;要得到前3位同学所形成的最大上升子序列,便要求得 前2位同学的最大上升子序列,再加上1即可;同样要得到前2位同学的最大上升子序列,便要求得第1位同学的最大上升子序列。因此这是一个递推关系,只要我 们将前i个同学为最高个时做形成的最大上升子序列的值记录下来,取其中最大值加1便可得到第i位同学的最大上升子序列即up[i]。同理我们用 down[i]来记录第i位同学的最大下降子序列,只要我们将后(N-i)位同学每位同学为最高个时的最大下降子序列记录下来,取其中最大者再加1便可得 到第i位同学为最高个时的最大下降子序列即down[i]。那么,第i位同学为最高个时做能形成的最长合唱队形的长度为up[i]+down[i]-1。 求得所有同学为最高个时所能形成的最长合唱队形的长度,取其中最大值为最终的结果。

[java] view plain copy

 

  1. public class Main {  
  2.     public static void main(String[] args) {  
  3.         int []high={176,163,150,180,170,130,167,160};  
  4.         int []up=new int[8];   //记录每位同学的最大上升子序列  
  5.         int []down=new int[8]; //记录每位同学的最大下降子序列  
  6.         for(int i=0;i<high.length;i++){  
  7.             up[i]=1; //每位同学的最大上升子序列初始值为1  
  8.             for(int j=0;j<i;j++){  
  9.                 if((high[j]<high[i])&&(up[i]<up[j]+1)) up[i]=up[j]+1; //前i位同学的最大上升子序列的最大值再加1  
  10.             }  
  11.         }  
  12.         for(int i=high.length-1;i>=0;i--){  
  13.             down[i]=1;  
  14.             for(int j=high.length-1;j>i;j--){  
  15.                 if((high[j]<high[i])&&(down[i]<down[j]+1)) down[i]=down[j]+1; //后N-i位同学的最大下降子序列的最大值再加1  
  16.             }  
  17.         }  
  18.         int max=0; //设每位同学所形成的最长合唱队形的最大值初值为0  
  19.         int index=0; //设最大值对应的索引为0  
  20.         for(int i=0;i<high.length;i++){  
  21.             if(up[i]+down[i]-1>max) {  
  22.                 max=up[i]+down[i]-1; //求得每位同学所形成的最长合唱队形的最大值  
  23.                 index=i;  //求得对应的索引  
  24.             }  
  25.         }  
  26.         System.out.println("最长合唱队形的长度为:"+max);  
  27.         System.out.println("对应的是第"+(index+1)+"位同学,需要"+(high.length-max)+"位同学出列");  
  28.   
  29.     }  
  30. }

© 著作权归作者所有

共有 人打赏支持
一贱书生
粉丝 19
博文 724
码字总数 600123
作品 0
算法:Dynamic Programing

一、动态规划干嘛的 二、可以解决哪些问题 动态规划一般可分为:线性动规,区域动规,树形动规,背包动规四类。 线性动规:拦截导弹,合唱队形,挖地雷,建学校,剑客决斗等; 区域动规:石子...

猫咪要感冒
2016/10/09
1
0
技术解析:如何实现K歌App中的实时合唱

之前我们解析过很多社交直播App中不同场景的开发,比如在线K歌、小程序直播、多人视频聊天、AR等。 我们最近在知乎看到了一个问题「为什么k歌软件始终没有开发出实时合唱功能?」,我们只在知...

Agora
05/31
0
0
信息学奥赛一本通(C++版) 第二部分 基础算法 第九章 动态规划

信息学奥赛一本通(C++版) 第二部分 基础算法 第九章 动态规划 第一节 动态规划的基本模型 //1258 【例9.2】数字金字塔 //动态规划,自底向上计算,找出当前节点的最大值,一步一步推到顶点...

mrcrack
2017/11/03
0
0
算法设计与分析(二)之动态规划

第一篇: 算法设计与分析之分治思想 动态规划(Dynamic Programming) 求解过程是多阶段决策过程,每步处理一个字问题,可用于求解组合优化问题 适用条件:问题要满足优化原则或最优子结构性质...

wenhui12345
2017/11/12
0
0
白话版 动态规划法

关于动态规划法的解释, 大多都是基于背包问题的, 但背包问题背负了很多算法的解释工作,经常让初学者混淆,刚刚刷leetcode的时候,发现了一个很不错的关于动态规划算法的例题,特来分享一下! 这是...

木子昭
01/31
0
0

没有更多内容

加载失败,请刷新页面

加载更多

linux运维人员必会运维工具

linux运维人员必会开源运维工具体系 说明:不同的技术人员,不同的阶段确定知识边界非常重要,否则,就像马拉车,不知道终点在哪,累死也达不到目标。例如拿8K要学多少,拿15K要学多少。一个...

寰宇01
1分钟前
0
0
10大PHP比特币开源项目

如果你是一个Phper,如果你希望学习区块链,那么本文列出的 10个开源的Php比特币项目,将有助于你了解在自己的应用中 如何加入对比特币的支持。 如果你希望快速掌握使用Php对接比特币钱包的方...

汇智网教程
22分钟前
0
0
springclould feign客户端添加全局参数

用springclould feign作为调用服务的客户端,一般来说参数可以写在feignclient的方法参数里 有时需要所有feign请求都统一添加一些参数,例如token用于鉴权等,可以这样做: 添加一个配置类,...

canneljls
22分钟前
0
0
win32截屏并rgb24转yuv420

//最终f的内存布局为BGRA格式,需要保证buf长度足够(>w*h*4)void ScreenCap(void* buf, int w, int h){ HWND hDesk = GetDesktopWindow(); HDC hScreen = GetDC(hDesk); ......

styleman
今天
1
0
php输出mysql取出的中文为??的问题

解决方法: @ $db=new mysqli(DB_HOST,DB_USER,DB_PASSWORD,DB_DB); $db->query("set names utf8");//添加此语句,可以解决问题...

Aomo
今天
1
2

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部