【算法每日一练]-图论(保姆级教程 篇6(图上dp))#最大食物链 #游走

news2025/1/11 6:02:01

目录

题目:最大食物链

解法一:

 解法二: 记忆化

题目:游走

思路: 


                

        

        

题目:最大食物链

        

解法一:

         

        

我们标记f[i]是被f[x]捕食的点对应的类食物链数

不难得出: f[x]=∑(f[i])   
首先从生产者开始,每去掉一个被捕食的点,那么相邻捕食者就要加上去掉点的类食物链数,但是我们还需要找到出度为0的消费者。
所以这道题,我们要同时记录入度,还有出度(其实单纯的topo排序就用不上出度,记录出度是为了找食物链结尾的个数)

      

        

#include<bits/stdc++.h> 
using namespace std;
const int MOD=80112002,M=500005,N=5005;
vector <int>v[N];
queue<int> q;
int n,m,ans,f[N],in[N],out[N];//我们需要标记每个点的入度和出度,f为以该点结尾的类食物链数
int main(){
	cin>>n>>m;int x,y;
	for(int i=1;i<=m;i++){
		cin>>x>>y;
		v[x].push_back(y);//y吃x,x指向y
		out[x]++;in[y]++;
	}
	for(int i=1;i<=n;i++){//找到所有没有入度的点为起点
		if(in[i]==0){q.push(i);f[i]=1;}
	}
	while(q.size()){//进行拓扑排序
		int cur=q.front();q.pop();
		for(int i=0,sz=v[cur].size();i<sz;i++){
			int t=v[cur][i];
			f[t]=(f[t]+f[cur])%MOD;in[t]--;//去掉cur点的话,就要把f[cur]加到捕食它的点上
			if(in[t]==0) q.push(t);
		}		
	}
	for(int i=1;i<=n;i++){
		if(out[i]==0)ans=(ans+f[i])%MOD;//出度为0的点的f是我们要的真正食物链数
	}
	cout<<ans;
	return 0;
}

        

        

 解法二: 记忆化


#include <bits/stdc++.h>    //记忆化搜索
using namespace std;
const int N=1e5+5;
int n,m,ans,tot,in[N],out[N],f[N];
vector<int>ve[N];
int dfs(int x){//就是从每个生产者开始,看看能到多少个最终消费者,然后记忆化,最终计算所有生产者就是答案
	if(f[x])return f[x];
	if(!out[x])return 1;
	for(int i=0,sz=ve[x].size();i<sz;i++){
		int v=ve[x][i];
		f[x]+=dfs(v);
	}
	return f[x];
}
int main(){
	cin>>n>>m;int u,v,w;
	while(m--){
		cin>>u>>v;
		ve[u].push_back(v);
		out[u]++;in[v]++;
	}
	for(int i=1;i<=n;i++){
		if(in[i]==0&&out[i])
		ans+=dfs(i);
	}
	cout<<ans;
}

        

        

题目:游走

         

思路: 

       

给一个DAG(有向无环图),求所走路径长度的期望呗!也就是:所有路径长度总和/所有路径个数(因为每条路径概率都一样嘛)

        

明明是DAG图,topo一下完事了

我们设置g[i]表示以i为终点的路径数,f[i]表示i为终点的长度和

   
topo:从点j到一个点i,则g[i]+=g[j],f[i]+=f[j]+g[i](因为啊,从j到i的每个路径长度都只增加1就行,一共增加了g[i])

        
最后就是求(L/S)MOD,也就是L*(S^(MOD-2))MOD即可(逆元小知识)

        

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN=1e5+5;
const ll MOD=998244353;
int n,m;
ll L,S,f[MAXN],g[MAXN];//g[i]表示以i为终点的路径数,f[i]表示i为终点的长度和
vector<int> edge[MAXN];
int in[MAXN],vis[MAXN];
void topo(void)//模板
{
	queue<int> q;
	for(int i=1;i<=n;i++)
    if(!in[i]) q.push(i);
    while(!q.empty())
    {
    	const int u=q.front();
    	q.pop();
    	if(vis[u]) continue; vis[u]=true;//没有环,所以这句话可以不要
    	for(auto v:edge[u])
    	{
    		in[v]--;
    		if(!in[v])	q.push(v);
    		f[v]=(f[v]+f[u]+g[u])%MOD;
    		g[v]=(g[v]+g[u])%MOD;
		}
	}
}

ll qpow(ll base,ll k)//快速幂求逆元
{
	ll res=1;
	while(k)
	{
		if(k&1)	res=res*base%MOD;
		base=base*base%MOD;
		k>>=1;
	}
	return res;
}

int main()
{
	cin>>n>>m;
	for(int i=1;i<=m;i++)
	{
		int u,v;cin>>u>>v;
		edge[u].push_back(v);
		in[v]++;
	}
	for(int i=1;i<=n;i++)	g[i]=1;//初始化:每个点的路径数初始化为1
	topo();
	for(int i=1;i<=n;i++)	L=(L+f[i])%MOD;//获取最终的总长度
	for(int i=1;i<=n;i++)	S=(S+g[i])%MOD;//获取最终的路径个数
	cout<<(L*qpow(S,MOD-2))%MOD<<endl;//求L/S的结果,即L*S的逆元,即L*S^(MOD-2)
	return 0;
}

提一嘴: 

                    
为什么要引入逆元呢?
因为(a+b)%MOD=(a%MOD+b%MOD)%MOD
(a-b)%MOD=(a%MOD-b%MOD)%MOD
(a*b)%MOD=(a%MOD*b%MOD)%MOD  

            
但是除法不满足,我们要求(a/b)%p=1等价于(a*x)%p=1,这个x就是b的乘法逆元(可以理解成x为1/b),也就是(b*x)%p=1

        
再引入费马小定理:假如a和p互质,那么a^(p-1)=1(%p),故a*a^(p-2)=1(%p),故a的逆元x=a^(p-2)%p
因此:(x/y)%p等价于x*y^(p-2)%p,注意每乘一次就要去一次模
     

        

        

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1275000.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ECU Bootloader程序开发

文章目录 前言前言 本篇主要介绍ECU BootLoader的“一般”开发逻辑,对正在、或想要做这方面工作的小伙伴一个参考。 BootLoader的稳定性至关重要,前期设计要规避可能存在的所有使ECU变“砖”的风险点,尽可能地设计多重防范机制,并做充分的正常、异常测试,才能保证量产的质…

由于找不到xapofx1_5.dll,无法继续执行代码的修复方法

在使用特定应用程序或游戏时&#xff0c;Windows 用户可能会遇到一个错误消息“由于找不到xapofx1_5.dll,无法继续执行代码”&#xff0c;这可能导致程序无法启动。本文将介绍解决此问题的多种方法&#xff0c;并对比各自的优点和缺点。 一.什么是xapofx1_5.dll xapofx1_5.dll…

WhatsApp群发消息脚本功能介绍及代码分享!

随着社交媒体的普及&#xff0c;通讯应用成为了人们日常沟通的主要工具之一&#xff0c;其中&#xff0c;WhatsApp凭借其简洁、易用的特点&#xff0c;成为了全球广受欢迎的通讯应用之一&#xff0c;除了基础的聊天功能&#xff0c;WhatsApp还提供了一系列辅助工具功能&#xf…

二十年前就在用的DDOS攻击,为什么一直不过时?

文章目录 一、DOS二、DDOS三、如何防范DDOS1.可以过滤IP地址2.增加设备3.在骨干节点配置防火墙4.开启过滤5.配置DNS抗攻击6.白帽团队 四、白帽子 为什么二十年前中国红客们就在用的DDOS攻击直到现在还依然是黑客们最爱的攻击方法&#xff1f;二十年前的攻击技术为什么还不过时&…

JavaScript类型判断:解密变量真实身份的神奇技巧

文章目录 1. typeof运算符2. instanceof运算符3. Object.prototype.toString4. Array.isArray5. 使用constructor属性6. 使用Symbol.toStringTag7. 使用is类型判断库8. 谨慎使用隐式类型转换结语 &#x1f389;JavaScript类型判断&#xff1a;解密变量真实身份的神奇技巧 ☆* o…

文献速递:(第三部分)— (超声非破坏性评估中合成和增强训练数据生成与评估的最新研究进展)

文献速递&#xff1a;&#xff08;第三部分&#xff09;— &#xff08;超声非破坏性评估中合成和增强训练数据生成与评估的最新研究进展&#xff09; Title 题目 A review of synthetic and augmented training data for machine learning in ultrasonic non-destructive e…

手持收银机|移动收银机|POS终端安卓主板定制方案

智能手持收银机是一种集成了显示触摸控制、IC卡刷卡、磁条卡刷卡、二维码扫码、票据打印等多种功能的设备。它支持支付宝、微信、银行卡、云闪付、Apple Pay、会员储值等多种支付方式&#xff0c;为商家和用户提供了更便捷的支付体验。 该手持收银机采用了联发科MTK6761平台开发…

已知数组A[1..n]中元素类型为非负整数,设计算法将其调整为左右两部分,左边所有为奇数,右边所有为偶数,并要求算法的时间复杂度为O(n)

//左边奇数右边偶数 void Swap(int* a, int* b) {int tmp *b;*b *a;*a tmp; } void LeftRight(int arr[],int n) {int i 0;int j n - 1;while(i<j){if (arr[i] % 2 0 && arr[j] % 2 1) {Swap(&arr[i], &arr[j]);i;j--;}else if (arr[i] % 2 1 &…

小众实用的Python 爬虫库RoboBrowser推荐

文章目录 前言安装及用法实战一下1 打开目标网站2 自动化表单提交3 数据爬取 最后关于Python技术储备一、Python所有方向的学习路线二、Python基础学习视频三、精品Python学习书籍四、Python工具包项目源码合集①Python工具包②Python实战案例③Python小游戏源码五、面试资料六…

JSP格式化标签 formatDate日期格式转换

我们继续来讲格式化标签 formatDate 这个标签 作用是 将一个date时间类型的值转成指定格式的字符串 语法格式如下 value 是需要格式化的数据 type 是确定你要转什么类型的数据 这里有 日期型 时间型 日期时间型 dateStyle 专门用来设置日期格式 timestyle 的话 是专门用来设…

STM32 自定义UART数据格式(串口通信点亮LED实验)

起始位&#xff1a;0xaa告诉机器我们要开始传输数据了。 校验位&#xff1a;等于前几项数据位的相加。 结束位&#xff1a;结束传输。 自定义UART数据格式&#xff1a; 1》CPU与CPU之间 2》外设与CPU之间 这里举例&#xff0c;利用串口调试助手发送一串数据&#xff0c;…

Qt实现右键菜单

一、实现方法 QWidget提供了虚函数: virtual void contextMenuEvent(QContextMenuEvent*event);覆写该函数&#xff0c;即可。 二、Example 创建一个基本的mainwindow项目&#xff0c; 头文件&#xff1a; class MainWindow : public QMainWindow {Q_OBJECTpublic:MainWin…

在AI时代我们的必修课,从提示词工程到大语言模型,进行多场景实战的经验分享总结

在AI时代&#xff0c;我们正面临着与人工智能技术日益紧密相连的需求和挑战。无论是从事计算机科学、数据科学还是其他相关领域的人员&#xff0c;都需要掌握AI技术&#xff0c;并将其应用到不同的场景中。本文将分享一些经验总结&#xff0c;帮助读者在多种场景下进行AI实战。…

linux NAT网卡配置static

由于是内网&#xff0c;资料无法拷贝&#xff0c;借助参考资料&#xff0c;整理发出。 镜像安装 基本操作。 查看VM配置 图1&#xff0c;有几个信息。一个是NAT借用了网卡里的VMnet8适配器。 子网IP是从192.168.142.0 子网掩码255.255.255.255&#xff0c;对应下面配置的N…

后端项目操作数据库增删改查-使用MyBatis配置实现数据操作

一、创建一个数据表对应的实体类 在src/main/java/包名/路径下新建pojo.entity文件夹&#xff0c;如com.luoyang.small.pojo.entity&#xff0c;并在该文件夹下新增实体类java文件&#xff1a;如相册Album.java 该实体类的属性应与数据表的字段对应 数据表样例如下&#xff1a…

Quest 3圆满结束,500万SUI正在送出!

的最后审核阶段于北京时间今日凌晨结束并公布了获奖情况&#xff0c;此次总奖池金额达到了500万SUI&#xff0c;共有16.4万参与者获得了奖励。参与Quest 3的人们可以在Sui网络上的九款游戏中进行选择&#xff0c;并利用促销活动注册SuiNS域名。 Quest 3以游戏为重点&#xff0…

spring boot 2 升级到 spring boot 3 后文件上传失败

背景 项目需要&#xff0c;要求升级 spring boot 2.7 到 spring boot 3.2&#xff0c;升级过程中发现很多不兼容问题&#xff0c;下面说明文件上传失败的解决方案。 问题 spring boot 2 中不需要额外的配置&#xff0c;直接在 Controller 中配置 MultipartFile 接收页面传的…

python之pyqt专栏5-信号与槽1

在上一篇文章&#xff0c;我们了解到如果想要用代码改变QLabel的文本内容&#xff0c;可以调用QLabel类的text()函数。 但是现在有个这样的需求&#xff0c;界面中有一个Button与一个Label&#xff0c;当点击Button时&#xff0c;将Label的内容改变为“Hello world&#xff01;…

全网最新最全面的Jmeter接口测试:jmeter模拟http请求实战

1、get请求 http://www.hnxmxit.com/ 2、带参数的get请求 微信公众号获取token请求 3、自定义头部信息的请求 百度搜索请求 https://www.baidu.com/s?wd猫 4、post请求 微信公众号添加用户标签请求 注&#xff1a;post请求中如果body中的数据为json,一定要在信息头管理器中…

【SpringBoot系列】SpringBoot时间字段格式化

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…