P3199 【[HNOI2009]最小圈】

疑似三倍经验

因为和机房一些大佬一起做的这道题,所以emmm他们貌似也写了题解,在做这道题的时候也参照了其他大佬写的一些题解,所以如果程序有雷同请见谅(手动鞠躬)


题目也是莫名其妙地给了一大串数学式,简洁地重新说一下题目

给你一张图,图中有环,定义一个环的平均值为环的边权和÷环中点的个数,那么就应该有了一中非常暴力的思路

  • 想办法找出来图中所有的环并求出其平均值,再比较出最小值

这个方法确实怎么想都很暴力,但是一点也不好实现,可能是我太弱,我实在想不太出来有什么算法可以找出来所有的环,所以这种思路先给我PASS掉

而正解呢?应该是二分答案。为什么?

  • 既然我们找不出来环,我们逆向思维,直接枚举平均值,看是否会有一个环符合条件,那么这道题就变成了:二分枚举平均值,找到是否有环符合条件

若我们此时枚举的平均值为 a n s ans ans,有 k k k个字符串,那么就有

a n s ∗ k = l e n 1 + l e n 2 + l e n 3 + . . . + l e n k ans * k = len1 + len2 + len3 + ... + lenk ans∗k=len1+len2+len3+...+lenk

那道这个式子之后,我们对它进行移项

0 = l e n 1 − a n s + l e n 2 − a n s + l e n 3 − a n s + . . . + l e n k − a n s 0=len1-ans+len2-ans+len3-ans+...+lenk-ans 0=len1−ans+len2−ans+len3−ans+...+lenk−ans

那么对于满足以下式子,就可以判断是环了,所以在跑 S P F A SPFA SPFA更新距离的时候,就应该像下面这样

0 ≤ l e n 1 − a n s + l e n 2 − a n s + l e n 3 − a n s + . . . + l e n k − a n s 0 \leq len1-ans+len2-ans+len3-ans+...+lenk-ans 0≤len1−ans+len2−ans+len3−ans+...+lenk−ans

所以这里就直接给程序了(三道题的)

SP2885 WORDRING - Word Rings

P3199 [HNOI2009]最小圈

复制代码
#include<bits/stdc++.h>
using namespace std;
const int MAXN=30000000+51;
const double INF=(1e5)*1.0;
const double eqs=1e-9;
int n,m;
struct node {
	int net,to;
	double z;
} e[MAXN];
int head[MAXN],tot;
void add(int x,int y,double z) {
	e[++tot].net=head[x];
	e[tot].to=y;
	e[tot].z=z;
	head[x]=tot;
}

double d[MAXN];
bool v[MAXN],flag;
bool spfa(int x,double k) {
	v[x]=true;
	for(register int i=head[x]; i; i=e[i].net) {
		int y=e[i].to;
		double z=e[i].z;
		if(d[y]>d[x]+z-k) {
			d[y]=d[x]+z-k;
			if(v[y]==true||spfa(y,k)==true) return true;
		}
	}
	v[x]=false;
	return false;
}
bool check(double x) {
	for(register int i=1; i<=n; i++) {
		d[i]=20040915;
		v[i]=false;
	}
	for(register int i=1; i<=n; i++) {
		if(spfa(i,x)==true) return true;
	}
	return false;
}
int main() {
	scanf("%d%d",&n,&m);
	for(register int i=1; i<=m; i++) {
		int x,y;
		double z;
		scanf("%d%d%lf",&x,&y,&z);
		add(x,y,z);
	}
	double l=-INF,r=INF;
	while(r-l>eqs) {
		double mid=(l+r)/2;
		if(check(mid)==true) r=mid;
		else l=mid;
	}
	printf("%.8lf",l);
	return 0;
}

UVA11090 Going in Cycle!!

复制代码
#include <bits/stdc++.h>
using namespace std;
int T,n,m,u,v,w,tot;
double dis[520010];
int vis[520010],head[520010];

struct node {
	int to,net;
	double val;
} e[520010];

inline void add(int u,int v,double w) {
	e[++tot].to=v;
	e[tot].val=w;
	e[tot].net=head[u];
	head[u]=tot;
}

inline bool dfs(int now,double x) {
	vis[now]=1;
	for(register int i=head[now];i;i=e[i].net) {
		int v=e[i].to;
		if(dis[v]>dis[now]+e[i].val-x) {
			dis[v]=dis[now]+e[i].val-x;
			if(vis[v]==1||dfs(v,x)==true) return true;
		}
	}
	vis[now]=0;
	return false;
}

inline bool check(double x) {
	for(register int i=1;i<=n;i++) {
		vis[i]=0;
		dis[i]=20050206;
	}
	for(register int i=1;i<=n;i++) {
		if(dfs(i,x)==true) return true;
	}
	return false;
}

int main() {
	scanf("%d",&T);
	for(register int k=1;k<=T;k++) {
		tot=0;
		for(register int i=1;i<=n;i++) head[i]=0;
		scanf("%d%d",&n,&m);
		for(register int i=1;i<=m;i++) {
			scanf("%d%d%d",&u,&v,&w);
			add(u,v,w);
		}
		double l=-10000000,r=10000000;
		while(r-l>1e-10) {
			double mid=(l+r)/2;
			if(check(mid)==true) r=mid;
			else l=mid;
		}
		printf("Case #%d: ",k);
		if(l==10000000) puts("No cycle found.");
		else printf("%.2lf\n",l);
	}
	return 0;
}
相关推荐
不如自挂东南吱10 分钟前
空间相关性 和 怎么捕捉空间相关性
人工智能·深度学习·算法·机器学习·时序数据库
荒诞硬汉23 分钟前
对象数组.
java·数据结构
洛生&31 分钟前
Elevator Rides
算法
老歌老听老掉牙31 分钟前
优化样条曲线拟合参数解决三维建模中的截面连续性问题
c++·opencascade·样条曲线
2501_9335130440 分钟前
关于一种计数的讨论、ARC212C Solution
算法
Wu_Dylan41 分钟前
智能体系列(二):规划(Planning):从 CoT、ToT 到动态采样与搜索
人工智能·算法
散峰而望1 小时前
【算法竞赛】栈和 stack
开发语言·数据结构·c++·算法·leetcode·github·推荐算法
知乎的哥廷根数学学派1 小时前
基于多尺度注意力机制融合连续小波变换与原型网络的滚动轴承小样本故障诊断方法(Pytorch)
网络·人工智能·pytorch·python·深度学习·算法·机器学习
蚊子码农1 小时前
算法题解记录-208实现Trie前缀树
运维·服务器·算法
2301_800256111 小时前
【人工智能引论期末复习】第3章 搜索求解2 - 对抗搜索
人工智能·算法·深度优先