摘果子【DFS】(伪AC)

2024-01-30 10:38
文章标签 dfs ac 果子

本文主要是介绍摘果子【DFS】(伪AC),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目大意:

题目图片:
http://wx2.sinaimg.cn/mw690/0060lm7Tly1fwdtih8rllj30gw081q5h.jpg
http://wx1.sinaimg.cn/mw690/0060lm7Tly1fwdtih5bq8j30hb042wex.jpg
http://wx2.sinaimg.cn/mw690/0060lm7Tly1fwdtih52t0j30gu01r3yk.jpg
http://wx4.sinaimg.cn/mw690/0060lm7Tly1fwdtih5hmxj30j50gudgq.jpg

给出一棵树,选则结点 x x x可以获得价值 v [ x ] v[x] v[x],但是代价是 p [ x ] p[x] p[x]。可以选择这个结点仅当这个结点的父亲被选择。求代价不超过 m m m时的最大价值。


思路:

这道题数据 n ≤ 1000 n\leq 1000 n1000。但是数据太菜(不是JZOJ的),所以 O ( 玄 学 ) O(玄学) O()也是可以过的。。。
首先树形 D P DP DP肯定是可取的。设 f [ i ] [ j ] f[i][j] f[i][j]表示以第 i i i个结点为根,代价为 j j j是的最大价值。方程略。
这里要讲的是 D F S DFS DFS的做法。
我们假设有一棵树是这样的(蓝色代表已选):
在这里插入图片描述
很明显,可选的点就是父亲选了,自己没选的点。
那么可选的点就有(红色):
在这里插入图片描述
那么对于任意一个状态,我们先枚举每一个点,判断它是不是已经被选择,如果已经被选择,那么我们就枚举它的子节点,如果它的子节点没有被选择,就选择这个子节点,继续搜索。
时间复杂度: O ( ≥ n 2 ) O(\geq n^2) O(n2)


代码:

#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 2100
using namespace std;int v[N],p[N],n,m,tot,ans,head[N];
bool vis[N];struct edge
{int next,to;
}e[N*2];void add(int from,int to)
{e[++tot].to=to;e[tot].next=head[from];head[from]=tot;
}void dfs(int x,int s,int du)  //s表示价值,du表示代价
{if (du>m) return;  //代价不能超过mvis[x]=1;  //记录这个点是否走过if (s>ans) ans=s;  //记录最优答案for (int u=1;u<=n;u++)if (vis[u])  //选择过这个点for (int i=head[u];~i;i=e[i].next)if (!vis[e[i].to])  //这个选择过的点的儿子dfs(e[i].to,s+v[e[i].to],du+p[e[i].to]);vis[x]=0;
}int main()
{memset(head,-1,sizeof(head));scanf("%d%d",&n,&m);for (int i=1;i<=n;i++)scanf("%d%d",&v[i],&p[i]);int x,y;for (int i=1;i<n;i++){scanf("%d%d",&x,&y);add(x,y);add(y,x);}ans=-1e9;dfs(1,v[1],p[1]);printf("%d\n",max(ans,0));
}

这篇关于摘果子【DFS】(伪AC)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/659971

相关文章

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

poj 3050 dfs + set的妙用

题意: 给一个5x5的矩阵,求由多少个由连续6个元素组成的不一样的字符的个数。 解析: dfs + set去重搞定。 代码: #include <iostream>#include <cstdio>#include <set>#include <cstdlib>#include <algorithm>#include <cstring>#include <cm

hdu 3065 AC自动机 匹配串编号以及出现次数

题意: 仍旧是天朝语题。 Input 第一行,一个整数N(1<=N<=1000),表示病毒特征码的个数。 接下来N行,每行表示一个病毒特征码,特征码字符串长度在1—50之间,并且只包含“英文大写字符”。任意两个病毒特征码,不会完全相同。 在这之后一行,表示“万恶之源”网站源码,源码字符串长度在2000000之内。字符串中字符都是ASCII码可见字符(不包括回车)。

zoj 3228 ac自动机

给出一个字符串和若干个单词,问这些单词在字符串里面出现了多少次。单词前面为0表示这个单词可重叠出现,1为不可重叠出现。 Sample Input ab 2 0 ab 1 ab abababac 2 0 aba 1 aba abcdefghijklmnopqrstuvwxyz 3 0 abc 1 def 1 jmn Sample Output Case 1 1 1 Case 2

D4代码AC集

贪心问题解决的步骤: (局部贪心能导致全局贪心)    1.确定贪心策略    2.验证贪心策略是否正确 排队接水 #include<bits/stdc++.h>using namespace std;int main(){int w,n,a[32000];cin>>w>>n;for(int i=1;i<=n;i++){cin>>a[i];}sort(a+1,a+n+1);int i=1

ural 1149. Sinus Dances dfs

1149. Sinus Dances Time limit: 1.0 second Memory limit: 64 MB Let  An = sin(1–sin(2+sin(3–sin(4+…sin( n))…) Let  Sn = (…( A 1+ n) A 2+ n–1) A 3+…+2) An+1 For given  N print  SN Input One

hdu 6198 dfs枚举找规律+矩阵乘法

number number number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Problem Description We define a sequence  F : ⋅   F0=0,F1=1 ; ⋅   Fn=Fn

深度优先(DFS)和广度优先(BFS)——算法

深度优先 深度优先搜索算法(英语:Depth-First-Search,DFS)是一种用于遍历或搜索树或图的算法。 沿着树的深度遍历树的节点,尽可能深的搜索树的分支,当节点v的所在边都己被探寻过,搜索将回溯到发现节点v的那条边的起始节点。这一过程一直进行到已发现从源节点可达的所有节点为止。如果还存在未被发现的节点,则选择其中一个作为源节点并重复以上过程,整个进程反复进行直到所有节点都被访

nyoj99(并查集+欧拉路+dfs)

单词拼接 时间限制: 3000 ms  |  内存限制: 65535 KB 难度: 5 描述 给你一些单词,请你判断能否把它们首尾串起来串成一串。 前一个单词的结尾应该与下一个单词的道字母相同。 如 aloha dog arachnid gopher tiger rat   可以拼接成:aloha.arachnid.dog.gopher.rat.tiger 输入 第一行是一个整

【CF】E. Anya and Cubes(双向DFS)

根据题意的话每次递归分3种情况 一共最多25个数,时间复杂度为3^25,太大了 我们可以分2次求解第一次求一半的结果,也就是25/2 = 12,记录结果 之后利用剩余的一半求结果 s-结果 = 之前记录过的结果 就可以 时间复杂度降低为 3 ^ (n/2+1) 题目链接:http://codeforces.com/contest/525/problem/E #include<set