动态规划(冬令营课堂笔记)

简单问题

  01背包

  012背包

  部分背包

  机器分配

  烽火传递

  花店橱窗问题

简单问题

01背包

一个容量为m的背包,有n个物品,第i个物品的体积为wi,价值为ci。选择若干物品,使得体积总和不超过m的情况下价值总和最大。

n<=100,m<=10000。

搜索  复杂度为2^n

void dfs(int x,int y,int z)//当前物品编号,当前容量,当前价值
{
    if(x==n+1)
    {
        if(y<m)ans=max(ans,z);
        return;
    }
    dfs(x+1,y+w[x],z+c[i]);
    dfs(x+1,y,z);
}

如果前两维相同,只需选择第三维的最大值

小小的变形

const int INF=0x7f;
int dfs(int x,int y)
{
    int ans;
    if(x==n+1)
    {
        if(y>m)return -INF;
        return 0;
    }
    ans=max(dfs(x+1,y+w[x])+c[x],dfs(x+1,y));
    return ans;
}
dfs(1,0);

加一个数组记录

const int INF=0x7f;
int dfs(int x,int y)
{
    if(x==n+1)
    {
        if(y>m)return -INF;
        return 0;
    }
    dp[x][y]=max(dfs(x+1,y+w[x])+c[x],dfs(x+1,y));
    return dp[x][y];
}
dfs(1,0);

应用记录状态  记忆化搜索 n*m

const int INF=0x7f;
int dfs(int x,int y)
{
    if(y>m)return -INF;
    if(x==n+1)
    {
        return 0;
    }
    if(dp[x+1][y+w[x]])             //选该物品
    dp[x][y]=max(dp[x+1][y+w[x]]+c[x],dp[x][y]);
    else
    dp[x][y]=max(dfs(x+1,y+w[x])+c[x],dp[x][y]); 

    if(dp[x+1][y])             //不选该物品
    dp[x][y]=max(dp[x+1][y]+c[x],dp[x][y]);
    else
    dp[x][y]=max(dfs(x+1,y))+c[x],dp[x][y]);
    return dp[x][y];
}
dfs(1,0);

递归变递推

先写出搜索

最大/最小值 存放到数组中

后面递归该状态时 直接使用数组中的值来代替

O(dfs里的状态个数*转移的时间复杂度)

找边界

找递推方法(顺序还是逆序)

转移(推转移方程)

搜索->记忆化搜索->递归变递推

for(int i=n;i>=1;i--)
     for(int j=0;j<=m;j++)
         dp[i][j]=max(dp[i+1][j],dp[i+1][j+w[i]]+c[i]);
ans=dp[1][0]

012背包

一个容量为m的背包,有n个物品,第i个物品的体积为wi,价值为ci,有2个。

选择若干物品,使得体积总和不超过m的情况下价值总和最大。

记忆化搜索

const int INF=0x7f;
int dfs(int x,int y,int z)
{
    if(x==n+1){if(y<m)ans=max(ans,z);return -INF;}

    if(dp[x+1][y])                    //一个都不选
    dp[x][y]=max(dp[x][y],dp[x+1][y]);
    else
    dp[x][y]=max(dp[x][y],dfs(x+1,y));

    if(dp[x+1][y+w[x]])                //选一个
    dp[x][y]=max(dp[x][y],dp[x+1][y+w[x]])+c[x]);
    else
    dp[x][y]=max(dp[x][y],dfs(x+1,y+w[x])+c[x]);

    if(dp[x+1][y+w[x]])                //选两个
    dp[x][y]=max(dp[x][y],dp[x+1][y+2*w[x]])+2*c[x]);
    else
    dp[x][y]=max(dp[x][y],dfs(x+1,y+2*w[x])+2*c[x]);

    return dp[x][y];
}
dfs(1,0); 

部分背包

一个容量为m的背包,有n个物品,第i个物品的体积为wi,价值为ci,有ki个。

选择若干物品,使得体积总和不超过m的情况下价值总和最大。

记忆化搜索

int dfs(int x,int y)
{
    if (y>m) return -INF;
    if (x==n+1) return 0;
    for (int i=0; i<=k[x]; i++)
    {
        if (dp[x+1][y+i*w[x]])
            dp[x][y]=max(dp[x][y],dp[x+1][y+i*w[x]]+c[x]);
        else
              dp[x][y]=max(dp[x][y],dfs(x+1,y+i*w[x])+i*c[x]);
    }
    return dp[x][y];
}
dfs(1,0);

递推

for (i=n; i>=1; i--)
    for (j=0; j<=m; j++)
    for (l=0; l<=k[i]; l++)
    dp[i][j]=max(dp[i][j],dp[i+1][j+l*w[i]]+l*c[i]);
dp[1][0]

机器分配

有n家店,每家店都有m台机器,第i家店购买j台机器花费a[i][j]元(可能存在a[i][j]>a[i][j+1]),要购买总共m台机器,求最小花费。

搜索

//机器分配
dfs(int x,int y,int z)
{
    if(y>m)return;
    if(x==n+1)
    {
        if(y==n+1)ans=min(ans,z);
        return;
    }
    for(int i=0;i<=m;i++)
        dfs(x+1;y+i,z+a[x][i]);
}
dfs(1,0,0);

小小的变形

//机器分配
int dfs(int x,int y)
{
    if(y>m)return -INF;
    if(x==n+1)
    {
        if(y==m)return 0;//再也不需要买机器
        return -INF;
    }
    for(int i=0;i<=m;i++)
    dp[x][y]=min(dp[x][y],dfs(x+1,y+i)+a[x][i]);
    return dp[x][y];
}
dfs(1,0);

记忆化搜索

//机器分配
int dfs(int x,int y)
{
    if(y>m)return -INF;
    if(x==n+1)
    {
        if(y==m)return 0;//再也不需要买机器
        return -INF;
    }
    for(int i=0;i<=m;i++)
    if(dp[x+1][y+i])
    dp[x][y]=min(dp[x][y],dp[x+1][y+i]+a[x][i]);
    else
    dp[x][y]=min(dp[x][y],dfs(x+1,y+i)+a[x][i]);
    return dp[x][y];
}
dfs(1,0);

递推

for(int i=n;i>=1;i--)
    for(int j=0;j<=m;j++)
    {
        dp[i][j]=INF;
        for(k=0;k<=m-j;k++)
        dp[i][j]=min(dp[i][j],dp[i+1][j+k]+a[i][k]);
    }
ans=dp[1][0]

烽火传递

给定n个非负整数,选择其中若干数字,使得每连续k个数中至少有一个数被选出。

要求选择的数字之和尽可能小。

搜索

void dfs(int x,int y)
{
    if(x==n+1){ans=min(ans,y);return;}
    for(int i=x+1;i<=min(n+1,x+k);i++)
        dfs(i,y+a[i]);//i是下一个
}
dfs(0,0);

记忆化

int dfs(int x)
{
    if(x==n+1)return 0;
    dp[x]=INF;
    for(int i=x+1;i<=min(n+1,x+k);i++)
    if(dp[i])dp[x]=min(dp[x],dp[i]+a[i]);
    else
    dp[x]=min(dp[x],dfs(i)+a[i]);
    return dp[x];
}
dfs(0);

递推

/*dp[n+1]=0;
dp[1]  dp[2] dp[3] .. dp[1+k]*/

for (i=n; i>=0; i--)
{
    dp[i]=INF;
    for (int j=i+1; j<=min(n+1,i+k); j++)
        dp[i]=min(dp[i],dp[j]+a[j]);
}
dp[0]

花店橱窗问题

给定一个n*m的矩阵A(n<=m),求一个序列a1,a2,…,an满足1<=a1<a2<…<an<=m。使得A[1][a1]+A[2][a2]+…+A[n][an]最大。A可能有负数。

搜索

void dfs(int x,int y,int z)
{
    if(x==n+1)
    {
        if(y==m+1)ans=max(ans,z);
        return;
    }
    for(int i=y+1;i<=m;i++)
    dfs(x+1,i,z+a[x+1][i]);
}
dfs(0,0,0);

记忆化搜索

int dfs(int x,int y)
{
    if(x==n+1)
    {
        if(y==m+1)return 0;
        else return -INF;
    }
    for(int i=y+1;i<=m;i++)
    {
        if(dp[x+1][i])
        dp[x][y]=max(dp[x][y],dp[x+1][i]+a[x+1][i]);
        else
        dp[x][y]=max(dp[x][y],dfs(x+1,i)+a[x+1][i]);
        return dp[x][y];
    }
}
dfs(0,0);

递推

/*dp[n+1][m+1]=0; dp[n+1][0~m]=-INF;
dp[1][]  dp[2][] dp[3][]*/
for (i=n; i>=0;i--)
    for (j=0; j<=m+1; j++)
    for (k=j+1; k<=m+1; k++)
    dp[i][j]=max(dp[i][j],dp[i+1][k]+a[i+1][k]);
cout<<dp[0][0];

时间: 2024-12-22 18:47:45

动态规划(冬令营课堂笔记)的相关文章

九章算法系列(#3 Dynamic Programming)-课堂笔记

前言 时隔这么久才发了这篇早在三周前就应该发出来的课堂笔记,由于懒癌犯了,加上各种原因,实在是应该反思.好多课堂上老师说的重要的东西可能细节上有一些急记不住了,但是幸好做了一些笔记,还能够让自己回想起来.动态规划算是我的一道大坎了,本科的时候就基本没有学过,研一的时候老师上课也是吃力的跟上了老师的步伐,其实那个时候老师总结的还是挺好的:把动态规划的题目都分成了一维动规.二维遍历.二维不遍历等一系列的问题.这次听了老师的课程,觉得还是需要更加集中的去把各种题进行一个分类吧,然后有针对的去准备,虽然

九章算法系列(#2 Binary Search)-课堂笔记

前言 先说一些题外的东西吧.受到春跃大神的影响和启发,推荐了这个算法公开课给我,晚上睡觉前点开一看发现课还有两天要开始,本着要好好系统地学习一下算法,于是就爬起来拉上两个小伙伴组团报名了.今天听了第一节课,说真的很实用,特别是对于我这种算法不扎实,并且又想找工作,提高自己的情况. 那就不多说废话了,以后每周都写个总结吧,就趁着这一个月好好把算法提高一下.具体就从:课堂笔记.leetcode和lintcode相关习题.hdu和poj相关习题三个方面来写吧.希望自己能够坚持下来,给大家分享一些好的东

2017年5月12号课堂笔记

2017年5月12号 星期五 空气质量:轻度污染(昨天的北风转今天的南风) 内容:html表格的基本使用,表格跨行跨列,高级表格,播放音乐,播放视频,网页布局,iframe内联框架: 文本框,密码框,单选按钮,复选框,下拉框  备注:周日晚想起来补上的周五课堂笔记(一带一路今天开会天气好晴朗) 一.html表格的基本使用 模仿老师代码: <!DOCTYPE html><html><head lang="en"> <meta charset=&q

?统计学习精要(The Elements of Statistical Learning)?课堂笔记(一)

前两天微博上转出来的,复旦计算机学院的吴立德吴老师在开?统计学习精要(The Elements of Statistical Learning)?这门课,还在张江...大牛的课怎能错过,果断请假去蹭课...为了减轻心理压力,还拉了一帮同事一起去听,eBay浩浩荡荡的十几人杀过去好不壮观!总感觉我们的人有超过复旦本身学生的阵势,五六十人的教室坐的满满当当,壮观啊. 这本书正好前阵子一直在看,所以才会屁颠屁颠的跑过去听.确实是一本深入浅出讲data mining models的好书.作者网站上提供免

?统计学习精要(The Elements of Statistical Learning)?课堂笔记(三)

照例文章第一段跑题,先附上个段子(转载的哦~): I hate CS people. They don't know linear algebra but want to teach projective geometry. They don't know any probability but want to use graphical models. They don't understand stats at all but still do machine learning like c

线程(java课堂笔记)

1.两种方式的差异 2.线程的生命周期 3.线程控制(线程的方法) 4.线程同步 5.线程同步锁 一. 两种方式的差异 A extends Thread :简单 不能再继承其他类了(Java单继承)同份资源不共享 B implements Runnable:( 推荐) )多个线程共享一个目标资源,适合多线程处理同一份资源. 该类还可以继承其他类,也可以实现其他接口. 二. 线程的生命周期 新建:当程序使用new创建一个线程后,该线程处于新建状态,此时他和其他java对象一样,仅仅由Java虚拟机

CSS基础入门 第4天课堂笔记

CSS基础入门 第4天课堂笔记(本课程共6天) 前端与移动开发学院 http://web.itcast.cn 目录 目录 2 一.复习 3 二.浮动性质的复习 4 三.浮动的清除 5 3.1 清除浮动方法1:给浮动的元素的祖先元素加高度. 5 3.2 清除浮动方法2:clear:both; 6 3.3 清除浮动方法3:隔墙法 7 3.4 清除浮动方法4:overflow:hidden; 8 3.5 清除浮动总结与案例 9 3.6 浏览器兼容问题 11 四.margin 13 4.1 margin

SQL课堂笔记

--注释 公司里一般而是用绝不重复的guid()做主键(web项目不常用) 如null参与运算,结果都是null 在数据库中创建索引能提高查询效率,)只在经常要检索的字段创建索引) sql查询null的数据 selsct * from table where name is null 查询年龄介于20到30的数据 selsct * from table where age between 20 and 30 查询年龄是20,22,32,46的数据 selsct * from table wher

JAVA的面向对象编程--------课堂笔记

JAVA的面向对象编程--------课堂笔记 面向对象主要针对面向过程. 面向过程的基本单元是函数.   什么是对象:EVERYTHING IS OBJECT(万物皆对象)   所有的事物都有两个方面: 有什么(属性):用来描述对象. 能够做什么(方法):告诉外界对象有那些功能. 后者以前者为基础. 大的对象的属性也可以是一个对象.   为什么要使用面向对象: 首先,面向对象符合人类看待事物的一般规律. 对象的方法的实现细节是屏蔽的,只有对象方法的实现者了解细节. 方法的定义非常重要.方法有参