#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 (987654321)
#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, int 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, int 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への最大流を求める
int max_flow( int s, int t)
{
int flow = 0 ;
for ( ;; )
{
bfs( s) ;
if ( level[ t] < 0 ) break ;
fill( iter.begin ( ) ,iter.end ( ) ,0 ) ;
int 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(%d)," ,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で探す
int dfs( int v, int t, int 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 ] )
{
int 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 int DINIC_INF = INT_MAX ; // 容量をllにしたいときは、ここも変える
struct edge
{
edge( int input_to, int input_cap, int input_rev) : to( input_to) , cap( input_cap) , rev( input_rev) , init_cap( input_cap) { }
int to; // 行先
int cap; // 容量
int rev; // 逆辺
int init_cap; // 初期容量(デバッグ用)
} ;
int maxv;
vector < vector < edge> > G; // グラフの隣接リスト
vector < int > level; // sからの距離
vector < int > iter; // どこまで調べ終わったか
} ;
class SurroundingGame {
public :
int maxScore( vector < string> cost, vector < string> benefit) {
// BEGIN CUT HERE
// 1. 制約条件から読むこと
// 2. 問題が理解できたら、Exampleで理解があってるか確認
// 3. 解法が分からないときは、とにかく小さいケースの、「答え」を書き出す。プログラムも使って可
// 4. それでも分からないときはグラフやx-y図を描く
// MODの問題はlong longで。=0, =1, ="0001", ="0357"とか危険
// END CUT HERE
H = SZ( cost) ;
W = SZ( cost[ 0 ] ) ;
const int S = H* W* 2 ;
const int T = S+ 1 ;
const int V = T+ 1 ;
Dinic* dinic = new Dinic( V) ;
int totalDefaultProfit = 0 ;
for ( int y = 0 ; y < H; y++ )
{
for ( int x = 0 ; x < W; x++ )
{
//---------- 置く or 置かない ----------
// 置く profit = benefit-cost
// 置かない profit = 0
{
const int bc = getDecode( benefit[ y] [ x] ) - getDecode( cost[ y] [ x] ) ;
int capS;
int capT;
if ( bc> 0 )
{
// benefit-cost>0のとき
// defaultProfit = benefit-cost
// 置く cap = defaultProfit - profit = 0
// 置かない cap = defaultProfit - profit = benefit-cost
capS = 0 ; // 置く
capT = bc; // 置かない
totalDefaultProfit + = bc;
}
else
{
// benefit-cost<0のとき
// defaultProfit = 0
// 置く cap = defaultProfit - profit = cost-benefit;
// 置かない cap = defaultProfit - profit = 0;
capS = - bc; // 置く
capT = 0 ; // 置かない
}
// 隣り合う条件を使うときに、二部グラフを利用。S側とT側を逆にする。
if ( ( x+ y) & 1 )
{
swap( capS,capT) ;
}
dinic- > add_edge( S, oku( y,x) , capS) ;
dinic- > add_edge( oku( y,x) , T, capT) ;
}
//---------- 囲まれない or 囲まれる ----------
{
const int b = getDecode( benefit[ y] [ x] ) ;
// 囲まれない profit = 0
// 囲まれる profit = benefit
// この問題ではbenefitは常に0以上
// defaultProfit = b
// 囲まない cap = defaultProfit - profit = benefit;
// 囲む cap = defaultProfit - profit = 0;
totalDefaultProfit + = b;
int capS = b;
int capT = 0 ;
// 隣り合う条件を使うときに、二部グラフを利用。S側とT側を逆にする。
if ( ( x+ y) & 1 )
{
swap( capS,capT) ;
}
dinic- > add_edge( S, kakomu( y,x) , capS) ;
dinic- > add_edge( kakomu( y,x) , T, capT) ;
}
//----- 同じ場所で 置く && 囲まれる は同時に成立禁止 -----
// 問題の条件より、置くか囲まれると、利益bもらえる。
// このままだと、置いてかつ囲まれると、利益が2bもらえてしまう。
// 置く && 囲む は同時に成立しても、グラフが切れないようにする。
if ( ( x+ y) & 1 )
{
dinic- > add_edge( oku( y,x) , kakomu( y,x) , BIG) ;
}
else
{
dinic- > add_edge( kakomu( y,x) , oku( y,x) , BIG) ;
}
//----- 囲まれる条件 -----
// 囲まれるときは、隣の場所はすべて置く必要がある。
// 言い換えると、囲まれたマスのまわりに、1個でも置かないマスがあったら、グラフが切れないようにする。
const static int dy[ ] = { - 1 , 0 , 1 , 0 } ; // U,R,D,L
const static int dx[ ] = { 0 , 1 , 0 ,- 1 } ;
for ( int d = 0 ; d < 4 ; d++ )
{
const int ny = y+ dy[ d] ;
const int nx = x+ dx[ d] ;
if ( INRANGE( ny,0 ,H- 1 ) && INRANGE( nx,0 ,W- 1 ) )
{
if ( ( x+ y) & 1 )
{
dinic- > add_edge( oku( ny,nx) , kakomu( y,x) , BIG) ;
}
else
{
dinic- > add_edge( kakomu( y,x) , oku( ny,nx) , BIG) ;
}
}
}
}
}
const int result = totalDefaultProfit - dinic- > max_flow( S,T) ;
delete dinic;
return result;
}
private :
int getDecode( const char c)
{
if ( isdigit ( c) )
{
return c- '0' ;
}
else if ( islower ( c) )
{
return c- 'a' + 10 ;
}
else if ( isupper ( c) )
{
return c- 'A' + 36 ;
}
return NG;
}
int oku( int y, int x)
{
return y* W+ x;
}
int kakomu( int y, int x)
{
return y* W+ x+ H* W;
}
int H;
int W;
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 int & Expected, const int & 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[ ] = { "21" ,"12" } ;
string Arr1[ ] = { "21" ,"12" } ;
int Arg2 = 4 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
vector < string> Arg1( Arr1, Arr1 + ( sizeof ( Arr1) / sizeof ( Arr1[ 0 ] ) ) ) ;
verify_case( n, Arg2, maxScore( Arg0, Arg1) ) ;
}
n++ ;
// test_case_1
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "ZZ" ,"ZZ" } ;
string Arr1[ ] = { "11" ,"11" } ;
int Arg2 = 0 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
vector < string> Arg1( Arr1, Arr1 + ( sizeof ( Arr1) / sizeof ( Arr1[ 0 ] ) ) ) ;
verify_case( n, Arg2, maxScore( Arg0, Arg1) ) ;
}
n++ ;
// test_case_2
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "XXX" ,"XXX" ,"XXX" } ;
string Arr1[ ] = { "aaa" ,"aZa" ,"aaa" } ;
int Arg2 = 2 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
vector < string> Arg1( Arr1, Arr1 + ( sizeof ( Arr1) / sizeof ( Arr1[ 0 ] ) ) ) ;
verify_case( n, Arg2, maxScore( Arg0, Arg1) ) ;
}
n++ ;
// test_case_3
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "asam" ,"atik" } ;
string Arr1[ ] = { "123A" ,"45BC" } ;
int Arg2 = 71 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
vector < string> Arg1( Arr1, Arr1 + ( sizeof ( Arr1) / sizeof ( Arr1[ 0 ] ) ) ) ;
verify_case( n, Arg2, maxScore( Arg0, Arg1) ) ;
}
n++ ;
// test_case_4
if ( ( Case == - 1 ) || ( Case == n) ) {
string Arr0[ ] = { "IIIIIIII" ,
"IIWWWWII" ,
"IIWIIIII" ,
"IIWIIIII" ,
"IIWWWWII" ,
"IIIIIWII" ,
"IIIIIWII" ,
"IIWWWWII" ,
"IIIIIIII" }
;
string Arr1[ ] = { "IIIIIIII" ,
"II0000II" ,
"II0II0II" ,
"II0II0II" ,
"II0000II" ,
"II0II0II" ,
"II0II0II" ,
"II0000II" ,
"IIIIIIII" }
;
int Arg2 = 606 ;
vector < string> Arg0( Arr0, Arr0 + ( sizeof ( Arr0) / sizeof ( Arr0[ 0 ] ) ) ) ;
vector < string> Arg1( Arr1, Arr1 + ( sizeof ( Arr1) / sizeof ( Arr1[ 0 ] ) ) ) ;
verify_case( n, Arg2, maxScore( Arg0, Arg1) ) ;
}
n++ ;
}
} ;
int main( ) {
SurroundingGame* ___test = new SurroundingGame( ) ;
___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 (987654321)
#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, int 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, int 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への最大流を求める
	int max_flow(int s, int t)
	{
		int flow = 0;
		for(;;)
		{
			bfs(s);
			if(level[t]<0) break;
			fill(iter.begin(),iter.end(),0);
			int 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(%d),",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で探す
	int dfs(int v, int t, int 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])
			{
				int 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 int DINIC_INF = INT_MAX; // 容量をllにしたいときは、ここも変える

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

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


class SurroundingGame {
public:


	int maxScore(vector <string> cost, vector <string> benefit) {
// BEGIN CUT HERE
		// 1. 制約条件から読むこと
		// 2. 問題が理解できたら、Exampleで理解があってるか確認
		// 3. 解法が分からないときは、とにかく小さいケースの、「答え」を書き出す。プログラムも使って可
		// 4. それでも分からないときはグラフやx-y図を描く
		// MODの問題はlong longで。=0, =1, ="0001", ="0357"とか危険
// END CUT HERE
		H = SZ(cost);
		W = SZ(cost[0]);
		const int S = H*W*2;
		const int T = S+1;
		const int V = T+1;
		Dinic*	dinic = new Dinic(V);

		int totalDefaultProfit = 0;
		for (int y = 0; y < H; y++)
		{
			for (int x = 0; x < W; x++)
			{

				//---------- 置く or 置かない ----------
				// 置く     profit = benefit-cost
				// 置かない profit = 0       
				{
					const int bc = getDecode(benefit[y][x])-getDecode(cost[y][x]);
					int capS;
					int capT;

					if(bc>0)
					{
						// benefit-cost>0のとき
						// defaultProfit = benefit-cost
						// 置く     cap = defaultProfit - profit = 0
						// 置かない cap = defaultProfit - profit = benefit-cost
						capS = 0;	// 置く
						capT = bc;	// 置かない
						totalDefaultProfit += bc;
					}
					else
					{
						// benefit-cost<0のとき
						// defaultProfit = 0
						// 置く     cap = defaultProfit - profit = cost-benefit;
						// 置かない cap = defaultProfit - profit = 0;
						capS = -bc;	// 置く
						capT = 0;	// 置かない
					}

					// 隣り合う条件を使うときに、二部グラフを利用。S側とT側を逆にする。
					if((x+y)&1)
					{
						swap(capS,capT);
					}

					dinic->add_edge(S, oku(y,x), capS);
					dinic->add_edge(oku(y,x), T, capT);
				}

				//---------- 囲まれない or 囲まれる ----------
				{
					const int b = getDecode(benefit[y][x]);
					// 囲まれない profit = 0
					// 囲まれる   profit = benefit

					// この問題ではbenefitは常に0以上
					// defaultProfit = b
					// 囲まない cap = defaultProfit - profit = benefit;
					// 囲む     cap = defaultProfit - profit = 0;
					totalDefaultProfit += b;
					int capS = b;
					int capT = 0;

					// 隣り合う条件を使うときに、二部グラフを利用。S側とT側を逆にする。
					if((x+y)&1)
					{
						swap(capS,capT);
					}

					dinic->add_edge(S, kakomu(y,x), capS);
					dinic->add_edge(kakomu(y,x), T, capT);
				}

				//----- 同じ場所で 置く && 囲まれる は同時に成立禁止 ----- 
				// 問題の条件より、置くか囲まれると、利益bもらえる。
				// このままだと、置いてかつ囲まれると、利益が2bもらえてしまう。
				// 置く && 囲む は同時に成立しても、グラフが切れないようにする。
				if((x+y)&1)
				{
					dinic->add_edge(oku(y,x), kakomu(y,x), BIG);
				}
				else
				{
					dinic->add_edge(kakomu(y,x), oku(y,x), BIG);
				}

				//-----  囲まれる条件 ----- 
				// 囲まれるときは、隣の場所はすべて置く必要がある。
				// 言い換えると、囲まれたマスのまわりに、１個でも置かないマスがあったら、グラフが切れないようにする。
				const static int dy[] = {-1, 0, 1, 0}; // U,R,D,L
				const static int dx[] = { 0, 1, 0,-1};
				for(int d = 0; d < 4; d++)
				{
					const int ny = y+dy[d]; 
					const int nx = x+dx[d];
					if(INRANGE(ny,0,H-1)&&INRANGE(nx,0,W-1))
					{
						if((x+y)&1)
						{
							dinic->add_edge(oku(ny,nx), kakomu(y,x), BIG);
						}
						else
						{
							dinic->add_edge(kakomu(y,x), oku(ny,nx), BIG);
						}
					}
				}
			}
		}

		const int result = totalDefaultProfit - dinic->max_flow(S,T);

		delete dinic;

		return result;
	}

private:
	int getDecode(const char c)
	{
		if(isdigit(c))
		{
			return c-'0';
		}
		else if (islower(c))
		{
			return c-'a'+10;
		}
		else if (isupper(c))
		{
			return c-'A'+36;
		}

		return NG;
	}

	int oku(int y, int x)
	{
		return y*W+x;
	}
	int kakomu(int y, int x)
	{
		return y*W+x+H*W;
	}

	int H;
	int W;

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 int &Expected, const int &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[] = {"21","12"};
			string Arr1[] = {"21","12"};
			int Arg2 = 4;

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

		// test_case_1
		if ((Case == -1) || (Case == n)){
			string Arr0[] = {"ZZ","ZZ"};
			string Arr1[] = {"11","11"};
			int Arg2 = 0;

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

		// test_case_2
		if ((Case == -1) || (Case == n)){
			string Arr0[] = {"XXX","XXX","XXX"};
			string Arr1[] = {"aaa","aZa","aaa"};
			int Arg2 = 2;

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

		// test_case_3
		if ((Case == -1) || (Case == n)){
			string Arr0[] = {"asam","atik"};
			string Arr1[] = {"123A","45BC"};
			int Arg2 = 71;

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

		// test_case_4
		if ((Case == -1) || (Case == n)){
			string Arr0[] = {"IIIIIIII",
 "IIWWWWII",
 "IIWIIIII",
 "IIWIIIII",
 "IIWWWWII",
 "IIIIIWII",
 "IIIIIWII",
 "IIWWWWII",
 "IIIIIIII"}
;
			string Arr1[] = {"IIIIIIII",
 "II0000II",
 "II0II0II",
 "II0II0II",
 "II0000II",
 "II0II0II",
 "II0II0II",
 "II0000II",
 "IIIIIIII"}
;
			int Arg2 = 606;

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

	}

};

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

