求序列最长不下降子序列_最长不下降子序列nlogn算法详解

今天花了很长时间终于弄懂了这个算法……毕竟找一个好的讲解真的太难了,所以励志我要自己写一个好的讲解QAQ

这篇文章是在懂了这个问题n^2解决方案的基础上学习。

解决的问题:给定一个序列,求最长不下降子序列的长度(nlogn的算法没法求出具体的序列是什么)

定义:a[1..n]为原始序列,d[k]表示长度为k的不下降子序列末尾元素的最小值,len表示当前已知的最长子序列的长度。

初始化:d[1]=a[1]; len=1; (0个元素的时候特判一下)

现在我们已知最长的不下降子序列长度为1,末尾元素的最小值为a[1],那么我们让i从2到n循环,依次求出前i个元素的最长不下降子序列的长度,循环的时候我们只需要维护好d这个数组还有len就可以了。

关键问题就是怎么维护?

可以看出我们是要用logn的复杂度维护的。实际上利用了d数组的一个性质:单调性。(长度更长了,d[k]的值是不会减小的)

考虑新进来一个元素a[i]:

如果这个元素大于等于d[len],直接让d[len+1]=a[i],然后len++。这个很好理解,当前最长的长度变成了len+1,而且d数组也添加了一个元素。

如果这个元素小于d[len]呢?说明它不能接在最后一个后面了。那我们就看一下它该接在谁后面。

准确的说,并不是接在谁后面。而是替换掉谁。因为它接在前面的谁后面都是没有意义的,再接也超不过最长的len,所以是替换掉别人。那么替换掉谁呢?就是替换掉那个最该被它替换的那个。也就是在d数组中第一个大于它的。第一个意味着前面的都小于等于它。假设第一个大于它的是d[j],说明d[1..j-1]都小于等于它,那么它完全可以接上d[j-1]然后生成一个长度为j的不下降子序列,而且这个子序列比当前的d[j]这个子序列更有潜力(因为这个数比d[j]小)。所以就替换掉它就行了,也就是d[j]=a[i]。其实这个位置也是它唯一能够替换的位置(前面的替了不满足d[k]最小值的定义,后面替换了不满足不下降序列)

至于第一个大于它的怎么找……STL upper_bound。每次复杂度logn。

至此,我们就神奇的解决了这个问题。按照这个思路,如果需要求严格递增的子序列怎么办?

仍然考虑新进来一个元素a[i]:

如果这个元素大于d[len],直接让d[len+1]=a[i],然后len++。这个很好理解,当前最长的长度变成了len+1,而且d数组也添加了一个元素。

如果这个元素小于等于d[len]呢?说明它不能接在最后一个后面了。那我们就看一下它该接在谁后面。

同样的道理,只是换成lower_bound即可。每次复杂度logn。

--------2018.4.14更新--------

很久没看,没想到这篇文章有这么多人阅读了,感觉最长递增子序列这里讲的不是太清楚,因此补充一下。

最长递增子序列和最长不下降子序列的不同之处在于,d数组的单调性更严格了:一定是单调递增的。

可以用反证法来证明这一点:假设有d[i]=d[i+1],也就是长度为i+1的子序列最后一位最小是d[i+1],那假设这个子序列是a[1], a[2], ..., a[i+1],在这个子序列里面,a[i]

那这个性质有什么意义呢?

仍然考虑新进来一个元素a[i]:

如果这个元素大于d[len],直接让d[len+1]=a[i],然后len++。这个很好理解,当前最长的长度变成了len+1,而且d数组也添加了一个元素。

如果这个元素等于d[len],那么可以保证d[1..len-1]都是小于a[i]的(根据上面的证明),因此这个元素就没有什么意义了,直接忽略就好,因为它无法接在任何一个元素d后面产生一个更有优势的子序列。

如果这个元素小于d[len],那么就在d数组中找到第一个大于等于它的元素(这个元素必然存在,至少d[len]就是),把这个元素替换成a[i]即可。

实际上可以发现,小于等于的时候可以统一按照lower_bound替换的方式来处理。

这样做肯定是对的,而邝斌的模板上实际上是求的最长不下降子序列,而不是最长上升子序列。不信可以测试一下"1 2 3 2 3 2"这个样例。切记,不要迷信权威,学会自己思考。

------------------------------------

下面是最长不下降子序列的代码,注释里面说明了如何改成最长上升子序列。

//最长不下降子序列nlogn Song

#include#include

using namespacestd;int a[40005];int d[40005];intmain()

{intn;

scanf("%d",&n);for (int i=1;i<=n;i++) scanf("%d",&a[i]);if (n==0) //0个元素特判一下

{

printf("0\n");return 0;

}

d[1]=a[1]; //初始化

int len=1;for (int i=2;i<=n;i++)

{if (a[i]>=d[len]) d[++len]=a[i]; //如果可以接在len后面就接上,如果是最长上升子序列,这里变成>

else //否则就找一个最该替换的替换掉

{int j=upper_bound(d+1,d+len+1,a[i])-d; //找到第一个大于它的d的下标,如果是最长上升子序列,这里变成lower_bound

d[j]=a[i];

}

}

printf("%d\n",len);return 0;

}

想了一晚上这个问题终于想通了。前面说的“最该替换的位置”实际上不是很精确,那个位置替换掉是它唯一能够替换的位置,之所以要替换,就是为了维护d这个数组,让它始终满足最初的定义。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/431334.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

cmap参数 plt_plt.imshow的参数有哪些?

plt.imshow( X, cmap=None, norm=None, aspect=None, interpolation=None, alpha=None, vmin=None, vmax=None, origin=None, extent=None, shape=None, filternorm=1, filterrad=4.0, imlim=None, resample=None, url=None, *, data=None, **kwargs, ) 参数:X图像数据。支持的…

安装Linux后windows在哪启动,在Linux下安装windows后解决Linux不能启动能问题

我的机器是Windows和Linux双系统&#xff0c;那天由于Windows用就了比较卡&#xff0c;就重装了一下Windows&#xff0c;结果导致重启后linux系统不知道到哪去了。懒得去重装Linux了&#xff0c;于是查找了一下资料&#xff0c;经过一番折腾&#xff0c;终于把我的Linux救回来了…

asp.net web开发步骤_WEB开发流程

1.需求确定2.分析与设计 (1)架构分析与设计 (2)业务逻辑分析 (3)业务逻辑设计 (4)界面设计3.开发环境搭建4.开发-测试-开发-测试5.文档编纂各个阶段过程都是有文档的,需求有需求的文档,设计有设计的文档,测试也有测试的文档.首先来看需求确定:需求…

Dart基础学习02--变量及内置类型

Dart基础学习02--变量及内置类型 Dart中的变量 首先看一个变量的定义和赋值 var name Bob; 在Dart中变量名都是引用&#xff0c;这里的name就是一个指向值为Bob的字符串的引用。 默认值 在Dart中没有初始化的变量都有一个默认值null&#xff0c;哪怕是int型的变量它的初始值也…

课题申报书范文_课题申报书(范文5篇)

《课题申报书》课题申报书(一)&#xff1a;一、课题研究论证报告1.课题提出的背景与所要解决的主要问题(1)课题提出的背景第一、新的教材、新的课程越来越注重了学生学习主动性、用心性的调动。它的最大特点就是不是仅注重知识研究的结果&#xff0c;而是更重视研究知识的过程&…

psv黑商店pkgj最新版下载_e收银app下载安装_e收银软件最新版免费下载

e-Cashier应用程序是一种高效的收银员服务客户端应用程序。 e-Cashier的移动版本支持多种刷卡方法&#xff0c;使商户可以自定义订单日期&#xff0c;从而使每天的收入一目了然&#xff0c;并且从多个角度分析商店的业务状况非常实用 。特征电子收银机是一种高效便捷的收银工具…

linux vi使用手册,史上最全VIM使用手册

vim是一个向上兼容于vi的文本编辑器&#xff0c;可用于编辑各种纯文本。它对编辑程序特别有用。基本模式&#xff1a;编辑模式&#xff1a;默认编辑文件进入的模式输入模式&#xff1a;末行模式&#xff1a;内置的命令行接口基本使用&#xff1a;# vim [options] [file ..]#&am…

shell判断false_Linux脚本shell常用判断式

在shell脚本编程中&#xff0c;判断式是经常被使用的。判断式的使用有两种方式&#xff0c;其中一种是使用test&#xff0c;另一种是使用 []。下面我们通过两个简单的例子来看一下这两种方式如何使用。例一# test –e /tmp/a.txt ; echo $?结果为0我们看到&#xff0c;上面的结…

微分方程的数值解法与程序实现 pdf_初中数学知识点|一元一次方程的概念及讲解(二)建议收藏!内含pdf版...

长按关注【初中数学解题大全】 获取更多&#xff01;电子课本|学习资料|学习方法|家教知识后台回复“数学”等学科名称即可获取初中数学电子课本后台回复“初一”等相应年级即可获取数学视频讲解及知识点大全后台回复“真题”即可获取2017-2019初中数学各市真题试卷大全后台回复…

Linux下rgmii接口与fpga相连,FPGA控制RGMII接口PHY芯片88E1512网络通信

一、前言网络通信中的PHY芯片接口种类有很多&#xff0c;之前接触过GMII接口的PHY芯片RTL8211EG。但GMII接口数量较多&#xff0c;本文使用RGMII接口的88E1512搭建网络通信系统。这类接口总线位宽小&#xff0c;可以降低电路成本&#xff0c;在实际项目中应用更广泛。二、从GMI…

面试题12:打印1到最大的n位数

题目描述 输入数字n&#xff0c;按顺序打印出1到最大的n位十进制数。比如输入3&#xff0c;则打印出1、2、3一直到最大的3位数即999。 题目分析 剑指Offer&#xff08;纪念版&#xff09;P94 代码实现 void Print1ToMaxOfNDigits_2(int n) {if(n < 0)return;char* number n…

tp3 普通模式url模式_[tp3.2.1]开启URL(重写模式),省略URL中的index.php

重写模式(省略url中的index.php)在apache配置文件httpd.conf中,查找1.mod_rewrite.so, 启动此模块2.AllowOverride , 值 All3. 把下面的内容保存为.htaccess文件放到应用入口文件的同级目录下Options FollowSymlinksRewriteEngine onRewriteCond %{REQUEST_FILENAME} !-dRewrit…

esxi usb插口_酷暑大作战 | USB-C风扇新体验

“枯藤老树昏鸦&#xff0c;空调 Wi-Fi 西瓜”每到炎热的夏天&#xff0c;人类的梦想就变得朴实、无华而简单但当我们吹不到空调时&#xff0c;如何还能享受阵阵清凉&#xff1f;今天我们介绍的两款新型电扇或许可以提供新思路&#xff1a;edon 循环电扇GUZI 挂颈电扇越来越多人…

删除linux内核多余架构,删除多余Linux内核方法

我使用的是Linux Mint&#xff0c;更新频繁&#xff0c;旧版本的Linux内核只会浪费硬盘空间&#xff0c;因此我会定期删除多余的内核。查看已安装所有内核&#xff1a;sudo dpkg --get-selections |grep linux-image查看当前正在使用的内核&#xff1a;uname -r删除不需要的内核…

数据库连接

http://www.cnblogs.com/JamyWong/archive/2009/06/03/1495499.html 转载于:https://www.cnblogs.com/wqy61/p/4835265.html

arduino 光控灯_Arduino光控开关

扩展实验&#xff0c;使用光敏电阻同样可以达到类似电位计调节LED亮度的效果&#xff0c;如实验电位计控制LED亮度&#xff1a;http://jingyan.baidu.com/article/49711c61785102fa441b7c81.html&#xff0c;只需要将代码修改如下为所示代码&#xff0c;并正确上传执行即可&…

手写tomcat socket closed_【消费电子】:重在阅读,新增手写 BOOX Nova Pro 体验评测...

此文出自于《消费电子》对BOOX Nova Pro的评测&#xff0c;本文已征得李强作者授权转载。【《消费电子》杂志社】越来越多的人选择在移动设备上阅读、学习&#xff0c;正所谓“需求诞生市场”&#xff0c;如今这类产品已经相当之丰富。越来越多的人选择在移动设备上阅读、学习&…

Linux fwrite 什么时候刷新,linux的fwrite()使用方法,当前时间写入文本的程序

1.函数功能用来读写一个数据块。2.一般调用形式fwrite(buffer,size,count,fp);3.说明(1)buffer&#xff1a;是一个指针&#xff0c;对fread来说&#xff0c;它是读入数据的存放地址。对fwrite来说&#xff0c;是要输出数据的地址。(2)size&#xff1a;要读写的字节数&#xff1…

BFS HDOJ 1242 Rescue

题目传送门 题意&#xff1a;从r走到a&#xff0c;遇到x多走一步&#xff0c;问最小走到a的步数 分析&#xff1a;因为r有多个&#xff0c;反过来想从a走到某个r的最小步数&#xff0c;简单的BFS。我对这题有特殊的感情&#xff0c;去年刚来集训队时肉鸽推荐了这题&#xff0c;…

mysql insert 不需要日志_详解MySQL|你不知道的新特性-8.0错误日志增强

MySQL 8.0 重新定义了错误日志输出和过滤&#xff0c;改善了原来臃肿并且可读性很差的错误日志。比如增加了 JSON 输出&#xff0c;在原来的日志后面以序号以及 JSON 后缀的方式展示。比如我机器上的 MySQL 以 JSON 保存的错误日志 mysqld.log.00.json&#xff1a;[rootcentos-…