一个例子教你如何与出题人斗智斗勇
时间:2022-05-08
本文章向大家介绍一个例子教你如何与出题人斗智斗勇,主要内容包括P3385 【模板】负环、Try Second、Try Third、当然去了考场上还是乖乖的写dfs的SPFA吧,毕竟到时候可就只有一次机会啊、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。
我以前出过一道题,卡了10种贪心,但还是被第11种贪心A了,
一道题不会做?贪嘛,能怎么贪怎么贪,想怎么贪怎么贪!
现在NOIP题目的数据给的不都很明确嘛,简单,对着数据,一个一个的贪!
—By 贪心之神CCL
今天做了一道很bt的题
P3385 【模板】负环
这题居然卡广搜SPFA,哎作为一个只会写广搜SPFA的我就不服了 :triumph:
卡我者,必被我贪!:confused:
Try First
判断负环嘛,有一个很显然的结论,当你从开始点出发浪了一圈之后又回到开始点,发现还能继续更新,那就一定有负环!
本来以为它给的数据有那种多个联通分量的情况,但事实证明我想多了。:sweat_smile:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
using namespace std;
const int MAXN=1e6+10;
inline int read()
{
char c=getchar();int f=1,x=0;
while(c<'0'||c>'9') {if(c=='-') f=-1;c=getchar();}
while(c>='0'&&c<='9') x=x*10+c-48,c=getchar();return x*f;
}
struct node
{
int u,v,w,nxt;
}edge[MAXN];
int head[MAXN];
int num=1;
inline void add_edge(int x,int y,int z)
{
edge[num].u=x;
edge[num].v=y;
edge[num].w=z;
edge[num].nxt=head[x];
head[x]=num++;
}
int inger[MAXN];
int vis[MAXN];
int dis[MAXN];
inline void pre()
{
memset(head,-1,sizeof(head));
num=1;
memset(vis,0,sizeof(vis));
memset(dis,0x7f,sizeof(dis));
}
int n,m;
inline void SPFA()
{
queue<int>q;
q.push(1);
vis[1]=1;dis[1]=0;
while(q.size()!=0)
{
int p=q.front();
q.pop();
vis[p]=0;
for(int i=head[p];i!=-1;i=edge[i].nxt)
{
if(dis[edge[i].v]>dis[edge[i].u]+edge[i].w)
{
dis[edge[i].v]=dis[edge[i].u]+edge[i].w;
if(vis[edge[i].v]==0)
{
vis[edge[i].v]=1;
if(edge[i].v==1)
{
printf("YE5n");
return ;
}
q.push(edge[i].v);
}
}
}
}
printf("N0n");
}
int main()
{
int T=read();
while(T--)
{
pre();
n=read();m=read();
for(int i=1;i<=m;i++)
{
int x=read(),y=read(),z=read();
if(z<0) add_edge(x,y,z);
else add_edge(x,y,z),add_edge(y,x,z);
}
SPFA();
}
return 0;
}
这样就可以A啦
不过这个貌似有点借鉴dfs的思想了,
它卡的是入队次数,
正常情况下只有一个点入队次数超过n的时候才有负环,但这个太慢了,
我们考虑缩小一下这个n
Try Second
n太大了,那怎么办呢?
我们直接随机一个就好啦:grin:
然而。。:dizzy_face: :dizzy_face: :dizzy_face: :dizzy_face: :dizzy_face:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
#include<ctime>
#include<cstdlib>
using namespace std;
const int MAXN=1e6+10;
inline int read()
{
char c=getchar();int f=1,x=0;
while(c<'0'||c>'9') {if(c=='-') f=-1;c=getchar();}
while(c>='0'&&c<='9') x=x*10+c-48,c=getchar();return x*f;
}
struct node
{
int u,v,w,nxt;
}edge[MAXN];
int head[MAXN];
int num=1;
inline void add_edge(int x,int y,int z)
{
edge[num].u=x;
edge[num].v=y;
edge[num].w=z;
edge[num].nxt=head[x];
head[x]=num++;
}
int inger[MAXN];
int vis[MAXN];
int dis[MAXN];
inline void pre()
{
memset(head,-1,sizeof(head));
num=1;
memset(inger,0,sizeof(inger));
memset(vis,0,sizeof(vis));
memset(dis,0x7f,sizeof(dis));
}
int n,m,Attack;
inline void SPFA()
{
queue<int>q;
q.push(1);
vis[1]=1;dis[1]=0;
while(q.size()!=0)
{
int p=q.front();
q.pop();
vis[p]=0;
for(int i=head[p];i!=-1;i=edge[i].nxt)
{
if(dis[edge[i].v]>dis[edge[i].u]+edge[i].w)
{
dis[edge[i].v]=dis[edge[i].u]+edge[i].w;
if(vis[edge[i].v]==0)
{
vis[edge[i].v]=1;
inger[edge[i].v]++;
if(inger[edge[i].v]>=Attack)
{
printf("YE5n");
return ;
}
q.push(edge[i].v);
}
}
}
}
printf("N0n");
}
int main()
{
srand((unsigned)time(NULL));
int T=read();
while(T--)
{
pre();
n=read();m=read();
Attack=rand()%n^rand()%n&rand()%n;
for(int i=1;i<=m;i++)
{
int x=read(),y=read(),z=read();
if(z<0) add_edge(x,y,z);
else add_edge(x,y,z),add_edge(y,x,z);
}
SPFA();
}
return 0;
}
Try Third
看来rand是不行了
我们来指定一下吧
像这样:
if(inger[edge[i].v]>=250)
{
printf("YE5n");
return ;
}
250
200
100
。。。。:angry:
20
:grinning: :grinning: :grinning: :grinning:
想了一下,
SPFA的时间复杂度为:O(ke)
其中的k是平均入队次数,e是边数,k的值一般稳定在2左右,所以说,当一个点的入队次数大于20的话,十有八九就是出现负环了
当然去了考场上还是乖乖的写dfs的SPFA吧,毕竟到时候可就只有一次机会啊
- Hadoop 多表 join:map side join 范例
- 实战 windows7 下 eclipse 远程调试 linux hadoop
- Hive 在多维统计分析中的应用 & 技巧总结
- shell 学习笔记(18)
- Hive 中的复合数据结构简介以及一些函数的用法说明
- BloomFilter 简介及在 Hadoop reduce side join 中的应用
- 关于 hadoop reduce 阶段遍历 Iterable 的 2 个“坑”
- Hadoop Mapper 阶段将数据直接从 HDFS 导入 Hbase
- 译文 | 量化投资教程:投资组合优化与R实践
- 浅谈 java 中构建可执行 jar 包的几种方式
- python 日志模块 logging 详解
- 基于堆实现的优先级队列:PriorityQueue 解决 Top K 问题
- explain 深入剖析 MySQL 索引及其性能优化指南
- 图文并茂详解 SQL JOIN
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 动态规划算法练习(4)--medium
- Python基础面向对象-其他
- Python基础之面向对象基础知识
- Python基础之文件操作知识
- 前端|利用js实现在日历中的签到效果
- Python基础之函数加强
- Python基础之函数二
- 前端|动态发光按钮
- Windows系统命令拼接
- 利用Hexo打造属于自己的网页版简历
- RCE远程控制Windows服务器——以win10虚拟机为例
- 解决InnoDB: Table mysql/innodb_index_stats has length mismatch
- R语言中对文本数据进行主题模型topic modeling分析
- QT学习第3天:QSlider使用方法
- qt学习第2天:QRadioButtonTest+ButtonGroup单选后提示消息,QComBox