HDOJ题目页面传送门

给定一个无向带权图\(G=(V,E),|V|=n,|E|=m\),求边权之和最大的有\(s\)个节点的链的边权之和,即求\(\max\limits_{\forall i\in[1,s],\forall j\in(i,s],a_i\ne a_j,\forall i\in[1,s),(a_i,a_{i+1},l_i)\in E}\left\{\sum\limits_{i=1}^{s-1}l_i\right\}\)。如果没有符合要求的链,输出\(\texttt{impossible}\)。

\(n\in\left[2,10^4\right],m\in\left[1,10^4\right],s\in[2,6]\)。本题多测,最多有\(35\)组数据,\(\max(n,m)>100\)的最多有\(5\)组。时限\(15\mathrm s\)。

先说\(1\)种不是正解的玄学方法

暴搜。。。时间复杂度\(\mathrm O(\mathrm C_n^s)\),但很显然非常跑不满,因为这是个稀疏图。而且还可以最优性剪枝优化,假设所有边中最大的边权为\(longest\),当前的最大答案为\(ans\),当前还剩\(rst\)个节点(条边)要搜,目前的边权之和为\(now\),那么如果\(now+rst\times longest\le ans\),便可立即停止搜索。这样就可以水过去了。。。

这不是主要内容,时间复杂度到底是多少就不研究了,代码也不贴了。


正解

考虑对一个比原问题弱一点的问题求解:给每个节点\(i\)染一个\([0,s)\)的颜色\(col_i\),求边权之和最大的有\(s\)个节点且这\(s\)个节点的颜色是\(0\sim s-1\)的一个排列的链的边权之和。这个问题很好求,状压DP就可以了。设\(dp_{i,j}\)表示边权之和最大的以\(i\)为\(1\)个端点,上面节点颜色互不相同且bitmask为\(j\)的链的边权之和,边界为\(dp_{i,\{j\}}=\begin{cases}0&i=j\\-\infty&i\ne j\end{cases}\),状态转移方程为\(dp_{i,j}=\begin{cases}\max\limits_{(i,k,l)\in E}\left\{dp_{k,j-\{col_i\}}+l\right\}&col_i\in j\\-\infty&col_i\notin j\end{cases}\),最终答案为\(\max\limits_{i=1}^{n}\{dp_{i,[1,n]\cap\mathbb Z}\}\)。DP顺序要注意,要按照\(j\)的递增顺序DP,不能按照\(i\)的顺序(我就在上面WA过)。(简单DP

那么既然这个弱问题这么好求,那我们就强制给每个节点染色。假设我们给每个节点随机染色,那我们来分析一下弱问题的答案就是原问题的正确答案的概率:如果原问题所求得的最长链,在弱问题中,上面所有节点的颜色正好组成\(0\sim s-1\)的一个排列,那么弱问题的答案就是原问题的答案,而在所有\(s^s\)种最长链的颜色序列中,有\(s!\)种是\(0\sim s-1\)的排列,所以概率为\(\dfrac{s!}{s^s}\)。当\(s=6\)时,概率最小为\(\dfrac{6!}{6^6}\approx1.5\%\)。那么我们只需要在\(35\)组数据中每组随机染色、求解足够多次,就有足够大的的概率在每组中都至少有一次弱问题的答案是原问题的答案,这样把每次弱问题的答案\(\max\)起来即可。我们希望这个概率至少为\(99\%\),设随机\(x\)次,那么可以列出不等式\(\left(1-\left(1-\dfrac{6!}{6^6}\right)^x\right)^{35}\ge99\%\)。不难解出\(x\ge524.397\cdots\)。但由于HDOJ比较卡常,随机\(525\)次会TLE,于是我们只能用正确率换时间,只随机\(300\)次,这样才能勉强卡进\(15\mathrm s\)的时限。此时正确率为\(\left(1-\left(1-\dfrac{6!}{6^6}\right)^{300}\right)^{35}\approx71.8\%\),侥幸过。

btw,随机的时候不能写rand()*rand()%s。第一,在某些系统下两个rand()乘起来会爆int;第二,把两个随机数乘起来,期望因数个数会增加很多,\(\bmod s\)的结果等于\(0\)的概率也就会增加很多,就不均匀了。可以这样写:abs(rand()*rand()*rand()+rand())%s

UPD 2020.4.12:你看看,ycx,你看看,上面那段删除线写的是smjbwy,狗屁不通。。看来我还是太年轻了。昨天经hsc神仙的提醒才知道,abs(rand()*rand()*rand()+rand())这样的写法会生成出来分布极度不均匀的随机数,虽然在本题里没事,但显然不应该这么写。不如直接用mt19937,先定义一个随机数生成器mt19937 rng;,那么\([l,r]\)之间的随机数就是uniform_int_distribution<>(l,r)(rng)

下面贴正解的代码:UPDed 2020.4.12

#include<bits/stdc++.h>
using namespace std;
#define int long long//爆int
#define pb push_back
#define mp make_pair
#define X first
#define Y second
#define uid uniform_int_distribution
const int inf=0x3f3f3f3f3f3f3f3f;
mt19937 rng(20060617/*信仰优化*/);
int ppc(int x){return __builtin_popcount(x);}
const int N=10000,S=6;
int n/*节点数*/,m/*边数*/,s/*要求的链的节点数*/;
vector<pair<int,int> > nei[N+1];//邻接表
int col[N+1];//颜色
int ans;//原问题答案
int dp[N+1][1<<S];
void random(){
for(int i=1;i<=n;i++)col[i]=uid<>(0,s-1)(rng);//随机染色
for(int i=1;i<=n;i++)for(int j=1;j<1<<s;j++)dp[i][j]=j==1<<col[i]?0:-inf;//处理边界,顺便初始化
for(int j=1;j<1<<s;j++)for(int i=1;i<=n;i++)if(j&1<<col[i]&&ppc(j)>1/*大小为1的bitmask是边界*/)//枚举状态
for(int k=0;k<nei[i].size();k++){//转移
int x=nei[i][k].X,len=nei[i][k].Y;
dp[i][j]=max(dp[i][j],dp[x][j^1<<col[i]]+len);
}
for(int i=1;i<=n;i++)ans=max(ans,dp[i][(1<<s)-1]);//把每次弱问题的答案max起来就有大概率是原问题的答案
}
void mian(){
cin>>n>>m>>s;
for(int i=1;i<=n;i++)nei[i].clear();//数据不清空,爆零两行泪
while(m--){
int x,y,z;
scanf("%lld%lld%lld",&x,&y,&z);
nei[x].pb(mp(y,z));nei[y].pb(mp(x,z));
}
ans=-inf;//初始化
int randomnum=300;//随机次数
while(randomnum--)random();//随机
if(ans>=0)cout<<ans<<"\n";
else puts("impossible");
}
signed main(){
int testnum;//数据组数
cin>>testnum;
while(testnum--)mian();
return 0;
}

HDOJ 6664 Andy and Maze的更多相关文章

  1. 【HDOJ6664】Andy and Maze(color coding)

    题意:给定一张n点m边的无向带权图,问从任意结点出发,不能走已经经过的点,共经过k个点的最长路径的值 n,m<=1e4,k<=6 思路:color coding算法 考虑每次给每个点随机编 ...

  2. 2019DX#8

    Solved Pro.ID Title Ratio(Accepted / Submitted)   1001 Acesrc and Cube Hypernet 7.32%(3/41)   1002 A ...

  3. hdoj 5094 Maze 【BFS + 状态压缩】 【好多坑】

    Maze Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 100000/100000 K (Java/Others) Total Sub ...

  4. HDU 4067 hdoj 4067 Random Maze 最小费用流

    给出n个点,m条边,入口s和出口t,对于每条边有两个值a,b,如果保留这条边需要花费:否则,移除这条边需要花费b. 题目要求用最小费用构造一个有向图满足以下条件: 1.只有一个入口和出口 2.所有路都 ...

  5. 【HDOJ】1484 Basic wall maze

    BFS. /* 1484 */ #include <iostream> #include <queue> #include <string> #include &l ...

  6. BFS+贪心 HDOJ 5335 Walk Out

    题目传送门 /* 题意:求从(1, 1)走到(n, m)的二进制路径值最小 BFS+贪心:按照标程的作法,首先BFS搜索所有相邻0的位置,直到1出现.接下去从最靠近终点的1开始, 每一次走一步,不走回 ...

  7. 最大流增广路(KM算法) HDOJ 1533 Going Home

    题目传送门 /* 最小费用流:KM算法是求最大流,只要w = -w就可以了,很经典的方法 */ #include <cstdio> #include <cmath> #incl ...

  8. BFS HDOJ 1728 逃离迷宫

    题目传送门 /* BFS:三维BFS,加上方向.用dp[x][y][d]记录当前需要的最少转向数 */ #include <cstdio> #include <algorithm&g ...

  9. 离散化+BFS HDOJ 4444 Walk

    题目传送门 /* 题意:问一个点到另一个点的最少转向次数. 坐标离散化+BFS:因为数据很大,先对坐标离散化后,三维(有方向的)BFS 关键理解坐标离散化,BFS部分可参考HDOJ_1728 */ # ...

随机推荐

  1. Shiro RememberMe 1.2.4 反序列化漏洞详细复现

    0x00 前言 今天上班的时候收到了一个复测的任务,打开一看,shiro反序列化漏洞,What?这是个什么东西,经百度查找后才知道,原来是Java的开发框架,好吧,还是没听说过..由于初测报告上的过程 ...

  2. 网传英特尔酷睿第十代桌面处理器(Comet Lake 14nm)规格

    自从农企(AMD)2016年开始崛起时,牙膏厂(英特尔)就开始发力,陆续两代推出性价比颇高的桌面处理器, 第八代.第九代酷睿桌面处理器相当的给力,而第十代酷睿桌面处理器会很猛啊,据传从酷睿i3到酷睿i ...

  3. maven 配置文件

    <properties> <project.builder.sourcesEncoding>UTF-8</project.builder.sourcesEncoding& ...

  4. Java 为什么需要包装类,如何使用包装类?

    出处:https://cloud.tencent.com/developer/article/1362754

  5. 13-cmake语法-路径设置

    路径设置: 包括头文件路径.库文件路径.库文件名等 INCLUDE_DIRECTORIES 向工程添加多个特定的头文件搜索路径,路径之间用空格分隔,如果路径包含空格,可以使用双引号将它括起来,默认的行 ...

  6. 码云因为认证失败导致推送失败 生成 SSH 密钥对

  7. crystalreport使用方法

    使用: 打开CrystalReport官网下载页 目前最新版本为13.0.4 选择“SAP Crystal Reports, version for Visual Studio 2010 - Stan ...

  8. 【oracle】定时任务

    --创建定时任务-------------------------------------------------------------------------------------------- ...

  9. [Javascript] Creating an Iterator from an Array

    Every Array has a function which you can use to create an iterator. This function can only be access ...

  10. SWA2G422&485JK2G基础篇: 硬件使用说明

    开发板板载介绍(当前使用的测试板,以后期最终版为准) 一,实物图 硬件说明 一,开发板主控芯片说明: 1. 单片机: STM32F103RET6 2. GPRS模块: Air202 二,开发板外设说明 ...