嘟嘟嘟


这题卡精度!,我对拍了近1w组数据都没拍出来,辛亏最后看了HDU的讨论版。


首先对于每一个房间,令\(x_u\)表示从这个点出发的期望步数,很容易列出方程:\(x_u = K_i * x_1 + (1 - K_i - E_i) * (1 + \sum x_v)\)
当我快快乐乐的写完了高斯消元后,发现这个\(n\)有点大啊……
然后我就想这题有啥特别的地方,毕竟以前做图的这种题,高斯消元都是可以过的。
仔细一想,发现这是一棵树,对没错,它是树!这样每一个节点的方程只和他的父亲、儿子、以及根节点有关。而叶子节点就只和他的父亲和根节点有关。所以对于叶子节点,可以用他的父亲和根节点表示,然后代入到父亲的方程中,这样父亲的方程中的未知数只剩他自己和根节点了!这样一路递归上去,根节点的方程就只有他自己一个未知数了,此题得解。时间复杂度\(O(n)\)


(精度至少要设成1e-9……)

#include<cstdio>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<cctype>
#include<vector>
#include<stack>
#include<queue>
#include<assert.h>
using namespace std;
#define enter puts("") 
#define space putchar(' ')
#define Mem(a, x) memset(a, x, sizeof(a))
#define In inline
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-10;
const int maxn = 1e4 + 5;
In ll read()
{
  ll ans = 0;
  char ch = getchar(), last = ' ';
  while(!isdigit(ch)) last = ch, ch = getchar();
  while(isdigit(ch)) ans = (ans << 1) + (ans << 3) + ch - '0', ch = getchar();
  if(last == '-') ans = -ans;
  return ans;
}
In void write(ll x)
{
  if(x < 0) x = -x, putchar('-');
  if(x >= 10) write(x / 10);
  putchar(x % 10 + '0');
}
In void MYFILE()
{
#ifndef mrclr
  freopen("random.in", "r", stdin);
  freopen("ac.out", "w", stdout);
#endif
}

int n, du[maxn], K[maxn], E[maxn];
struct Edge
{
  int nxt, to;
}e[maxn << 1];
int head[maxn], ecnt = -1;
In void addEdge(int x, int y)
{
  ++du[y];
  e[++ecnt] = (Edge){head[x], y};
  head[x] = ecnt;
}

struct Node
{
  db a, b, d;  //a * x1, b * xfa, d
}t[maxn];

db f[maxn];
In void dfs(int now, int _f)
{
  for(int i = head[now], v; ~i; i = e[i].nxt)
    if((v = e[i].to) ^ _f) dfs(v, now);
  db tp = (100.0 - K[now] - E[now]) / 100.0;
  f[now] = 1; f[n + 1] = tp;
  if(now ^ 1) f[1] = -K[now] / 100.0;
  for(int i = head[now]; ~i; i = e[i].nxt)
    f[e[i].to] += -tp * 1.0 / du[now];
  for(int i = head[now], v; ~i; i = e[i].nxt)
    {
      if((v = e[i].to) == _f) continue;
      f[1] += t[v].a * f[v];
      f[now] += t[v].b * f[v];
      f[n + 1] -= t[v].d * f[v];
      f[v] = 0;
    }
  t[now].a = -f[1] / f[now];
  if(_f ^ 1) t[now].b = -f[_f] / f[now];
  t[now].d = f[n + 1] / f[now];
  if(now ^ 1) f[1] = f[now] = f[n + 1] = f[_f] = 0;
}

In void init()
{
  Mem(head, -1); ecnt = -1;
  Mem(f, 0), Mem(du, 0);
  for(int i = 1; i <= n; ++i) t[i] = (Node){0, 0, 0};
}

int main()
{
  MYFILE();
  int TT = read();
  for(int T = 1; T <= TT; ++T)
    {
      n = read();
      init();
      for(int i = 1; i < n; ++i)
	{
	  int x = read(), y = read();
	  addEdge(x, y), addEdge(y, x);
	}
      for(int i = 1; i <= n; ++i) K[i] = read(), E[i] = read();
      dfs(1, 0);
      printf("Case %d: ", T);
      if(f[1] < eps) puts("impossible");
      else printf("%.8lf\n", f[n + 1] / f[1]);
    }
  return 0;
}