zoukankan      html  css  js  c++  java
  • 51nod 1850 抽卡大赛(期望)

    51nod 1850 抽卡大赛(期望)

    题目大意

    51Nod为了活跃比赛前的气氛,组织了场抽卡比赛。这场比赛共 n个人参加,主办方根据非欧血统鉴定器,得到了一些数据。每个人抽卡有 Mi 种可能,得到的卡能力值为 Aij 代价为 Gij 的可能性为 Pij ,所谓代价指的是玩家需要将一轮比赛后所得的点头盾的 Gij% 交给主办方。每轮比赛每个人都随机抽取卡片,待全部人抽取完毕后进行排名(按照A从大到小排),排在第 i 位的人有 Vi 的点头盾收入。现在主办方想知道一轮比赛后每个人的期望收入。

    数据范围

    第一行一个正整数 n
    接下来 n 个部分
    每个部分第一行为正整数 Mi,接下来 Mi 行有三个整数 Aij Gij Pij
    接下来一行 n 个整数,分别为 Vi
    设 ∑Pij=Qi,则第 i 个人抽到第 j 张卡的概率为 Pij/Qi
    1<=n,Mi<=200,1<=Aij<=1000000000,保证 Aij 互不相同,0<=Gij<=100,1<=Pij<=1000,1<=Vi<=1000

    解题思路

    没见过的思路

    首先很好有一个 (Theta(n^4)) 的做法,想知道第 i 个人的期望,假设他抽到的是第 j 个,能力值为 A,可以直接 (Theta(n^2)) 的 dp 得出他排在第 k 名的概率

    也有另一种表示方法 (prod p_ix+(1-p_i)),其中 (p_i) 表示第 i 个人的能力值比 (A) 大的概率

    即使这样也还是暴力的,但我们发现枚举 i,j 的顺序不会有任何影响,所以我们将所有卡按能力值排序,从低到高枚举,不难发现每次只有一个二项式发生变化,也就是我们暴力除法在乘法即可,因为多项式只有两项,所以可以 (Theta(n)) 的从一个转移到另一个

    代码

    const int P = 1e9+7;
    const int N = 205;
    
    ll fpw(ll x, ll mi) {
    	ll res = 1;
    	for (; mi; mi >>= 1, x = x * x % P)
    		if (mi & 1) res = res * x % P;
    	return res;
    }
    
    struct node {
    	ll a, g, p, num;
    	bool operator < (const node &i) const {
    		return a < i.a;
    	}
    }a[N][N], all[N * N];
    ll sum[N], m[N], cnt;
    ll inv[N], f[N * N], g[N * N], Inv = fpw(100, P - 2);
    ll v[N], p[N], ans[N], n;
    
    void Mul(ll a, ll b) {
    	for (int i = 0;i <= n; i++) g[i] = f[i] * b % P;
    	for (int i = 0;i <= n; i++) g[i + 1] = (g[i + 1] + f[i] * a) % P;
    	for (int i = 0;i <= n; i++) f[i] = g[i];
    }
    
    void Div(ll a, ll b) {
    	ll Inv = fpw(a, P - 2);
    	for (int i = n;i >= 0; i--) {
    		ll k = f[i + 1] * Inv % P;
    		g[i] = k, f[i] = (f[i] - b * k % P + P) % P; 
    	}
    	for (int i = 0;i <= n; i++) f[i] = g[i];
    }
    
    int main() {
    	freopen ("hs.in","r",stdin);
    //	freopen ("hs.out","w",stdout);
    	read(n);
    	for (int i = 1;i <= n; i++) {
    		read(m[i]); p[i] = 0;
    		for (int j = 1;j <= m[i]; j++) {
    			read(a[i][j].a), read(a[i][j].g);
    			read(a[i][j].p), a[i][j].num = i;
    		}
    		sort(a[i] + 1, a[i] + m[i] + 1);
    		for (int j = 1;j <= m[i]; j++) 
    			sum[i] = sum[i] + a[i][j].p;
    		inv[i] = fpw(sum[i], P - 2);
    		for (int j = 1;j <= m[i]; j++) {
    			a[i][j].p = a[i][j].p * inv[i] % P;
    			a[i][j].g = 100 - a[i][j].g;
    			all[++cnt] = a[i][j]; 
    		}
    	}
    	for (int i = 1;i <= n; i++) read(v[i]);
    	sort(all + 1, all + cnt + 1);
    	f[0] = 1;
    	for (int i = 1;i <= n; i++) Mul(p[i] = 1, 0);
    	for (int i = 1;i <= cnt; i++) {
    		int k = all[i].num;
    		Div(p[k], (1 - p[k] + P) % P);
    		for (int j = 1;j <= n; j++)
    			ans[k] = (ans[k] + all[i].g * Inv % P * v[j] % P * f[j-1] % P * all[i].p) % P;
    		p[k] = (p[k] - all[i].p + P) % P;
    		Mul(p[k], (1 - p[k] + P) % P);
    	}
    	for (int i = 1;i <= n; i++) write(ans[i]);
    	return 0;
    }
    
  • 相关阅读:
    (005)Linux 复制命令cp总提示是否覆盖的解决方法,在cp前加
    (030)Spring Boot之RestTemplate访问web服务案例
    Gym
    Gym
    Gym.102006:Syrian Collegiate Programming Contest(寒假自训第11场)
    BZOJ-5244 最大真因数(min25筛)
    HDU
    HDU 1272 小希的迷宫(并查集)
    HDU 3038 How Many Answers Are Wrong(带权并查集)
    POJ 1182 食物链(带权并查集)
  • 原文地址:https://www.cnblogs.com/Hs-black/p/13320691.html
Copyright © 2011-2022 走看看