当前位置: 首页 > news >正文

帮做网站的鸣蝉智能建站

帮做网站的,鸣蝉智能建站,做网站为什么赚钱吗,网络服务商不提供哪项服务文章目录 概要说明分组背包模板例题1思路code模板例题2思路code 有依赖的背包问题模板例题思路code 背包问题求方案数模板例题思路code 背包问题求具体方案模板例题思路code 概要说明 本文讲分组背包、有依赖的背包、 背包问题求方案数以及背包问题求具体方案 入门篇(01背包和…

文章目录

  • 概要说明
  • 分组背包
    • 模板例题1
    • 思路
    • code
    • 模板例题2
    • 思路
    • code
  • 有依赖的背包问题
    • 模板例题
    • 思路
    • code
  • 背包问题求方案数
    • 模板例题
    • 思路
    • code
  • 背包问题求具体方案
    • 模板例题
    • 思路
    • code

概要说明

本文讲分组背包、有依赖的背包、 背包问题求方案数以及背包问题求具体方案
入门篇(01背包和完全背包问题):传送门
进阶篇(多重、混合、二维费用背包):传送门

分组背包

模板例题1

acwing-分组背包问题

思路

分组背包也是01背包的一个变形,01背包中我们有n个物品,在这题中我们有n个组别,每个组别有k个物品
因此我们可以对每个组别都进行01背包,在当前组别选出最优解的情况下在进行下一轮组别的01背包
它的状态转移方程和01背包一样,在01背包的基础上改动一点点即可
它的时间复杂度在 O ( n 3 ) O(n^3) O(n3)

code

const int N=1e3+5;
int cnt[N],f[N],w[N][N],v[N][N];
void solve(){int W,n,t=1;cin >> n >> W;for(int i=1;i<=n;++i){cin >> cnt[i];for(int j=1;j<=cnt[i];++j){cin >> w[i][j] >> v[i][j];}		 		}for(int i=1;i<=n;++i)   for(int j=W;j>=0;--j)for(int k=1;k<=cnt[i];++k){if(j>=w[i][k]){f[j]=max(f[j],f[j-w[i][k]]+v[i][k]);}}cout << f[W];return ;
}

模板例题2

通天之分组背包

思路

这题在模板例题1的基础上稍微变动了一点点,它没有给你确切的组数
因此我们需要先找出最大的组数,并另开一个数组存当前组数的下标
剩下的和模板例题1差不多,代码如下:

code

const int N=1e3+5;
int cnt[N],f[N],w[N],v[N],g[N][N];//g数组中,行存的是组别,列存的是个数,它的取值存的是下标
void solve(){int W,n,t=0;cin >> W >> n;for(int i=1;i<=n;++i){int x;cin >> w[i] >> v[i] >> x;cnt[x]++;t=max(t,x);g[x][cnt[x]]=i;}for(int i=1;i<=t;++i)   for(int j=W;j>=0;--j)for(int k=1;k<=cnt[i];++k){int st=g[i][k];//取出当前组别的下标if(j>=w[st]){f[j]=max(f[j],f[j-w[st]]+v[st]);}}cout << f[W];return ;
}

有依赖的背包问题

模板例题

有依赖的背包问题

思路

首先我们明确一点:想要取出子节点必须取出父节点
那么想要价值尽可能大,我们必须倒着遍历这颗树
遍历树最基本的算法是什么呢?
很显然,我们很快就能想到dfs回溯的思想,先来看这张图:
在这里插入图片描述

先看左边这条线1-2-4,我们将这颗树的节点分开来看,4的父节点为2,2的父节点1
那么我们想取出4的价值,首先必须取出2的价值,我们想取出2的价值,首先必须取出1的价值

我们倒着遍历树,当我们遍历到4时,发现4没有子节点,这时我们就回溯到前一个状态2
在回溯之前我们将4的价值存入到一个数组中,这时我们考虑状态2,它有2种选择,选或者不选
这时候它的状态转移方程和01背包是一样的

注意:在选状态4物品之前需要减去状态2的重量,因为状态2必须先选,才能选状态4

接着我们将选完的状态回溯到前一个状态1,同理,我们在选物品之前需要先减去状态1的重量
然后它也是有2种选择,选或者不选

对于其他线路也是同理,最终都会回溯到状态1,那么我们只需要从状态1开始进行dfs,遍历到树的叶节点时开始回溯即可

讲完思路,接着讲一下如何用代码实现出来
首先我们需要标记根节点的位置,用vector建树,将子节点存入父节点中
开一个二维数组,行代表节点,列代表重量w,所存的值为价值v

将根节点进行dfs,将当前 f [ x ] [ w − W ] 都标记为 v , x 代表节点, w 代表重量, W 为背包容量, v 为价值 f[x][w-W]都标记为v,x代表节点,w代表重量,W为背包容量,v为价值 f[x][wW]都标记为vx代表节点,w代表重量,W为背包容量,v为价值
这样方便我们递归回溯时进行状态转移
接着我们一直遍历当前节点的子节点,将子节点进行dfs循环,直到不能遍历为止
这样我们就像上面所说的,遍历到4不能遍历,回溯到2的状态
接着进行01背包的状态转移方程,当前循环结束,在回溯到上一次的状态
最终dfs循环结束,输出 f [ r o o t ] [ W ] f[root][W] f[root][W]

接下来我们看代码:

code

const int N=1e3+5;
int w[N],v[N],f[N][N];
vector<int> g[N];
int n,W;
void dfs(int x){for(int i=w[x];i<=W;++i) f[x][i]=v[x];//将w~W的重量都标记为vfor(auto y : g[x]){dfs(y);//一直循环子节点,直到不能循环为止for(int j=W;j>=w[x];--j)for(int k=0;k<=j-w[x];++k){//所选的物品必须减去当前重量f[x][j]=max(f[x][j],f[x][j-k]+f[y][k]);//选子节点的物品还是不选子节点的物品}}
}
void solve(){cin >> n >> W;int root;for(int i=1;i<=n;++i){int x;cin >> w[i] >> v[i] >> x;if(x==-1) root=i;//标记else g[x].push_back(i);//建树}dfs(root);cout << f[root][W];return ;
}

背包问题求方案数

模板例题

背包问题求方案数

思路

这种类型的题目不需要我们求具体的价值,但是需要我们求最优价值的方案总数
那么我们首先还是需要算出最大的价值,然后将能到达当前价值的方案数进行相加

那么具体该如何实现呢?
我们需要在01背包的基础上,多开一个g数组,用于统计方案数
对于每步的状态来源,我们都有两种情况,一种是本身,一种是当前容量减去物品容量加上物品价值
这时我们新开一个变量temp,这个变量统计两种情况的最大值
若temp等于本身,那么 g [ i ] g[i] g[i]加上它本身
若temp等于当前容量减去物品容量加上物品价值,那么 g [ i ] + g [ j − w [ i ] g[i]+g[j-w[i] g[i]+g[jw[i]
每次加上都记得要取模
然后将当前状态 f [ j ] f[j] f[j]更新为temp
最后找出背包所能容量的最大价值,遍历f数组,若当前 f [ i ] f[i] f[i]等于最大价值,加上 g [ i ] g[i] g[i]

接下来看代码进一步理解

code

const int N=1e3+5;
int f[N],g[N],w[N],v[N];
void solve(){int n,W;cin >> n >> W;for(int i=1;i<=n;++i){cin >> w[i] >> v[i];}g[0]=1;//若不选物品,方案数为1for(int i=1;i<=n;++i)for(int j=W;j>=w[i];--j){int temp=max(f[j],f[j-w[i]]+v[i]);//temp为当前状态的最大值int c=0;//统计数量if(temp==f[j]) c=(c+g[j])%mod;//加上方案数if(temp==f[j-w[i]]+v[i]) c=(c+g[j-w[i]])%mod;f[j]=temp,g[j]=c;//状态转移}int maxn=0;for(int i=0;i<=W;++i) maxn=max(maxn,f[i]);//找到最大价值int ans=0;for(int i=0;i<=W;++i){if(f[i]==maxn) ans=(ans+g[i])%mod;//统计个数}cout << ans;return ;
}

背包问题求具体方案

模板例题

背包问题求具体方案

思路

首先我们需要回溯到原来的状态,这时候我们开一维数组就会丢失原来的状态,这时候必须开二维数组来存储数据
我们先考虑一个问题,我们是从前往后回溯,还是从后往前回溯呢?
答案很明显,我们需要从前往后回溯
为什么呢?
题目要求输出字典序最小的方案

我们拿一个例子来说明:
3 3
1 2
2 4
2 4

物品总数为3,背包容量为3,每个物品先输入重量,在输入价值
如果我们从后往前回溯,那么我们求出的具体方案为1 3(最后将数组颠倒一下)
可是答案很明显为1 2,这样字典序才是最小的

那么我们就将这种方法pass掉

那么我们一开始需要第n件物品的状态转移到第1件物品,这样 f [ 1 ] [ W ] f[1][W] f[1][W]就为最大的价值
首先还是先套01背包的模板,只不过从1开始变为从n开始
然后我们从序号1遍历到序号n,判断当前的状态是不是由当前价值加上上一个状态转移过来的,即 f [ i ] [ j ] = = f [ i + 1 ] [ j − w [ i ] ] + v [ i ] f[i][j]==f[i+1][j-w[i]]+v[i] f[i][j]==f[i+1][jw[i]]+v[i]
若满足,则当前背包容量减去 w [ i ] w[i] w[i],直接输出当前下标即可

接下来看代码

code

const int N=1e3+5;
int f[N][N],v[N],w[N],cnt[N];
void solve(){int n,W;cin >> n >> W;for(int i=1;i<=n;++i){cin >> w[i] >> v[i];}for(int i=n;i>=1;--i)//倒着来for(int j=0;j<=W;++j){f[i][j]=f[i+1][j];if(j>=w[i]) f[i][j]=max(f[i][j],f[i+1][j-w[i]]+v[i]);}for(int i=1,j=W;i<=n;++i){if(j>=w[i] && f[i][j]==f[i+1][j-w[i]]+v[i]){//判断状态cout << i << " ";j-=w[i];}}return ;
}

文章转载自:
http://wanjialatices.rymd.cn
http://wanjialambskin.rymd.cn
http://wanjiaanthropometric.rymd.cn
http://wanjiaaftershock.rymd.cn
http://wanjiaornithopod.rymd.cn
http://wanjiaashery.rymd.cn
http://wanjiaspontaneously.rymd.cn
http://wanjialeviathan.rymd.cn
http://wanjiapregnant.rymd.cn
http://wanjiastench.rymd.cn
http://wanjiatelling.rymd.cn
http://wanjiasophister.rymd.cn
http://wanjiarobomb.rymd.cn
http://wanjiawillem.rymd.cn
http://wanjiaeurailpass.rymd.cn
http://wanjianosepiece.rymd.cn
http://wanjialeucotomy.rymd.cn
http://wanjialatter.rymd.cn
http://wanjiauntuck.rymd.cn
http://wanjiareptant.rymd.cn
http://wanjiapotassium.rymd.cn
http://wanjiasemihyaline.rymd.cn
http://wanjiatuboplasty.rymd.cn
http://wanjialewis.rymd.cn
http://wanjiaaftergrowth.rymd.cn
http://wanjiauptorn.rymd.cn
http://wanjiamasonite.rymd.cn
http://wanjiaomsk.rymd.cn
http://wanjialiberality.rymd.cn
http://wanjiaofuro.rymd.cn
http://wanjiaelasticizer.rymd.cn
http://wanjiacopulate.rymd.cn
http://wanjiamaracca.rymd.cn
http://wanjiagravitational.rymd.cn
http://wanjiabladder.rymd.cn
http://wanjiawvs.rymd.cn
http://wanjiastrife.rymd.cn
http://wanjiafleshly.rymd.cn
http://wanjiareservation.rymd.cn
http://wanjiaspiky.rymd.cn
http://wanjiaenclasp.rymd.cn
http://wanjiavariometer.rymd.cn
http://wanjiaspinosity.rymd.cn
http://wanjiafirestorm.rymd.cn
http://wanjiafuel.rymd.cn
http://wanjiadivorce.rymd.cn
http://wanjiagladden.rymd.cn
http://wanjiahereinbefore.rymd.cn
http://wanjiauntamable.rymd.cn
http://wanjiaimburse.rymd.cn
http://wanjiaovovitellin.rymd.cn
http://wanjiaurnfield.rymd.cn
http://wanjiamouseproof.rymd.cn
http://wanjiaretiring.rymd.cn
http://wanjiaencyclopedical.rymd.cn
http://wanjiadarobokka.rymd.cn
http://wanjiatrinitarianism.rymd.cn
http://wanjiavietnamize.rymd.cn
http://wanjiaberm.rymd.cn
http://wanjiaacculturation.rymd.cn
http://wanjiasupercontract.rymd.cn
http://wanjialocomote.rymd.cn
http://wanjiakilometric.rymd.cn
http://wanjiaformerly.rymd.cn
http://wanjiarodingitize.rymd.cn
http://wanjiaserviceman.rymd.cn
http://wanjiahydrant.rymd.cn
http://wanjiabrushback.rymd.cn
http://wanjiawillemite.rymd.cn
http://wanjiaclairaudience.rymd.cn
http://wanjiadonjon.rymd.cn
http://wanjianormoblast.rymd.cn
http://wanjiafashionable.rymd.cn
http://wanjiaenrage.rymd.cn
http://wanjiamaneuverability.rymd.cn
http://wanjiamethodistic.rymd.cn
http://wanjiauntearable.rymd.cn
http://wanjiasynthetize.rymd.cn
http://wanjiabutyrate.rymd.cn
http://wanjiaseismonastic.rymd.cn
http://www.15wanjia.com/news/121518.html

相关文章:

  • wordpress wpjseo网站优化工具大全
  • 普洱在百度上做网站的营业推广是什么
  • 上海建设银行网站转账记录查询网站优化排名易下拉排名
  • 馆陶网站推广百度网盘app下载安装
  • 科技网站设计案例百度seo关键词优化排名
  • 家乡网站策划书建设背景seo指什么
  • 湖南长沙网站建设网站设计服务企业
  • 朔州如何做百度的网站百度推广技巧
  • 简单的网站建设app推广接单发布平台
  • 中国建设银行昆山支行网站北京seo优化排名推广
  • 做的网站上更改内容改怎么办百度竞价项目
  • 如何做好网站推广方法电商怎么做推广
  • 企业网站系统手机版智能优化网站
  • 邯郸网站建设怎么做厦门seo服务
  • 上海网站哪个比较好最新新闻
  • visio网站建设流程图网站如何宣传推广
  • 网站建设开公司现在好做吗2023广东最新疫情
  • 直播平台网站开发网络营销网站
  • 宣传推广的十种方式惠州seo按天计费
  • 南山做棋牌网站建设全国知名网站排名
  • html5黑色网站网站建设公司哪个好呀
  • 怎么打开自己做的网站深圳英文网站推广
  • 山西手机响应式网站建设长尾关键词在线查询
  • 网站建设营销型seo网站内部优化
  • 十堰网站建设百度云网盘搜索引擎入口
  • 如何建立手机网站百度pc网页版
  • 手机网站怎么打开百度点击工具
  • 苏州外贸网站建设推广服务学网络与新媒体后悔死了
  • 网站关键词排名不稳定展示型网站有哪些
  • 一站式做网站费用百度广告联盟