1、前言

大合集总共14道题,出自江哥之手(这就没什么好戏了),做得让人花枝乱颤。虽说大部分是NOIP难度,也有简单的几道题目,但是还是做的很辛苦,有几道题几乎没思路,下面一道道边看边分析一下。

2、lis 最长上升子序列

唯一一道裸题,但是O(n^2)过不了,临时看了看O(n log n)的二分做法和线段树做法。先来讲讲简单的二分做法,其本质就是在O(n^2)上进行优化,需要证明一个结论。设当前处理数列第k位,存在:

  (1)a[i]<a[j]<a[k];

  (2)i<j<k;

  (3)f[i]=f[j]。

  此时我们选择a[i]好还是a[j]好?显然是a[i]好。为什么?倘若存在a[i]<a[x]<a[j],i<x<j,则选择a[i]可以得到更长的子序列——a[i],a[x],a[k]……这样,对于f[t]=k,我们只需要保存a[t]值最小的那个t了,可以证明得f数组将会单调递增,故我们每次不需要向前每一位都进行查找,而可以进行二分查找,故时间复杂度为O(n log n)。

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

#include<cstdio>

int min(int a,int b) { return (a<b)?a:b; }

int v[200010],n,x,ans;

int main()
{
  freopen("lis.in","r",stdin);
  freopen("lis.out","w",stdout);
  scanf("%d", &n);
  for (int i=1;i<=n;i++)
  {
    scanf("%d",&x);
    int l=1,r=ans,mid=(l+r)>>1,loc=0;
    while (l<=r)
    {
      if (v[mid]<=x) loc=mid,l=mid+1;
      else r=mid-1;
      mid=(l+r)>>1;
    }
    if (loc==ans) v[++ans]=x;
    else v[loc+1]=min(v[loc+1],x);
  }
  printf("%d\n",ans);

}

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

3、chess 黑白棋盘

根据题目分析,我们需要维护该列是否存在棋子;该行是否存在棋子;由于n<=15,我们可以考虑二进制状态压缩进行转移,最多有2^(n+1)种状态。设f[i][j]为当前第i行,状态为j(压缩的二进制状态 )时的方案数。今天才刚刚学会用&、^、|等位运算符,上次写状压的时候还是手写的。。。方程:f[i][j]=sum{f[i-1][j^(1<<(k-1))]} (k∈[1,n],j&(1<<(k-1)),!a[i][k])

代码:

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

#include <cstdio>
#define MOD 1e9+7

int n,a[20][20],f[20][100000],ans;
char ch;

int main()
{
  freopen("chess.in","r",stdin);
  freopen("chess.out","w",stdout);
  scanf("%d",&n);
  for (int i=1;i<=n;i++)
    for (int j=1;j<=n;j++)
    {
      do ch=getchar(); while ((ch!='0')&&(ch!='1'));
      a[i][j]=ch-'0';
    }
  f[0][0]=1;
  for (int i=1;i<=n;i++)
    for (int j=0;j<1<<n;j++)
    {
      f[i][j]=f[i-1][j];
      for (int k=1;k<=n;k++)
        if ((j&(1<<(k-1))) && (!a[i][k])) 
        {
          f[i][j]+=f[i-1][j^(1<<(k-1))];
          if (f[i][j]>=MOD) f[i][j]-=MOD;
        }
    }
  for (int i=0;i<1<<n;i++) { ans+=f[n][i]; if (ans>=MOD) ans-=MOD; }
  printf("%d",ans);
  return 0;
}

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

4、bound 多背包问题

背包问题,但是需要用到单调队列优化。题解有点没懂?但是看代码大概清楚了。

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

#include<cstdio>
#include<algorithm>
#define MAXN 10005
using namespace std;

int q[MAXN],f[MAXN],g[MAXN];
int n,m,s,w,v;

int dp(int x,int y) { return g[x]+(y-x)/w*v; }

int main()
{
  freopen("bound.in","r",stdin);
  freopen("bound.out","w",stdout);
  scanf("%d %d",&n,&m);
  for (int i=1;i<=n;i++)
  {
    scanf("%d %d %d",&s,&w,&v);
    for (int start=0;start<w;start++)
    {
      int tail=0,head=1;
      for (int j=start;j<=m;j+=w)
      {
        while (head<=tail && (j-q[head])/w>s) ++head;
        while (head<=tail && dp(q[tail],j)<=dp(j,j)) --tail;
        q[++tail]=j;
        f[j]=dp(q[head],j);
      }
    }
    swap(f,g);
  }
  int ans=0;
  for (int i=0;i<=m;i++) ans=max(ans,g[i]);
  printf("%d",ans);
  return 0;
}

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

5、matrix 子矩阵

也是所有题目中最需要码力的了吧,首先步骤比较复杂,情况比较繁琐,但是没有什么需要优化的地方。由于必定存在某条直线将三个矩阵割开,主要思路为枚举切割线,然后就只需要考虑两个矩阵的情况了。

6、windy windy数

数位动态规划,考试时想出来了但是没写下去。由于我们所需的信息明显:当前确定的数是否小于原数的这些位所对应的数;最后一位数;是否存在前导0,所以用记忆化搜索可能更易理解。下面代码来自yxj。

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

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;

int p[20], r[20];
int f[20][20][2][2];
int a, b;

int dfs(int pos, int last, int s, int zero) {
  if (pos > 10) return s == 0 && zero == 0;
  int &u = f[pos][last][s][zero];
  if (u != -1) return u;
  u = zero == 0 && (r[pos] || s == 0);
  for (int i = 0; i <= 9; ++i) {
    if (abs(last - i) < 2) continue;
    int ns = i > p[pos] ? 1 : (i == p[pos] ? s : 0);
    u += dfs(pos + 1, i, ns, i == 0);
  }
  return u;
}

int tot(int x) {
  for (int i = 0; i <= 10; ++i) {
    p[i] = x % 10;
    r[i] = x;
    x /= 10;
  }
  memset(f, -1, sizeof(f));
  return dfs(0, 11, 0, 0);
}

int main() {
  freopen("windy.in", "r", stdin);
  freopen("windy.out", "w", stdout);

  cin >> a >> b;
  cout << tot(b) - tot(a - 1) << endl;
}

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

[题解+总结]动态规划大合集II的更多相关文章

  1. 动态规划大合集II

    1.前言 大合集总共14道题,出自江哥之手(这就没什么好戏了),做得让人花枝乱颤.虽说大部分是NOIP难度,也有简单的几道题目,但是还是做的很辛苦,有几道题几乎没思路,下面一道道边看边分析一下. 2. ...

  2. [题解+总结]NOIP动态规划大合集

    1.前言 NOIP2003-2014动态规划题目大合集,有简单的也有难的(对于我这种动态规划盲当然存在难的),今天就把这些东西归纳一下,做一个比较全面的总结,方便对动态规划有一个更深的理解. 2.NO ...

  3. NOIP动态规划大合集

    1.前言 NOIP2003-2014动态规划题目大合集,有简单的也有难的(对于我这种动态规划盲当然存在难的),今天就把这些东西归纳一下,做一个比较全面的总结,方便对动态规划有一个更深的理解. 2.NO ...

  4. Lucene搜索方式大合集

    package junit; import java.io.File; import java.io.IOException; import java.text.ParseException; imp ...

  5. 【收藏】Java多线程/并发编程大合集

    (一).[Java并发编程]并发编程大合集-兰亭风雨    [Java并发编程]实现多线程的两种方法    [Java并发编程]线程的中断    [Java并发编程]正确挂起.恢复.终止线程    [ ...

  6. 从零开始学数据分析,什么程度可以找到工作?( 内附20G、5000分钟数据分析工具教程大合集 )

    从零开始学数据分析,什么程度可以找到工作?( 内附20G.5000分钟数据分析工具教程大合集 )   我现在在Coursera上面学data science 中的R programming,过去很少接 ...

  7. 直接拿来用!Facebook移动开源项目大合集

    直接拿来用!Facebook移动开源项目大合集 时间:2014-04-22 15:37 作者:唐小引 随着iOS依赖管理工具CocoaPods和大量第三方开源库成熟起来,业界积累了大量的优秀开源项目. ...

  8. python字符串操作实方法大合集

    python字符串操作实方法大合集,包括了几乎所有常用的python字符串操作,如字符串的替换.删除.截取.复制.连接.比较.查找.分割等,需要的朋友可以参考下:   #1.去空格及特殊符号 s.st ...

  9. 【Oracle教程资源大合集】Oracle数据库免费学习资源汇总

    Oracle的产品非常丰富,各类学习资源也五花八门,本文将介绍Oracle官方的免费教程与风哥整理的Oracle视频教程: 1.Oracle帮助中心 Oracle帮助中心也称为Oracle文档中心,这 ...

随机推荐

  1. ASP.NET的Cookie和Session

    HTTP属于应用层,HTTP协议一共有五大特点:1.支持客户/服务器模式;2.简单快速;3.灵活;4.无连接;5.无状态. 无状态HTTP协议是无状态的协议.一旦数据交换完毕,客户端与服务器端的连接就 ...

  2. JavaWeb学习之JSP常用标签、EL表达式的运算符、JSTL标签库(6)

    1.JSP常用标签 * 只要支持JSP文件,常用标签有可以直接使用 * 格式: jsp:xxxx * jsp:forward ,完成jsp页面的转发 * page属性:转发的地址 <% requ ...

  3. Linux Shell 高级编程技巧3----运行级别脚本介绍

    3.运行级别脚本介绍    3.1.运行级别        运行级别介绍:            0    关机            1    单用户模式            2    多用户模式 ...

  4. 在RedHat.Enterprise.Linux_v6.3系统中安装Oracle_11gR2教程

    在RedHat.Enterprise.Linux_v6.3系统中安装Oracle_11gR2教程 本教程提供PDF格式下载: 在RedHat.Enterprise.Linux_v6.3系统中安装Ora ...

  5. AOJ789 买酒

    买酒 Time Limit: 1000 ms   Case Time Limit: 1000 ms   Memory Limit: 64 MBTotal Submission: 70   Submis ...

  6. HDU5489 Removed Interval(动态规划)

    一个长度为n的序列,删除任意长度为l的连续子序列后,求剩下的序列的最长公共子序列. 先求出以第i个元素为开始的LIS的长度,再一次循环,对所要求的结果更新 #include<iostream&g ...

  7. HYSBZ 2440 完全平方数(莫比乌斯反演)

    链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2440 若i为质数,n为i*i的倍数,则称n为含平方因子数. 求1~n的无平方因子数. F(x) ...

  8. 在Xcode中想要清屏该怎么实现

    XCODE的控制台不会有清屏效果,并没有像终端一样可以clear.但在某些时候我们非得想要清屏该怎么办呢??你去打开可执行文件,就会有类似清屏的效果.实际上是它帮你自动换页了,xcode左边是可以看到 ...

  9. T-SQL 高级查询

    基本常用查询 --select select * from student; --all 查询所有 select all sex from student; --distinct 过滤重复 selec ...

  10. nginx配置文件nginx.conf

    #定义Nginx运行的用户和用户组user www www; #nginx进程数,建议设置为等于CPU总核心数.worker_processes 8; #全局错误日志定义类型,[ debug | in ...