XSY5049 哈密顿(hamilton)

《有人看完题10分钟就想到了正解但是不会实现花了赛时114514h从10分优化到40分再优化到100分做法但是被卡常于是又在赛后花费1h卡过去了是谁我不说。》

题目大意

给你一个 n n n 个点的无向图,你要求出任意两个点之间是否存在哈密顿路径。

n ≤ 24 n ≤24 n24

题解

赛时心路历程如上文所述。

注意到 n n n 只有 24,很自然想到状压。

首先钦定起点,枚举可能的不经过重复点的路径集合,复杂度为 O ( n 2 n ) O(n2^n) O(n2n)

然后枚举每个集合,判断一下两个集合拼在一起能否组成哈密顿路径即可。总复杂度 O ( n 2 n ) O(n2^n) O(n2n)

被卡常了。由于卡常技巧不精,最后使用火车头大法才跑过去,并且跑得飞快。

Code

//怎么全世界就我被卡常?
//卡你的母牛呢 
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize("Ofast")
#pragma GCC optimize("inline")
#pragma GCC optimize("-fgcse")
#pragma GCC optimize("-fgcse-lm")
#pragma GCC optimize("-fipa-sra")
#pragma GCC optimize("-ftree-pre")
#pragma GCC optimize("-ftree-vrp")
#pragma GCC optimize("-fpeephole2")
#pragma GCC optimize("-ffast-math")
#pragma GCC optimize("-fsched-spec")
#pragma GCC optimize("unroll-loops")
#pragma GCC optimize("-falign-jumps")
#pragma GCC optimize("-falign-loops")
#pragma GCC optimize("-falign-labels")
#pragma GCC optimize("-fdevirtualize")
#pragma GCC optimize("-fcaller-saves")
#pragma GCC optimize("-fcrossjumping")
#pragma GCC optimize("-fthread-jumps")
#pragma GCC optimize("-funroll-loops")
#pragma GCC optimize("-fwhole-program")
#pragma GCC optimize("-freorder-blocks")
#pragma GCC optimize("-fschedule-insns")
#pragma GCC optimize("inline-functions")
#pragma GCC optimize("-ftree-tail-merge")
#pragma GCC optimize("-fschedule-insns2")
#pragma GCC optimize("-fstrict-aliasing")
#pragma GCC optimize("-fstrict-overflow")
#pragma GCC optimize("-falign-functions")
#pragma GCC optimize("-fcse-skip-blocks")
#pragma GCC optimize("-fcse-follow-jumps")
#pragma GCC optimize("-fsched-interblock")
#pragma GCC optimize("-fpartial-inlining")
#pragma GCC optimize("no-stack-protector")
#pragma GCC optimize("-freorder-functions")
#pragma GCC optimize("-findirect-inlining")
#pragma GCC optimize("-fhoist-adjacent-loads")
#pragma GCC optimize("-frerun-cse-after-loop")
#pragma GCC optimize("inline-small-functions")
#pragma GCC optimize("-finline-small-functions")
#pragma GCC optimize("-ftree-switch-conversion")
#pragma GCC optimize("-foptimize-sibling-calls")
#pragma GCC optimize("-fexpensive-optimizations")
#pragma GCC optimize("-funsafe-loop-optimizations")
#pragma GCC optimize("inline-functions-called-once")
#pragma GCC optimize("-fdelete-null-pointer-checks")
#include<bits/stdc++.h>
using namespace std;

const int N=24+1,M=(1<<24)+1;

int n,K,a[N],ans[N],f[M],g[M];

int main(){
	scanf("%d",&n); 
	for(register int i=1;i<=n;i++)
		for(register int j=1;j<=n;j++){
			char s=getchar();
			while(s!='1'&&s!='0') s=getchar();
			if(s=='1') a[i]|=1<<j-1;
		}
	K=(1<<n);
	f[1]|=1;g[1]|=a[1];
	for(register int i=1;i<K;i++)
		for(register int j=1;j<=n;j++){
			if(!(g[i]>>j-1&1)||(i>>j-1&1)) continue;
			f[i|(1<<(j-1))]|=1<<j-1;
			g[i|(1<<(j-1))]|=a[j];
		}
	for(register int i=1;i<K;i++){
		for(register int j=1;j<=n;j++){
			if(!(i>>j-1&1)||!(f[i]>>j-1&1)) continue;
			ans[j]|=f[K-i];
		}
	}
	for(register int i=1;i<=n;i++){
		for(register int j=1;j<=n;j++)
			printf("%d",ans[i]>>j-1&1);
		putchar('\n');
	}
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值