/*
* cd.cpp
*
* Created on: 2012-6-25
* Author: mac
*/
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <climits>
#include <numeric>
#include <string>
#include <cassert>
#define foreach(e,x) for(__typeof(x.begin()) e=x.begin();e!=x.end();++e)
#define REP(i,n) for(int i=0;i<n;++i)
using namespace std;
//2,3,...,A
struct Card {
int p, r;
Card() {
}
Card(int p, int r) {
this->p = p;
this->r = r;
}
bool operator<(const Card&o) const {
return p != o.p ? p < o.p : r < o.r;
}
int id() {
return (p - 2) * 4 + r;
}
void read() {
cin >> p >> r;
if (p == 1)
p = 14;
r = 4 - r;
}
};
void show(int who) {
for (int i = 0; i < 5; ++i) {
cout << who % 52 << " ";
who /= 52;
}
cout << endl;
}
struct Hand {
int type, dot, col, who;
Hand() {
}
Hand(int type, int dot, int col, int who) {
this->type = type;
this->dot = dot;
this->col = col;
this->who = who;
}
bool operator<(const Hand&o) const {
if (type != o.type)
return type > o.type;
if (dot != o.dot)
return dot < o.dot;
if (col != o.col)
return col < o.col;
return false;
}
};
Card cards[5];
bool isStright(int&dot, int&col) {
bool ok = true;
for (int i = 1; i < 5; ++i) {
if (cards[i].p != cards[0].p + i)
ok = false;
}
if (ok) {
dot = cards[0].p, col = 0;
for (int i = 5 - 1; i >= 0; --i) {
col = col * 4 + cards[i].r;
}
return true;
}
//check A2345
if (cards[4].p == 14) {
ok = true;
for (int i = 0; i < 4; ++i) {
if (cards[i].p != i + 2)
ok = false;
}
if (ok) { //good! A2345
dot = 1, col = 0;
for (int i = 4 - 1; i >= 0; --i) {
col = col * 4 + cards[i].r;
}
col = col * 4 + cards[4].r;
return true;
}
}
return false;
}
bool isFlush() {
for (int i = 1; i < 5; ++i) {
if (cards[i].r != cards[0].r)
return false;
}
return true;
}
string makeIt(int&dot, int&col) {
pair<int, Card> ps[5];
string cnt = "";
for (int i = 0, j; i < 5; i = j) {
for (j = i; j < 5 && cards[j].p == cards[i].p; ++j)
;
cnt += char('0' + j - i);
for (int k = i; k < j; ++k) {
ps[k] = make_pair(j - i, cards[k]);
}
}
sort(ps, ps + 5);
dot = col = 0;
for (int i = 5 - 1; i >= 0; --i) {
dot = dot * 13 + ps[i].second.p - 2;
col = col * 4 + ps[i].second.r;
}
sort(cnt.rbegin(), cnt.rend());
return cnt;
}
void makeItSorted(int&dot, int&col) {
dot = 0, col = 0;
for (int i = 5 - 1; i >= 0; --i) {
dot = dot * 13 + cards[i].p - 2;
col = col * 4 + cards[i].r;
}
}
Hand process() {
sort(cards, cards + 5);
int dot, col, who = 0;
for (int i = 0; i < 5; ++i) {
who = who * 52 + cards[i].id();
}
bool isS = isStright(dot, col);
bool isF = isFlush();
int type;
int dotG, colG;
string cnt = makeIt(dotG, colG);
if (isS && isF) {
type = 1;
return Hand(type, dot, col, who);
}
if (cnt == "41") {
type = 2;
return Hand(type, dotG, colG, who);
}
if (cnt == "32") {
type = 3;
return Hand(type, dotG, colG, who);
}
int dotS, colS;
makeItSorted(dotS, colS);
if (isF) {
type = 4;
return Hand(type, dotS, colS, who);
}
if (isS) {
type = 5;
return Hand(type, dot, col, who);
}
if (cnt == "311") {
type = 6;
return Hand(type, dotG, colG, who);
}
if (cnt == "221") {
type = 7;
return Hand(type, dotG, colG, who);
}
if (cnt == "2111") {
type = 8;
return Hand(type, dotG, colG, who);
}
type = 9;
return Hand(type, dotS, colS, who);
}
typedef long long int64;
bool myCard[52] = { }, youCard[52] = { };
int n;
void readCardSet(int n, bool cardSet[]) {
Card c;
for (int i = 0; i < n; ++i) {
c.read();
cardSet[c.id()] = true;
}
}
Card allCard[52];
Card cur[5];
const int MAX_HANDS = 3000000 + 10;
Hand hands[MAX_HANDS];
int nHands = 0;
bool check(int who, bool cardSet[], int n) {
int cnt = 0;
for (int i = 0; i < 5; ++i) {
int x = who % 52;
who /= 52;
if (cardSet[x])
++cnt;
}
return cnt == n;
}
void dfs(int u, int who, int cnt) { //search all cardSet combination
if (u == 52) {
if (cnt == 5 && (check(who, myCard, n) || check(who, youCard, n - 1))) {
memcpy(cards, cur, sizeof cur);
hands[nHands++] = process();
}
return;
}
dfs(u + 1, who, cnt);
if (cnt < 5) {
cur[cnt] = allCard[u];
dfs(u + 1, who * 52 + u, cnt + 1);
}
}
bool check(const Hand&h, bool cardSet[], int n) {
return check(h.who, cardSet, n);
}
typedef long long int64;
int64 comb(int n, int m) {
int64 ret = 1;
for (int i = 0; i < m; ++i) {
ret *= n - i;
ret /= i + 1;
}
return ret;
}
int id[5];
int64 A, B;
int dp[6][60]; //rem,max
void prepare() {
memset(dp, 0, sizeof dp);
fill(dp[0], dp[0] + 60, 1);
for (int rem = 1; rem <= 5; ++rem) {
for (int mx = 1; mx <= 52; ++mx) {
dp[rem][mx] += dp[rem][mx - 1] + dp[rem - 1][mx - 1];
}
}
}
int curv[5];
int mp[4000000];
int eval(int v[], int n) {
int code = 0;
for (int i = 0; i < n; ++i) {
code += dp[i][52];
}
for (int i = 0; i < n; ++i)
code += dp[n - i][v[i]];
return code;
}
void dfs1(int u, int cnt) {
if (u == 5) {
int cd = eval(curv, cnt);
A += (cnt % 2 == 0 ? 1 : -1) * mp[cd];
return;
}
dfs1(u + 1, cnt);
curv[cnt] = id[u];
dfs1(u + 1, cnt + 1);
}
void dfs2(int u, int cnt) {
if (u == 5) {
int cd = eval(curv, cnt);
mp[cd]++;
return;
}
dfs2(u + 1, cnt);
curv[cnt] = id[u];
dfs2(u + 1, cnt + 1);
}
int main() {
cin >> n;
readCardSet(n, myCard);
readCardSet(n - 1, youCard);
for (int i = 2; i <= 14; ++i) {
for (int j = 0; j < 4; ++j) {
Card c(i, j);
allCard[c.id()] = c;
}
}
dfs(0, 0, 0);
sort(hands, hands + nHands);
int rem = 52 - n - (n - 1);
A = 0, B = 1;
B *= comb(rem, 5 - n);
rem -= 5 - n;
B *= comb(rem, 5 - (n - 1));
prepare();
for (int i = 0; i < nHands; ++i) {
Hand&h = hands[i];
if (check(h, myCard, n)) {
int who = h.who;
for (int j = 0; j < 5; ++j) {
id[j] = who % 52;
who /= 52;
}
sort(id, id + 5);
reverse(id, id + 5);
dfs1(0, 0);
}
if (check(h, youCard, n - 1)) {
int who = h.who;
for (int j = 0; j < 5; ++j) {
id[j] = who % 52;
who /= 52;
}
sort(id, id + 5);
reverse(id, id + 5);
dfs2(0, 0);
}
}
int64 G = __gcd(A, B);
A /= G, B /= G;
cout << A << "/" << B << endl;
return 0;
}
/*
 * cd.cpp
 *
 *  Created on: 2012-6-25
 *      Author: mac
 */
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <climits>
#include <numeric>
#include <string>
#include <cassert>
#define foreach(e,x) for(__typeof(x.begin()) e=x.begin();e!=x.end();++e)
#define REP(i,n) for(int i=0;i<n;++i)
using namespace std;

//2,3,...,A

struct Card {
	int p, r;
	Card() {
	}
	Card(int p, int r) {
		this->p = p;
		this->r = r;
	}
	bool operator<(const Card&o) const {
		return p != o.p ? p < o.p : r < o.r;
	}

	int id() {
		return (p - 2) * 4 + r;
	}

	void read() {
		cin >> p >> r;
		if (p == 1)
			p = 14;
		r = 4 - r;
	}
};

void show(int who) {
	for (int i = 0; i < 5; ++i) {
		cout << who % 52 << " ";
		who /= 52;
	}
	cout << endl;
}

struct Hand {
	int type, dot, col, who;
	Hand() {
	}
	Hand(int type, int dot, int col, int who) {
		this->type = type;
		this->dot = dot;
		this->col = col;
		this->who = who;
	}

	bool operator<(const Hand&o) const {
		if (type != o.type)
			return type > o.type;
		if (dot != o.dot)
			return dot < o.dot;
		if (col != o.col)
			return col < o.col;
		return false;
	}
};

Card cards[5];

bool isStright(int&dot, int&col) {
	bool ok = true;
	for (int i = 1; i < 5; ++i) {
		if (cards[i].p != cards[0].p + i)
			ok = false;
	}
	if (ok) {
		dot = cards[0].p, col = 0;
		for (int i = 5 - 1; i >= 0; --i) {
			col = col * 4 + cards[i].r;
		}
		return true;
	}

	//check A2345
	if (cards[4].p == 14) {
		ok = true;
		for (int i = 0; i < 4; ++i) {
			if (cards[i].p != i + 2)
				ok = false;
		}
		if (ok) { //good! A2345
			dot = 1, col = 0;
			for (int i = 4 - 1; i >= 0; --i) {
				col = col * 4 + cards[i].r;
			}
			col = col * 4 + cards[4].r;
			return true;
		}
	}

	return false;
}

bool isFlush() {
	for (int i = 1; i < 5; ++i) {
		if (cards[i].r != cards[0].r)
			return false;
	}
	return true;
}

string makeIt(int&dot, int&col) {
	pair<int, Card> ps[5];
	string cnt = "";
	for (int i = 0, j; i < 5; i = j) {
		for (j = i; j < 5 && cards[j].p == cards[i].p; ++j)
			;
		cnt += char('0' + j - i);
		for (int k = i; k < j; ++k) {
			ps[k] = make_pair(j - i, cards[k]);
		}
	}
	sort(ps, ps + 5);

	dot = col = 0;

	for (int i = 5 - 1; i >= 0; --i) {
		dot = dot * 13 + ps[i].second.p - 2;
		col = col * 4 + ps[i].second.r;
	}

	sort(cnt.rbegin(), cnt.rend());
	return cnt;
}

void makeItSorted(int&dot, int&col) {
	dot = 0, col = 0;
	for (int i = 5 - 1; i >= 0; --i) {
		dot = dot * 13 + cards[i].p - 2;
		col = col * 4 + cards[i].r;
	}
}

Hand process() {
	sort(cards, cards + 5);
	int dot, col, who = 0;

	for (int i = 0; i < 5; ++i) {
		who = who * 52 + cards[i].id();
	}

	bool isS = isStright(dot, col);
	bool isF = isFlush();
	int type;

	int dotG, colG;
	string cnt = makeIt(dotG, colG);

	if (isS && isF) {
		type = 1;
		return Hand(type, dot, col, who);
	}

	if (cnt == "41") {
		type = 2;
		return Hand(type, dotG, colG, who);
	}

	if (cnt == "32") {
		type = 3;
		return Hand(type, dotG, colG, who);
	}

	int dotS, colS;
	makeItSorted(dotS, colS);

	if (isF) {
		type = 4;
		return Hand(type, dotS, colS, who);
	}

	if (isS) {
		type = 5;
		return Hand(type, dot, col, who);
	}

	if (cnt == "311") {
		type = 6;
		return Hand(type, dotG, colG, who);
	}

	if (cnt == "221") {
		type = 7;
		return Hand(type, dotG, colG, who);
	}

	if (cnt == "2111") {
		type = 8;
		return Hand(type, dotG, colG, who);
	}

	type = 9;
	return Hand(type, dotS, colS, who);
}

typedef long long int64;

bool myCard[52] = { }, youCard[52] = { };
int n;

void readCardSet(int n, bool cardSet[]) {
	Card c;
	for (int i = 0; i < n; ++i) {
		c.read();
		cardSet[c.id()] = true;
	}
}

Card allCard[52];

Card cur[5];

const int MAX_HANDS = 3000000 + 10;

Hand hands[MAX_HANDS];
int nHands = 0;

bool check(int who, bool cardSet[], int n) {
	int cnt = 0;
	for (int i = 0; i < 5; ++i) {
		int x = who % 52;
		who /= 52;
		if (cardSet[x])
			++cnt;
	}
	return cnt == n;
}

void dfs(int u, int who, int cnt) { //search all cardSet combination
	if (u == 52) {
		if (cnt == 5 && (check(who, myCard, n) || check(who, youCard, n - 1))) {
			memcpy(cards, cur, sizeof cur);
			hands[nHands++] = process();
		}
		return;
	}
	dfs(u + 1, who, cnt);
	if (cnt < 5) {
		cur[cnt] = allCard[u];
		dfs(u + 1, who * 52 + u, cnt + 1);
	}
}

bool check(const Hand&h, bool cardSet[], int n) {
	return check(h.who, cardSet, n);
}

typedef long long int64;

int64 comb(int n, int m) {
	int64 ret = 1;
	for (int i = 0; i < m; ++i) {
		ret *= n - i;
		ret /= i + 1;
	}
	return ret;
}

int id[5];

int64 A, B;

int dp[6][60]; //rem,max

void prepare() {
	memset(dp, 0, sizeof dp);
	fill(dp[0], dp[0] + 60, 1);
	for (int rem = 1; rem <= 5; ++rem) {
		for (int mx = 1; mx <= 52; ++mx) {
			dp[rem][mx] += dp[rem][mx - 1] + dp[rem - 1][mx - 1];
		}
	}
}

int curv[5];
int mp[4000000];

int eval(int v[], int n) {
	int code = 0;
	for (int i = 0; i < n; ++i) {
		code += dp[i][52];
	}
	for (int i = 0; i < n; ++i)
		code += dp[n - i][v[i]];
	return code;
}

void dfs1(int u, int cnt) {
	if (u == 5) {
		int cd = eval(curv, cnt);
		A += (cnt % 2 == 0 ? 1 : -1) * mp[cd];
		return;
	}
	dfs1(u + 1, cnt);
	curv[cnt] = id[u];
	dfs1(u + 1, cnt + 1);
}

void dfs2(int u, int cnt) {
	if (u == 5) {
		int cd = eval(curv, cnt);
		mp[cd]++;
		return;
	}
	dfs2(u + 1, cnt);
	curv[cnt] = id[u];
	dfs2(u + 1, cnt + 1);
}

int main() {
	cin >> n;
	readCardSet(n, myCard);
	readCardSet(n - 1, youCard);

	for (int i = 2; i <= 14; ++i) {
		for (int j = 0; j < 4; ++j) {
			Card c(i, j);
			allCard[c.id()] = c;
		}
	}

	dfs(0, 0, 0);

	sort(hands, hands + nHands);

	int rem = 52 - n - (n - 1);
	A = 0, B = 1;
	B *= comb(rem, 5 - n);
	rem -= 5 - n;
	B *= comb(rem, 5 - (n - 1));

	prepare();

	for (int i = 0; i < nHands; ++i) {
		Hand&h = hands[i];
		if (check(h, myCard, n)) {
			int who = h.who;
			for (int j = 0; j < 5; ++j) {
				id[j] = who % 52;
				who /= 52;
			}

			sort(id, id + 5);
			reverse(id, id + 5);
			dfs1(0, 0);
		}

		if (check(h, youCard, n - 1)) {
			int who = h.who;
			for (int j = 0; j < 5; ++j) {
				id[j] = who % 52;
				who /= 52;
			}
			sort(id, id + 5);
			reverse(id, id + 5);
			dfs2(0, 0);
		}
	}

	int64 G = __gcd(A, B);
	A /= G, B /= G;
	cout << A << "/" << B << endl;

	return 0;
}
