Algorithm


E. Another Sith Tournament
time limit per test
2.5 seconds
memory limit per test
256 megabytes
input
standard input
output
standard output

The rules of Sith Tournament are well known to everyone. n Sith take part in the Tournament. The Tournament starts with the random choice of two Sith who will fight in the first battle. As one of them loses, his place is taken by the next randomly chosen Sith who didn't fight before. Does it need to be said that each battle in the Sith Tournament ends with a death of one of opponents? The Tournament ends when the only Sith remains alive.

Jedi Ivan accidentally appeared in the list of the participants in the Sith Tournament. However, his skills in the Light Side of the Force are so strong so he can influence the choice of participants either who start the Tournament or who take the loser's place after each battle. Of course, he won't miss his chance to take advantage of it. Help him to calculate the probability of his victory.

Input

The first line contains a single integer n (1 ≤ n ≤ 18) — the number of participants of the Sith Tournament.

Each of the next n lines contains n real numbers, which form a matrix pij (0 ≤ pij ≤ 1). Each its element pij is the probability that the i-th participant defeats the j-th in a duel.

The elements on the main diagonal pii are equal to zero. For all different ij the equality pij + pji = 1 holds. All probabilities are given with no more than six decimal places.

Jedi Ivan is the number 1 in the list of the participants.

Output

Output a real number — the probability that Jedi Ivan will stay alive after the Tournament. Absolute or relative error of the answer must not exceed 10 - 6.

Examples
input
Copy
3
0.0 0.5 0.8
0.5 0.0 0.4
0.2 0.6 0.0
output
Copy
0.680000000000000



 

Code Examples

#1 Code Example with C++ Programming

Code - C++ Programming

#include <bits/stdc++.h>

using namespace std;

int n;
double p[18][18], dp[18][(1 << 18)];

double rec(int lst, int msk) {
  if(__builtin_popcount(msk) == n - 1)
    return lst == 0;

  double &ret = dp[lst][msk];
  if(ret == ret)
    return ret;
  ret = 0;

  double cur;
  for(int i = 0; i < n; ++i)
    if(i != lst && ((msk >> i) & 1) == 0) {
      cur = 0;
      cur += p[lst][i] * rec(lst, msk | (1 << i));
      cur += p[i][lst] * rec(i, msk | (1 << lst));
      ret = max(ret, cur);
    }

  return ret;
}

int main() {
  scanf("%d", &n);
  for(int i = 0; i < n; ++i)
    for(int j = 0; j < n; ++j)
      scanf("%lf", &p[i][j]);

  memset(dp, -1, sizeof dp);
  double res = 0;
  for(int i = 0; i < n; ++i)
    res = max(res, rec(i, 0));

  printf("%0.10lf\n", res);

  return 0;
}
Copy The Code & Try With Live Editor

Input

x
+
cmd
3
0.0 0.5 0.8
0.5 0.0 0.4
0.2 0.6 0.0

Output

x
+
cmd
0.680000000000000
Advertisements

Demonstration


Codeforces Solution-Another Sith Tournament-Solution in C, C++, Java, Python

Previous
Codeforces solution 1080-B-B. Margarite and the best present codeforces solution
Next
CodeChef solution DETSCORE - Determine the Score CodeChef solution C,C+