第一问DP水过。dp[i]代表以i结尾的最长不下降子序列长度。

二三问网络流。

第二问是说每个子序列不能重复使用某个数字。

把每个点拆成p(i),q(i)。连边。

要是dp[i]=1,连源,p(i)

要是dp[i]=s,连q(i),汇

要是i<j && num[i]<=num[j] && dp[i]+1==dp[j],连q(i),p(j)。

上述各边容量均为1。

为什么呢?

这实际上是建立分层图的思想,每一层里dp[i]都不一样,那么从源走到汇的路径必定为dp[i]递增的合法序列。

求最大流

第三问加上几条容量INF的边

求最大流

  1. #include <iostream>
  2. #include <cstring>
  3. #include <cstdio>
  4. #include <queue>
  5. using namespace std;
  6. int n, a[505], dp[505], cnt, hea[1005], s, ss, tt, maxFlow, lev[1005];
  7. const int oo=0x3f3f3f3f;
  8. queue<int> d;
  9. struct Edge{
  10. int too, nxt, val;
  11. }edge[60005];
  12. inline int p(int x){
  13. return x;
  14. }
  15. inline int q(int x){
  16. return x+n;
  17. }
  18. void add_edge(int fro, int too, int val){
  19. edge[cnt].nxt = hea[fro];
  20. edge[cnt].too = too;
  21. edge[cnt].val = val;
  22. hea[fro] = cnt++;
  23. }
  24. void addEdge(int fro, int too, int val){
  25. add_edge(fro, too, val);
  26. add_edge(too, fro, 0);
  27. }
  28. bool bfs(){
  29. memset(lev, 0, sizeof(lev));
  30. d.push(ss);
  31. lev[ss] = 1;
  32. while(!d.empty()){
  33. int x=d.front();
  34. d.pop();
  35. for(int i=hea[x]; i!=-1; i=edge[i].nxt){
  36. int t=edge[i].too;
  37. if(!lev[t] && edge[i].val>0){
  38. d.push(t);
  39. lev[t] = lev[x] + 1;
  40. }
  41. }
  42. }
  43. return lev[tt]!=0;
  44. }
  45. int dfs(int x, int lim){
  46. if(x==tt) return lim;
  47. int addFlow=0;
  48. for(int i=hea[x]; i!=-1 && addFlow<lim; i=edge[i].nxt){
  49. int t=edge[i].too;
  50. if(lev[t]==lev[x]+1 && edge[i].val>0){
  51. int tmp=dfs(t, min(lim-addFlow, edge[i].val));
  52. edge[i].val -= tmp;
  53. edge[i^1].val += tmp;
  54. addFlow += tmp;
  55. }
  56. }
  57. return addFlow;
  58. }
  59. void dinic(){
  60. while(bfs()) maxFlow += dfs(ss, oo);
  61. }
  62. int main(){
  63. memset(hea, -1, sizeof(hea));
  64. cin>>n;
  65. ss = 0; tt = n * 2 + 1;
  66. for(int i=1; i<=n; i++){
  67. scanf("%d", &a[i]);
  68. dp[i] = 1;
  69. }
  70. for(int i=1; i<=n; i++)
  71. for(int j=1; j<i; j++)
  72. if(a[j]<=a[i])
  73. dp[i] = max(dp[i], dp[j]+1);
  74. for(int i=1; i<=n; i++)
  75. s = max(s, dp[i]);
  76. cout<<s<<endl;
  77. for(int i=1; i<=n; i++){
  78. addEdge(p(i), q(i), 1);
  79. if(dp[i]==1) addEdge(ss, p(i), 1);
  80. if(dp[i]==s) addEdge(q(i), tt, 1);
  81. for(int j=1; j<i; j++)
  82. if(a[j]<=a[i] && dp[j]+1==dp[i])
  83. addEdge(q(j), p(i), 1);
  84. }
  85. dinic();
  86. cout<<maxFlow<<endl;
  87. addEdge(p(1), q(1), oo);
  88. addEdge(p(n), q(n), oo);
  89. addEdge(ss, p(1), oo);
  90. if(dp[n]==s) addEdge(q(n), tt, oo);
  91. dinic();
  92. cout<<maxFlow<<endl;
  93. return 0;
  94. }

luogu2766 最长不下降子序列问题的更多相关文章

  1. luogu2766 最长不下降子序列问题 DP 网络流

    题目大意:给定正整数序列x1,...,xn .(1)计算其最长不下降子序列的长度s.(不一定是否连续)(2)计算从给定的序列中最多可取出多少个长度为s的不下降子序列.(序列内每一个元素不可重复)(3) ...

  2. 最长不下降子序列(LIS)

    最长上升子序列.最长不下降子序列,解法差不多,就一点等于不等于的差别,我这里说最长不下降子序列的. 有两种解法. 一种是DP,很容易想到,就这样: REP(i,n) { f[i]=; FOR(j,,i ...

  3. 最长不下降子序列 O(nlogn) || 记忆化搜索

    #include<stdio.h> ] , temp[] ; int n , top ; int binary_search (int x) { ; int last = top ; in ...

  4. tyvj 1049 最长不下降子序列 n^2/nlogn

    P1049 最长不下降子序列 时间: 1000ms / 空间: 131072KiB / Java类名: Main 描述 求最长不下降子序列的长度 输入格式 第一行为n,表示n个数第二行n个数 输出格式 ...

  5. 最长不下降子序列的O(n^2)算法和O(nlogn)算法

    一.简单的O(n^2)的算法 很容易想到用动态规划做.设lis[]用于保存第1~i元素元素中最长不下降序列的长度,则lis[i]=max(lis[j])+1,且num[i]>num[j],i&g ...

  6. 最长不下降子序列//序列dp

    最长不下降子序列 时间: 1000ms / 空间: 131072KiB / Java类名: Main 描述 求最长不下降子序列的长度 输入格式 第一行为n,表示n个数第二行n个数 输出格式 最长不下降 ...

  7. 【tyvj】P1049 最长不下降子序列

    时间: 1000ms / 空间: 131072KiB / Java类名: Main 描述 求最长不下降子序列的长度 输入格式 第一行为n,表示n个数 第二行n个数 输出格式 最长不下降子序列的长度 测 ...

  8. hdu 4604 Deque(最长不下降子序列)

    从后向前对已搜点做两遍LIS(最长不下降子序列),分别求出已搜点的最长递增.递减子序列长度.这样一直搜到第一个点,就得到了整个序列的最长递增.递减子序列的长度,即最长递减子序列在前,最长递增子序列在后 ...

  9. 最长不下降子序列nlogn算法详解

    今天花了很长时间终于弄懂了这个算法……毕竟找一个好的讲解真的太难了,所以励志我要自己写一个好的讲解QAQ 这篇文章是在懂了这个问题n^2解决方案的基础上学习. 解决的问题:给定一个序列,求最长不下降子 ...

随机推荐

  1. CF1168A Increasing by Modulo

    思路: 首先得做个转化,如果某个解法最终分别对a[i](i = 1, 2, ..., n)做了b[i](i = 1, 2, ..., n)次加1再取余的运算,那么可以等价地构造出x次(x = max( ...

  2. css hack 浏览器携带自身特有的属性 (二)

    css hack 浏览器携带自身特有的属性,才是我们真正要解决的css 兼容问题. 这里只是分享思路. 举例子: 1 outline,尤其是一些 自带继承特性的属性.这里指的是 隐性的inherite ...

  3. Android 设置资源字体,屏幕截图

    字体设置 将下载的资源字体放在assets中, 引用设置 edit..setTypeface(Typeface.createFromAsset(getAssets(), "字体名.ttf&q ...

  4. Android的bitmap和优化

    内存管理是个永恒的话题! 内存溢出:就是分配的内存不足以放下数据项序列.如在一个域中输入的数据超过了它的要求就会引发数据溢出问题,多余的数据就可以作为指令在计算机上运行.就是你要求分配的内存超出了系统 ...

  5. Jquery 如何获取表单中的全部元素的值

    1.使用var formData = $(formId).serialize()获取:获取数据的格式为url参数形式的字符串.例如:id=100&name=张三   2.服务器端使用parse ...

  6. Python +selenium之设置元素等待

    注:本文转载http://www.cnblogs.com/mengyu/p/6972968.html 当浏览器在加载页面时,页面上的元素可能并不是同时被加载完成,这给元素的定位增加了困难.如果因为在加 ...

  7. JavaScript_6_函数

    函数是由事件驱动的或者当它被调用执行的可重复使用的代码块 调用带参数的函数 带有返回值的函数 <!DOCTYPE html> <html> <head> <t ...

  8. (六)VMware Harbor简单使用

    VMware Harbor简单使用 1. 登陆: [用户:admin  , 密码:Harbor12345]配置文件里设置的 登陆后的界面: 2. 用户管理: 2.1 新近用户 3. 仓库管理: 3.1 ...

  9. Memcached笔记之分布式算法

    1.根据余数进行分散:离散度高,但是增加或者移除服务器的时候,缓存充足的代价非常大.添加服务器后,余数就会产生巨变,这样就无法获取与保存时相同的服务器,从而音像缓存的命中率. 2.Consistent ...

  10. javaweb基础(18)_jsp属性范围

    所谓的属性范围就是一个属性设置之后,可以经过多少个其他页面后仍然可以访问的保存范围. 一.JSP属性范围 JSP中提供了四种属性范围,四种属性范围分别指以下四种: 当前页:一个属性只能在一个页面中取得 ...