HDU 4921 Map

题意:

给n个节点  他们形成了最多10条链  每条最多1000的长度  每个节点有个val  你可以选择任意位置截断链  断点前的所有节点被你获得  通过题中计算公式得出你的val  问  通过随机截断  获得val的期望是多少

思路:

期望=所有方案val的和/方案数

这里明显有分层的现象  而且每层最多10个元素  因此想到状压  那么我们只要逐层统计  每层计算一下能对“所有方案val的和”产生多少贡献即可  方案数可以直接算出来  计算方法如下

对于方案数  它就等于 (amt[1]+1)*(amt[2]+1)*…  amt[i]为每条链上的节点总数  这个式子就表示对于每条链有amt+1种截断方式  即  一开始就截断+在每个元素后面截断

对于val的和  我们通过每层的状态来计算(刚才也说了要状态压缩)

如果状压中该位置为1表示选中该元素  那么序列一定是这样的111111XXXXXX  即1前面一定都是1  因此对应的方案有amt-层数+1 种

如果该位置为0  那么序列一定是这样的 XXXXXXX000000 即0后面一定都是0  那么方案就有 层数 种

知道了那一层所形成的方案数  那么只需要计算一下该层的节点val和与方案数乘一下就可以了

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define N 10010

int next[N], vis[N], val[N], amt[10], qu[10];
double x, y;
int t, n, m, tot;

int main() {
	int i, u, v, floor, have, num;
	double ways, res;
	//freopen("1001.in", "r", stdin);
	//freopen("1001.out", "w", stdout);
	scanf("%d", &t);
	while (t--) {
		scanf("%d%d", &n, &m);
		memset(next, 0, sizeof(next));
		memset(vis, 0, sizeof(vis));
		memset(amt, 0, sizeof(amt));
		tot = 0;
		x = 1;
		y = 0;
		for (i = 1; i <= n; i++)
			scanf("%d", &val[i]);
		for (i = 1; i <= m; i++) {
			scanf("%d%d", &u, &v);
			u++;
			v++;
			next[u] = v;
			vis[v] = 1;
		}
		for (i = 1; i <= n; i++)
			if (!vis[i]) {
				qu[tot] = i;
				for (u = i; u; u = next[u])
					amt[tot]++;
				x *= amt[tot] + 1;
				tot++;
			}
		for (floor = 1;; floor++) {
			num = 0;
			for (i = 0; i < tot; i++)
				if (qu[i])
					num++;
			if (!num)
				break;
			for (u = 1; u < (1 << tot); u++) {
				have = 0;
				ways = 1;
				res = 0;
				for (i = 0; i < tot; i++) {
					if (u & (1 << i)) {
						if (!qu[i])
							break;
						res += val[qu[i]];
						have++;
						ways *= amt[i] - floor + 1;
					} else
						ways *= min(floor, amt[i] + 1);
				}
				if (i == tot) {
					y += res * ways;
					if (have > 1)
						y += res * have * ways / num;
				}
			}
			for (i = 0; i < tot; i++)
				qu[i] = next[qu[i]];
		}
		//printf("%.3f %.3f  ", y, x);
		printf("%.3f\n", y / (x - 1));
	}
	return 0;
}

HDU 4921 Map

时间: 2024-10-09 22:45:39

HDU 4921 Map的相关文章

HDU 4921 Map DFS+状态压缩+乘法计数

算最多十条链,能截取某前缀段,每种方案都可以算出一个权值,每种方案的概率都是总数分之一,问最后能构成的所有可能方案数. 对计数原理不太敏感,知道是DFS先把链求出来,但是想怎么统计方案的时候想了好久,其实因为只能取某个链的前缀,所以直接取链长加+1 然后相乘即可,当然因为会出现都是空的那种情况,要去掉,全部乘完之后,要-1 然后就是算权值了,权值等于当前加进来的点的总和 以及 等级相同的点的加成,并不是特别好算,这时候考虑每个状态下的点对全局的贡献,对,就是这个思想,用状态压缩来表示状态,然后这

hdu 1247 map的使用

http://acm.hdu.edu.cn/showproblem.php?pid=1247 Hat’s Words Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 7760    Accepted Submission(s): 2814 Problem Description A hat’s word is a word in the

hdu 4941 map的应用+离散

1 #include<iostream> 2 #include<cstdio> 3 #include<map> 4 #include<algorithm> 5 using namespace std; 6 7 int main() 8 { 9 int T,n,m,k,a,b,c,q,cas=1; 10 scanf("%d",&T); 11 while(T--) 12 { 13 map<int,map<int,int&g

HDU 4287 &lt;map&gt;的使用

对于这道题我主要要讲得是STL中map的简单使用. 先说说map的用处. map就是从键(key)到值(value)的映射.因为重载了[]运算符,map像数组的"高级版",例如可以用一个map<string,int>month_name来表示"月份名字到月份编号"的映射,然后用month_name["July"] = 7这样的方式来赋值. 它的意思就是map<string,int>month_name表示[ ] 里面代表的

HDU 1075 map or 字典树

What Are You Talking About Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/204800 K (Java/Others)Total Submission(s): 12773    Accepted Submission(s): 4069 Problem Description Ignatius is so lucky that he met a Martian yesterday. But

hdu 2112 HDU Today(map与dijkstra的结合使用)

HDU Today Time Limit: 15000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 6147    Accepted Submission(s): 1485 Problem Description 经过锦囊相助,海东集团最终度过了危机,从此,HDU的发展就一直顺风顺水,到了2050年,集团已经相当规模了,据说进入了钱江肉丝经济开发区500强.这时候.

HDU 4329 MAP(stringstream的用法)

这个题目有点绕,但是按着他的意思写不难模拟出来.本来是一场学弟们的训练赛,我这个学长在赛场上却WA了四次都没过,三条黑线就一直在我的脑袋上挂着... 赛后开始找原因,后来发现题目看错了,1/R中的R是指原先URL的个数. 改过来发现还是过不了,想到了自己的输入处理可能是有问题的,既然自己模拟容易出错,那就使用了stringstream,自动提取单词,成功AC. 意外的发现stringstream的功能十分强大,但是因为速度比较慢,用处比较少,所以也不怎么用,这个哥们写的不错,感兴趣可以去看看哦.

hdu 1075 map

题意:给一个字典,和一句话,翻译一下 奇怪的格式 Sample Input START from fiwo hello difh mars riwosf earth fnnvk like fiiwj END START difh, i'm fiwo riwosf. i fiiwj fnnvk! END Sample Output hello, i'm from mars. i like earth! 1 #include<cstdio> 2 #include<iostream> 3

hdu 2923 map+Floyd 拉破车

有向图 具体方向看箭头 从起点到指定城市拉破车,一个城市可能有多个破车,一次只能拉一辆破车 也就是到了指定地点后要回到起点 假如有100辆破车 但是只有一个城市有 就得在起点与这个城市间往返100次所以要用s1记录 然后 貌似这题是有重边的.... sscanf(s4,"%d" ,&w) ; 这个是错的=.= 在这折腾了半天 Sample Input4 2 5 //城市数 破车数 边数NewTroy Midvale Metrodale //起点 + 有破车的城市NewTroy