#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <climits>
#include <cfloat>
#include <map>
#include <utility>
#include <set>
#include <iostream>
#include <memory>
#include <string>
#include <vector>
#include <algorithm>
#include <functional>
#include <sstream>
#include <complex>
#include <stack>
#include <queue>
#include <numeric>
#include <list>
#include <time.h>
#include <string.h>
#include <assert.h>
using namespace std;
typedef long long ll;
template < class T> bool INRANGE( T x,T a,T b) { return a<= x&& x<= b; }
#define NG (-1)
#define BIG (123456789)
#define BIGBIG (123456789123456789LL)
#define SZ(a) ((int)(a).size())
// 最大流 Dinic O(EV^2)だけど、実際もっとはやい。
class Dinic
{
public :
Dinic( int input_maxv) : maxv( input_maxv)
{
G.resize ( input_maxv) ;
level.resize ( input_maxv) ;
iter.resize ( input_maxv) ;
}
void add_edge_both( int from, int to, ll cap)
{
if ( cap> 0 )
{
const int rev_from = SZ( G[ from] ) ;
const int rev_to = SZ( G[ to] ) ;
G[ from] .push_back ( edge( to,cap,rev_to) ) ;
G[ to] .push_back ( edge( from,cap,rev_from) ) ;
}
}
void add_edge( int from, int to, ll cap)
{
if ( cap> 0 )
{
const int rev_from = SZ( G[ from] ) ;
const int rev_to = SZ( G[ to] ) ;
G[ from] .push_back ( edge( to,cap,rev_to) ) ;
G[ to] .push_back ( edge( from,0 ,rev_from) ) ;
}
}
// sからtへの最大流を求める
ll max_flow( int s, int t)
{
ll flow = 0 ;
for ( ;; )
{
bfs( s) ;
if ( level[ t] < 0 ) break ;
fill( iter.begin ( ) ,iter.end ( ) ,0 ) ;
ll f;
while ( ( f= dfs( s,t,DINIC_INF) ) > 0 )
{
flow + = f;
}
}
return flow;
}
// ノードsから辿れる範囲を求める(これ以上流せないところcap=0は、リンクがなくなる)
// (流し終わったあとsourceからたどれる範囲が、最小カット時のs側。たどれない法がt側。その境界がカットするところ。)
vector < bool > get_nodes_in_group( int s)
{
vector < bool > ret( maxv) ;
queue< int > que;
que.push ( s) ;
while ( ! que.empty ( ) )
{
int v = que.front ( ) ;
que.pop ( ) ;
ret[ v] = true ;
for ( int i= 0 ; i< SZ( G[ v] ) ; i++ )
{
edge & e = G[ v] [ i] ;
if ( e.cap > 0 && ! ret[ e.to ] )
{
que.push ( e.to ) ;
}
}
}
return ret;
}
void disp( )
{
for ( int v = 0 ; v < maxv; v++ )
{
printf ( "%d:" ,v) ;
for ( int i= 0 ; i< SZ( G[ v] ) ; i++ )
{
if ( G[ v] [ i] .init_cap > 0 )
{
printf ( "->%d(%lld)," ,G[ v] [ i] .to ,G[ v] [ i] .init_cap ) ;
}
}
printf ( "\n " ) ;
}
}
private :
// sからの最短距離をBFSで計算する
void bfs( int s)
{
fill( level.begin ( ) ,level.end ( ) ,NG) ;
queue< int > que;
level[ s] = 0 ;
que.push ( s) ;
while ( ! que.empty ( ) )
{
int v = que.front ( ) ;
que.pop ( ) ;
for ( int i= 0 ; i< SZ( G[ v] ) ; i++ )
{
edge & e = G[ v] [ i] ;
if ( e.cap > 0 && level[ e.to ] < 0 )
{
level[ e.to ] = level[ v] + 1 ;
que.push ( e.to ) ;
}
}
}
}
// 増加パスをDFSで探す
ll dfs( int v, int t, ll f)
{
if ( v== t) return f;
for ( int & i= iter[ v] ; i< SZ( G[ v] ) ; i++ )
{
edge& e = G[ v] [ i] ;
if ( e.cap > 0 && level[ v] < level[ e.to ] )
{
ll d = dfs( e.to , t, min( f, e.cap ) ) ;
if ( d> 0 )
{
e.cap - = d;
G[ e.to ] [ e.rev ] .cap + = d;
return d;
}
}
}
return 0 ;
}
static const ll DINIC_INF = LLONG_MAX; // 容量をllにしたいときは、ここも変える
struct edge
{
edge( int input_to, ll input_cap, int input_rev) : to( input_to) , cap( input_cap) , rev( input_rev) , init_cap( input_cap) { }
int to; // 行先
ll cap; // 容量
int rev; // 逆辺
ll init_cap; // 初期容量(デバッグ用)
} ;
int maxv;
vector < vector < edge> > G; // グラフの隣接リスト
vector < int > level; // sからの距離
vector < int > iter; // どこまで調べ終わったか
} ;
class RabbitWorking {
public :
bool ok( ll a, ll b, const vector < string> & profit)
{
const int N = SZ( profit) ;
const int S = N+ N* ( N- 1 ) / 2 ;
const int T = S+ 1 ;
const int V = T+ 1 ;
Dinic* dinic = new Dinic( V) ;
ll ret = 0 ;
int v = 0 ;
for ( int i = 0 ; i < N; i++ )
{
dinic- > add_edge( S,i,199 * a) ;
v++ ;
}
for ( int i = 0 ; i < N; i++ )
{
for ( int j = i+ 1 ; j < N; j++ )
{
ll p = profit[ i] [ j] - '0' ;
ll cost = b* p+ 2 * a;
ret + = cost;
dinic- > add_edge( v,T,cost) ;
dinic- > add_edge( i,v,BIGBIG) ;
dinic- > add_edge( j,v,BIGBIG) ;
v++ ;
}
}
ret - = dinic- > max_flow( S,T) ;
delete dinic;
return ret> 0 ;
}
double getMaximum( vector < string> profit) {
ll b = ( ll) 1e12 ;
ll lo = 0 ;
ll hi = ( ll) b* 100 ;
while ( lo< hi)
{
ll mid = lo+ ( hi- lo) / 2LL;
if ( ! ok( mid,b,profit) )
{
hi= mid;
}
else
{
lo= mid+ 1 ;
}
}
return ( double ) lo/ b;
}
private :
template < typename T> string print_array( const vector< T> & V) { ostringstream os; os << "{ " ; for ( typename vector< T> :: const_iterator iter = V.begin ( ) ; iter ! = V.end ( ) ; ++ iter) os << '\" ' << * iter << "\" ," ; os << " }" ; return os.str ( ) ; }
void verify_case( int Case, const double & Expected, const double & Received) { cerr << "Test Case #" << Case << "..." ; if ( Expected == Received) cerr << "PASSED" << endl; else { cerr << "FAILED" << endl; cerr << "\t Expected: \" " << Expected << '\" ' << endl; cerr << "\t Received: \" " << Received << '\" ' << endl; } }
public :
void run_test( int Case) {
int n = 0 ;
// test_case_0
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "071" ,
"702" ,
"120" }
;
double Arg1 = 0.017676767676767676 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
verify_case( n, Arg1, getMaximum( Arg0) ) ;
}
n++ ;
// test_case_1
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "061" ,
"602" ,
"120" }
;
double Arg1 = 0.015228426395939087 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
verify_case( n, Arg1, getMaximum( Arg0) ) ;
}
n++ ;
// test_case_2
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "0" }
;
double Arg1 = 0.0 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
verify_case( n, Arg1, getMaximum( Arg0) ) ;
}
n++ ;
// test_case_3
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "013040" ,
"100010" ,
"300060" ,
"000008" ,
"416000" ,
"000800" }
;
double Arg1 = 0.021996615905245348 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
verify_case( n, Arg1, getMaximum( Arg0) ) ;
}
n++ ;
// test_case_4
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "06390061" ,
"60960062" ,
"39090270" ,
"96900262" ,
"00000212" ,
"00222026" ,
"66761201" ,
"12022610" }
;
double Arg1 = 0.06871794871794872 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
verify_case( n, Arg1, getMaximum( Arg0) ) ;
}
n++ ;
}
} ;
int main( ) {
RabbitWorking* ___test = new RabbitWorking( ) ;
___test- > run_test( - 1 ) ;
delete ___test;
}
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <climits>
#include <cfloat>
#include <map>
#include <utility>
#include <set>
#include <iostream>
#include <memory>
#include <string>
#include <vector>
#include <algorithm>
#include <functional>
#include <sstream>
#include <complex>
#include <stack>
#include <queue>
#include <numeric>
#include <list>
#include <time.h>
#include <string.h>
#include <assert.h>
using namespace std;
typedef long long ll;

template<class T> bool INRANGE(T x,T a,T b) { return a<=x&&x<=b; }
#define NG (-1)
#define BIG (123456789)
#define BIGBIG (123456789123456789LL)
#define SZ(a) ((int)(a).size()) 

// 最大流 Dinic O(EV^2)だけど、実際もっとはやい。
class Dinic
{
public:
	Dinic(int input_maxv) : maxv(input_maxv)
	{
		G.resize(input_maxv);
		level.resize(input_maxv);
		iter.resize(input_maxv);
	}

	void add_edge_both(int from, int to, ll cap)
	{
		if(cap>0)
		{
			const int rev_from	= SZ(G[from]);
			const int rev_to	= SZ(G[to]);
			G[from].push_back(edge(to,cap,rev_to));
			G[to].push_back(edge(from,cap,rev_from));
		}
	}

	void add_edge(int from, int to, ll cap)
	{
		if(cap>0)
		{
			const int rev_from	= SZ(G[from]);
			const int rev_to	= SZ(G[to]);
			G[from].push_back(edge(to,cap,rev_to));
			G[to].push_back(edge(from,0,rev_from));
		}
	}

	// sからtへの最大流を求める
	ll max_flow(int s, int t)
	{
		ll flow = 0;
		for(;;)
		{
			bfs(s);
			if(level[t]<0) break;
			fill(iter.begin(),iter.end(),0);
			ll f;
			while( (f=dfs(s,t,DINIC_INF))>0)
			{
				flow += f;
			}
		}

		return flow;
	}

	//  ノードsから辿れる範囲を求める（これ以上流せないところcap=0は、リンクがなくなる）
	// （流し終わったあとsourceからたどれる範囲が、最小カット時のs側。たどれない法がt側。その境界がカットするところ。）
	vector <bool> get_nodes_in_group(int s)
	{
		vector <bool> ret(maxv);

		queue<int> que;
		que.push(s);
		while(!que.empty())
		{
			int v = que.front();
			que.pop();
			ret[v]=true;

			for(int i=0;i<SZ(G[v]);i++)
			{
				edge &e = G[v][i];
				if(e.cap>0 && !ret[e.to])
				{
					que.push(e.to);
				}
			}
		}
		return ret;
	}

	void disp()
	{
		for (int v = 0; v < maxv; v++)
		{
			printf("%d:",v);
			for(int i=0;i<SZ(G[v]);i++)
			{
				if(G[v][i].init_cap>0)
				{
					printf("->%d(%lld),",G[v][i].to,G[v][i].init_cap);
				}
			}
			printf("\n");
		}
	}

private:
	// sからの最短距離をBFSで計算する
	void bfs(int s)
	{
		fill(level.begin(),level.end(),NG);
		queue<int> que;
		level[s]=0;
		que.push(s);
		while(!que.empty())
		{
			int v = que.front();
			que.pop();
			for(int i=0;i<SZ(G[v]);i++)
			{
				edge &e = G[v][i];
				if(e.cap>0 && level[e.to]<0)
				{
					level[e.to] = level[v] + 1;
					que.push(e.to);
				}
			}
		}
	}

	// 増加パスをDFSで探す
	ll dfs(int v, int t, ll f)
	{
		if(v==t) return f;
		for (int &i=iter[v];i<SZ(G[v]);i++)
		{
			edge& e = G[v][i];
			if(e.cap>0 && level[v]<level[e.to])
			{
				ll d = dfs(e.to, t, min(f, e.cap));
				if(d>0)
				{
					e.cap -= d;
					G[e.to][e.rev].cap += d;
					return d;
				}
			}
		}
		return 0;
	}

	static const ll DINIC_INF = LLONG_MAX; // 容量をllにしたいときは、ここも変える

	struct edge 
	{
		edge(int input_to, ll input_cap, int input_rev) : to(input_to), cap(input_cap), rev(input_rev), init_cap(input_cap) {}
		int to;		// 行先
		ll cap;	// 容量
		int rev;	// 逆辺
		ll init_cap; // 初期容量（デバッグ用）
	};

	int	maxv;
	vector < vector <edge> > G;	// グラフの隣接リスト
	vector < int > level;		// sからの距離
	vector < int > iter;		// どこまで調べ終わったか
};


class RabbitWorking {
public:

	
	bool ok(ll a, ll b, const vector <string>& profit)
	{
		const int N = SZ(profit);
		const int S = N+N*(N-1)/2;
		const int T = S+1;
		const int V = T+1;
		Dinic*	dinic = new Dinic(V);

		ll ret = 0;
		int v = 0;
		for (int i = 0; i < N; i++)
		{
			dinic->add_edge(S,i,199*a);
			v++;
		}

		for (int i = 0; i < N; i++)
		{
			for (int j = i+1; j < N; j++)
			{
				ll p = profit[i][j]-'0';
				ll cost = b*p+2*a;
				ret += cost;
				dinic->add_edge(v,T,cost);
				dinic->add_edge(i,v,BIGBIG);
				dinic->add_edge(j,v,BIGBIG);
				v++;
			}
		}

		ret -= dinic->max_flow(S,T);
		delete dinic;

		return ret>0;
	}

	double getMaximum(vector <string> profit) {

		ll b = (ll)1e12;
		ll lo = 0;
		ll hi = (ll)b*100;

		while(lo<hi)
		{
			ll mid = lo+(hi-lo)/2LL;
			if(!ok(mid,b,profit))
			{
				hi=mid;
			}
			else
			{
				lo=mid+1;
			}
		}

		return (double)lo/b;
	}

private:
	template <typename T> string print_array(const vector<T> &V) { ostringstream os; os << "{ "; for (typename vector<T>::const_iterator iter = V.begin(); iter != V.end(); ++iter) os << '\"' << *iter << "\","; os << " }"; return os.str(); }

	void verify_case(int Case, const double &Expected, const double &Received) { cerr << "Test Case #" << Case << "..."; if (Expected == Received) cerr << "PASSED" << endl; else { cerr << "FAILED" << endl; cerr << "\tExpected: \"" << Expected << '\"' << endl; cerr << "\tReceived: \"" << Received << '\"' << endl; } }

public:
	void run_test(int Case) { 
		int n = 0;

		// test_case_0
		if ((Case == -1) || (Case == n)){
			string Arr0[] = { "071", 
  "702", 
  "120" }
;
			double Arg1 = 0.017676767676767676;

			vector <string> Arg0(Arr0, Arr0 + (sizeof(Arr0) / sizeof(Arr0[0])));
			verify_case(n, Arg1, getMaximum(Arg0));
		}
		n++;

		// test_case_1
		if ((Case == -1) || (Case == n)){
			string Arr0[] = { "061", 
  "602", 
  "120" }
;
			double Arg1 = 0.015228426395939087;

			vector <string> Arg0(Arr0, Arr0 + (sizeof(Arr0) / sizeof(Arr0[0])));
			verify_case(n, Arg1, getMaximum(Arg0));
		}
		n++;

		// test_case_2
		if ((Case == -1) || (Case == n)){
			string Arr0[] = { "0" }
;
			double Arg1 = 0.0;

			vector <string> Arg0(Arr0, Arr0 + (sizeof(Arr0) / sizeof(Arr0[0])));
			verify_case(n, Arg1, getMaximum(Arg0));
		}
		n++;

		// test_case_3
		if ((Case == -1) || (Case == n)){
			string Arr0[] = { "013040", 
  "100010", 
  "300060", 
  "000008", 
  "416000", 
  "000800" }
;
			double Arg1 = 0.021996615905245348;

			vector <string> Arg0(Arr0, Arr0 + (sizeof(Arr0) / sizeof(Arr0[0])));
			verify_case(n, Arg1, getMaximum(Arg0));
		}
		n++;

		// test_case_4
		if ((Case == -1) || (Case == n)){
			string Arr0[] = { "06390061", 
  "60960062", 
  "39090270", 
  "96900262", 
  "00000212", 
  "00222026", 
  "66761201", 
  "12022610" }
;
			double Arg1 = 0.06871794871794872;

			vector <string> Arg0(Arr0, Arr0 + (sizeof(Arr0) / sizeof(Arr0[0])));
			verify_case(n, Arg1, getMaximum(Arg0));
		}
		n++;

	}

};

int main() {
	RabbitWorking* ___test = new RabbitWorking();
	___test->run_test(-1);
	delete ___test;
}

