AcWing 3305. 作物杂交
创始人
2024-05-30 19:22:11
0

 

先看看题目,总结下来就是根据已有的种子进行杂交,以最快的速度得到目标种子。

这个题是蓝桥杯的原题,在蓝桥杯练习系统提交的时候,用dfs是可以过的,但是在acwing上是过不了的,好像是因为会出现环。

所以在这里把两种思路都记下来。

1.dfs

dfs的思路比较简单,查找可以杂交得到目标种子的植物a和b,如果a和b没有现成的种子,那么接着查找a和b通过杂交得到的最短时间。

这个思路很简单的,所以也没啥可说的。然后一个需要注意的点是dfs返回的是得到该种子的时间,所以杂交的时候得等到这个种子长成植物之后才可以。

那么我们在计算通过种子a和b得到c的时间就是seed[a] = max(seed[b],seed[c])+max(t[a],t[b]).先同时得到a和b的种子,然后同时种下去。

代码如下:

int T[N];
int K[N];
int flag[N];  //是否得到了这个种子 
int seed[N];  //得到每个种子的最小时间 
vectormp[N];
int dfs(int x){for(int i = 0; i < mp[x].size(); i += 2){int a = mp[x][i];int b = mp[x][i + 1];if(!flag[a]) seed[a] = dfs(a);if(!flag[b]) seed[b] = dfs(b);seed[x] = min(seed[x], max(seed[a], seed[b]) + max(T[a], T[b]));flag[x] = 1;} return seed[x];
} 
int main(){int n, m, k, t;cin>>n>>m>>k>>t;memset(seed, 0x3f, sizeof(seed));for(int i = 1; i <= n; i++){cin>>T[i];} for(int i = 0; i < m; i++){cin>>K[i];flag[K[i]] = 1;seed[K[i]] = 0; }for(int i = 0; i < k; i++){int a, b, c;cin>>a>>b>>c;mp[c].push_back(a);mp[c].push_back(b);}cout<

2.spfa

说实话,如果这道题的算法标签不打上spfa的话,我是肯定想不到的。但是即使打上了,我也还是没明白怎么建图。

下面记录一下我的草率理解。

图中有n个顶点,有m条边,我们在使用邻接表建图的时候,这条边上不但可以存储边权,也可以存储其他的信息,例如这条边相关的两个节点可以生成的新种子。那么我们在建边的时候不但要记录这条边的两个顶点还要记录这两个种子杂交生成的新种子。

然后这个题我是这么理解的,因为我们想要得到生成目标种子的最短时间,相当于走了很多条边然后得到一条到目标种子t的最短路。

但是这个spfa是计算的单源路径的,这个源在哪呢?于是我们想到了前几天做的那道题,我们也虚拟一个超级源点,然后对于每个已有的种子,和这个超级源点0之间的距离就是0。然后我们对现有的边开始枚举,不断更新每个顶点到源点的距离,直到我们得到目标种子的最短时间,也就是和源点之间的最短路径。

然后代码如下:

#include
#include
#include
#include
#include
#include
#define ll long long
using namespace std;
const int N = 2010;
const int M = 200005;
int h[N], e[M], ne[M], target[M], idx, w[N];
int dis[N]; 
bool st[N];
int n, m, k, t; 
queueq;
void add(int a, int b, int c){e[idx] = b;target[idx] = c;  //可以杂交生成的种子ne[idx] = h[a];h[a] = idx++;
} 
void spfa(){while(q.size()){int x = q.front();q.pop();st[x] = false;  //出队就标记for(int i = h[x]; i != -1; i = ne[i]){int y = e[i];int z = target[i];if(dis[z] > max(dis[x], dis[y]) + max(w[x], w[y])){  //用这两个顶点来更新边上的顶点zdis[z] = max(dis[x], dis[y]) + max(w[x], w[y]);if(!st[z]){q.push(z);st[z] = 1;  //入队}}} }
}
int main(){scanf("%d%d%d%d", &n, &m, &k, &t);memset(dis, 0x3f, sizeof(dis));memset(h, -1, sizeof(h));for(int i = 1; i <= n; i++) scanf("%d", &w[i]);for(int i = 1; i <= m; i++){int x;scanf("%d", &x);dis[x] = 0;  //现有的种子,最短路径为0q.push(x);st[x] = true;   //如入队了}for(int i = 1; i <= k; i++){int a, b, c;scanf("%d%d%d", &a, &b, &c);add(a, b, c);add(b, a, c);} spfa();printf("%d\n", dis[t]);return 0;
} 

我是真的不懂spfa是什么。先写到这,我饿了。

最近睡得不好,心脏都不好。我明天开始就好好养生,好好学习!!!

相关内容

热门资讯

我的耳朵在夜里能总能听到敲击声... 我的耳朵在夜里能总能听到敲击声,像打铁的声音Hhhh原来也有人和我一样的感觉,我很小的时候就觉得,安...
乘车的乘乘的成语是什么? 乘车的乘乘的成语是什么?乘风破浪、乘龙快婿、乘胜追击、乘人之危、乘虚而入、乘兴而来、乘其不备、乘热打...
娃娃晒太阳原因是什 娃娃晒太阳原因是什娃娃晒太阳原因是什... 娃娃晒太阳原因是什 展开 阳光可以促进人体对钙的吸...
请不要生气,他只是开开玩用英语... 请不要生气,他只是开开玩用英语翻译怎么写.Don't be angry,He was just ki...
驻日本大使馆提醒在日中国公民防... 来源:中国驻日本大使馆一、据日本气象厅统计,自6月21日8时至7月2日16时,鹿儿岛县吐噶喇列岛近海...
中国船舶:拟换股吸收合并中国重... 中证智能财讯中国船舶(600150)7月4日晚间公告,公司拟向中国重工(维权)(601989)全体换...
天箭科技:预计2025年上半年... 中证智能财讯天箭科技(002977)7月4日晚间披露业绩预告,预计2025年上半年归母净利润亏损59...
全红婵,确认缺席 7月4日,中国游泳协会正式公布新加坡世锦赛参赛名单,陈芋汐领衔中国跳水队出战,全红婵缺席本届比赛。据...
短到极致,也美到极致的诗词有哪... 短到极致,也美到极致的诗词有哪些?得成比目何辞死,愿作鸳鸯不羡仙。我觉得这句诗真的很美,是我们心目中...
欧洲央行管委温施:通胀的下行风... 格隆汇7月4日丨欧洲央行管委温施表示,通胀风险正指向下行而非上行。在发表讲话时,他将能源价格下跌、欧...