# ----------------------------------------------------------------------------
# -- HNU CE 데이터베이스(2025년 2학기 02분반): FD 관련 프로그래밍 과제 Part B
# ----------------------------------------------------------------------------
# -- 이름: 이은섭
# -- 학번: 20210508
# ----------------------------------------------------------------------------
import io
from contextlib import redirect_stdout
# 위 두개 import하는 이유:
# (2) 구할때 (1)의 closure() 사용하려고 하는데
# 함수 안에 print문이 출력 안되게 하려고
# (1) 3.2.4절 Algorithm 3.7 관련 closure 함수
# closure : FD 집합 , 속성 집합 -> 속성 집합
# closure(S, {A1,A2...})는 {A1,A2,...}+를 계산
def closure( S, R) :
# 1. split
temp = [ ]
for FD in S:
if len ( FD[ 1 ] ) >= 2 :
for j in range ( len ( FD[ 1 ] ) ) :
temp.append ( ( FD[ 0 ] , [ FD[ 1 ] [ j] ] ) )
S.remove ( FD)
S += temp
# 2. 초기화
X = R.copy ( )
# 3. 반복 확장
temp = [ ]
while ( temp != X) :
temp = X.copy ( )
for FD in S:
if set ( FD[ 0 ] ) .issubset ( set ( X) ) and not ( set ( FD[ 1 ] ) .issubset ( set ( X) ) ) :
X += FD[ 1 ]
X.sort ( )
return X
# (2) 주어진 한 FD가 기존의 FD 집합으로부터 유도 가능한지 검사하는 함수 (closure 활용하면 간단)
# is_derived_from : FD , FD 집합 -> Bool
# is_derived_from(fd, S)는 fd가 S로부터 유도 가능하면 참, 그렇지 않으면 거짓
def is_derived_from( fd, S) :
# 1. 주어진 fd에서 좌항 뽑아내 리스트(R형식으로 만들기)
R = fd[ 0 ]
# print("========fd to List========")
# print(R)
# 2. closure 함수 이용해 좌항에 있는 속성의 클로저 구하기
f = io.StringIO ( ) # closure()안에 있는 print안나오게 하려고 씀
with redirect_stdout( f) :
X = closure( S, R)
# print("========closure calcurate========")
# print(R)
# 3. X값 안에 fd의 우항이 있는지 확인
derived = set ( fd[ 1 ] ) .issubset ( set ( X) )
# 4. 결과 값 반환
return derived
# (3) 3.2.7절 관련 주어진 FD 집합이 기존 FD집합의 basis인지 검사
# is_basis_of :: FD 집합 , FD 집합 -> Bool
# is_basis_of(B, S)는 B가 S의 basis이면 (minimal이 아닌 경우도 포함) 참, 그렇지 않으면 거짓
def is_basis_of( B, S) :
# 1. 두 FD들의 집합 B, S의 좌항에 있는 속성 뽑아 합집합으로 만듬
lefts = [ ]
for FD in B:
if ( FD[ 0 ] not in lefts) :
lefts.append ( FD[ 0 ] )
for FD in S:
if ( FD[ 0 ] not in lefts) :
lefts.append ( FD[ 0 ] )
# print("========Combine leftS and leftB========")
# print(lefts)
# 2. B와 S각각 lefts의 속성들 closure함수 사용하여 비교 둘이 같으면 basis O 틀리면 X
basis = True
for attr in lefts:
f = io.StringIO ( ) # closure()안에 있는 print안나오게 하려고 씀
with redirect_stdout( f) :
if closure( S, attr) != closure( B, attr) :
basis = False
break
# 3. 출력 및 반환
return basis
# (4) 3.2.7절 관련 주어진 FD 집합이 minimal인지 검사
# is_minimal :: [FD] -> Bool
# is_minimal(B)는 B가 3.2.7절에 제시된 기준에 따라 minimal하면 참, 그렇지 않으면 거짓
def is_minimal( B) :
answer = True
# 1. 모든 FD의 우항이 singleton인지 검사
for FD in B:
if len ( FD[ 1 ] ) != 1 : # 2개 이상일 경우 minimal 아님
answer = False
return answer
# 2. FD 하나라도 제거하면 동등성 깨지는지 검사
for FD in B:
temp = B.copy ( )
temp.remove ( FD)
if is_basis_of( B, temp) : # 제거해도 동등하면 minimal 아님
answer = False
return answer
# 3. FD의 좌항에서 속성을 줄이면 동등성 깨지는지 검사
for FD in B:
left = FD[ 0 ]
right = FD[ 1 ]
if len ( left) >= 2 :
for attr in left:
reduced_left = left.copy ( )
reduced_left.remove ( attr)
temp = B.copy ( )
temp.remove ( FD)
temp.append ( ( reduced_left, right) )
if is_basis_of( B, temp) : # 왼쪽 속성 줄여도 동등하면 minimal 아님
answer = False
return answer
return answer
# (5) 3.2.8절 Algorithm 3.12
# project_FDs :: [Attr] -> [FD] -> [Attr] -> [FD]
# project_FDs(L, S, L1)은
# 원래 관계(R)의 속성 집합을 L과 FD집합을 S라 할 때
# L의 부분집합인 L1을 속성으로 하는 (즉, R의 프로젝션인 R1) 관계에서 성립하는 성질을 나타내는 FD집합(즉, S의 프로젝션인 S1)을 계산
def project_FDs( L, S, L1) :
T = [ ] # projection 결과 잠정 FD 집합
# 1. L1의 각 속성에 대해 closure 계산 → L1 내부에서 유효한 FD 추가
for attr in L1:
X = closure( S, [ attr] )
for i in X:
if i not in [ attr] and i in L1: # 자기 자신 제외, L1 내부 속성만
T.append ( ( [ attr] , [ i] ) )
# 2. 중복/비최소 FD 제거 (minimal form으로 정리)
i = 0
while i < len ( T) :
temp = T.copy ( )
FD = T[ i]
temp.remove ( FD)
if is_basis_of( T, temp) : # 제거해도 동등하면 삭제
T.remove ( FD)
i = 0 # 다시 처음부터 확인
else :
i += 1
return T
# ===============================TEST===============================
# (1)-1. closure함수 예제 1 (Example 3.8)
S = [
( [ 'A' , 'B' ] , [ 'C' ] ) ,
( [ 'B' , 'C' ] , [ 'A' , 'D' ] ) ,
( [ 'C' , 'F' ] , [ 'B' ] ) ,
( [ 'D' ] , [ 'E' ] )
]
R = [ 'A' , 'B' ]
print ( "== Example for closure test 1=============================" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
X = closure( S, R)
print ( "{" , end= '' )
for i in R:
print ( i, end= ',' if R.index ( i) != R.index ( R[ -1 ] ) else "" )
print ( "}+ = " , end= '' )
print ( "{" , end= '' )
for i in X:
print ( i, end= ',' if X.index ( i) != X.index ( X[ -1 ] ) else "" )
print ( "}" )
print ( "" )
# (1)-2. closure함수 예제 2 (MyExample)
R = [ 'C' , 'F' ]
print ( "== Example for closure test 2=============================" )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
X = closure( S, R)
print ( "{" , end= '' )
for i in R:
print ( i, end= ',' if R.index ( i) != R.index ( R[ -1 ] ) else "" )
print ( "}+ = " , end= '' )
print ( "{" , end= '' )
for i in X:
print ( i, end= ',' if X.index ( i) != X.index ( X[ -1 ] ) else "" )
print ( "}" )
print ( "" )
# (2)-1. is_drived_from함수 예제 1 (MyExample)
S = [
( [ 'A' , 'B' ] , [ 'C' ] ) ,
( [ 'B' , 'C' ] , [ 'A' , 'D' ] ) ,
( [ 'C' , 'F' ] , [ 'B' ] ) ,
( [ 'D' ] , [ 'E' ] )
]
fd = ( [ 'A' , 'B' ] , [ 'C' , 'D' ] )
print ( "== Example for is_derived_from test 1=====================" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
derived = is_derived_from( fd, S)
print ( "{" + ' ' .join ( fd[ 0 ] ) , "->" , ' ' .join ( fd[ 1 ] ) + "}" , " is derived from S :" , derived)
print ( "" )
# (2)-2. is_drived_from함수 예제 2 (MyExample)
S = [
( [ 'A' , 'B' ] , [ 'C' ] ) ,
( [ 'B' , 'C' ] , [ 'A' , 'D' ] ) ,
( [ 'C' , 'F' ] , [ 'B' ] ) ,
( [ 'D' ] , [ 'E' ] )
]
fd = ( [ 'A' , 'B' ] , [ 'C' , 'D' , 'F' ] )
print ( "== Example for is_derived_from test 2=====================" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
derived = is_derived_from( fd, S)
print ( "{" + ' ' .join ( fd[ 0 ] ) , "->" , ' ' .join ( fd[ 1 ] ) + "}" , " is derived from S :" , derived)
print ( "" )
# (3)-1. is_basis_of 함수 예제 1 (example 3.11)
S = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'A' ] )
]
B = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'A' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'B' ] )
]
print ( "== Example for is_basis_of test 1=====================" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
print ( "B = {" , end= '' )
for FD in B:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if B.index ( FD) != B.index ( B[ -1 ] ) else "" )
print ( "}" )
basis = is_basis_of( B, S)
print ( "B is basis of S :" , basis)
print ( "" )
# (3)-2. is_basis_of 함수 예제 2(MyExample)
S = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'A' ] )
]
B = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'A' ] ) ,
( [ 'D' ] , [ 'F' ] ) ,
( [ 'C' ] , [ 'B' ] )
]
print ( "== Example for is_basis_of test 2=====================" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
print ( "B = {" , end= '' )
for FD in B:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if B.index ( FD) != B.index ( B[ -1 ] ) else "" )
print ( "}" )
basis = is_basis_of( B, S)
print ( "B is basis of S :" , basis)
print ( "" )
# (4)-1 is_minimal 함수 예제1 (example 3.11)
print ( "== Example for is_minimal test 1===========================" )
B = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'A' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'B' ] )
]
print ( "B = {" , end= '' )
for FD in B:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if B.index ( FD) != B.index ( B[ -1 ] ) else "" )
print ( "}" )
minimal = is_minimal( B)
print ( "B is minimal :" , minimal)
print ( "" )
# (4)-2 is_minimal 함수 예제2 (my example)
print ( "== Example for is_minimal test 2===========================" )
B = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'A' ] , [ 'C' ] )
]
print ( "B = {" , end= '' )
for FD in B:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if B.index ( FD) != B.index ( B[ -1 ] ) else "" )
print ( "}" )
minimal = is_minimal( B)
print ( "B is minimal :" , minimal)
print ( "" )
# (5)-1. project_FDs 함수 예제 (example 3.13)
L = [ 'A' , 'B' , 'C' , 'D' ]
S = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'D' ] )
]
L1 = [ 'A' , 'C' , 'D' ]
print ( "== Example for project_FDs test 1===========================" )
print ( "L = {" , end= '' )
for FD in L:
print ( FD, end= "," if L.index ( FD) != L.index ( L[ -1 ] ) else "" )
print ( "}" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
print ( "L1 = {" , end= '' )
for FD in L1:
print ( FD, end= "," if L1.index ( FD) != L1.index ( L1[ -1 ] ) else "" )
print ( "}" )
S1 = project_FDs( L, S, L1)
print ( "S1 = {" , end= '' )
for FD in S1:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S1.index ( FD) != S1.index ( S1[ -1 ] ) else "" )
print ( "}" )
print ( "" )
# (5)-2. project_FDs 함수 예제 (my example)
L = [ 'A' , 'B' , 'C' , 'D' , 'E' ]
S = [
( [ 'A' ] , [ 'B' ] ) ,
( [ 'A' ] , [ 'C' ] ) ,
( [ 'B' ] , [ 'C' ] ) ,
( [ 'C' ] , [ 'D' ] ) ,
( [ 'D' ] , [ 'E' ] )
]
L1 = [ 'A' , 'B' , 'D' ]
print ( "== Example for project_FDs test 2===========================" )
print ( "L = {" , end= '' )
for FD in L:
print ( FD, end= "," if L.index ( FD) != L.index ( L[ -1 ] ) else "" )
print ( "}" )
print ( "S = {" , end= '' )
for FD in S:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S.index ( FD) != S.index ( S[ -1 ] ) else "" )
print ( "}" )
print ( "L1 = {" , end= '' )
for FD in L1:
print ( FD, end= "," if L1.index ( FD) != L1.index ( L1[ -1 ] ) else "" )
print ( "}" )
S1 = project_FDs( L, S, L1)
print ( "S1 = {" , end= '' )
for FD in S1:
print ( ' ' .join ( FD[ 0 ] ) , "->" , ' ' .join ( FD[ 1 ] ) , end= ", " if S1.index ( FD) != S1.index ( S1[ -1 ] ) else "" )
print ( "}" )
print ( "" )
# ----------------------------------------------------------------------------
# -- HNU CE 데이터베이스(2025년 2학기 02분반): FD 관련 프로그래밍 과제 Part B
# ----------------------------------------------------------------------------
# -- 이름: 이은섭
# -- 학번: 20210508
# ----------------------------------------------------------------------------


import io
from contextlib import redirect_stdout 
# 위 두개 import하는 이유: 
# (2) 구할때 (1)의 closure() 사용하려고 하는데 
# 함수 안에 print문이 출력 안되게 하려고

# (1) 3.2.4절 Algorithm 3.7 관련 closure 함수

# closure : FD 집합 , 속성 집합 -> 속성 집합
# closure(S, {A1,A2...})는 {A1,A2,...}+를 계산
def closure(S, R):
    

    # 1. split
    temp = []
    for FD in S:
        if len(FD[1]) >= 2:
            for j in range(len(FD[1])):
                temp.append((FD[0], [FD[1][j]]))
            S.remove(FD)
    S += temp

    # 2. 초기화
    X = R.copy()
    
    # 3. 반복 확장
    temp = []
    while(temp != X):
        temp = X.copy()
        for FD in S:
            if set(FD[0]).issubset(set(X)) and not(set(FD[1]).issubset(set(X))):
                X += FD[1]
    X.sort()
    
    return X




# (2) 주어진 한 FD가 기존의 FD 집합으로부터 유도 가능한지 검사하는 함수 (closure 활용하면 간단)

# is_derived_from : FD , FD 집합 -> Bool
# is_derived_from(fd, S)는 fd가 S로부터 유도 가능하면 참, 그렇지 않으면 거짓
def is_derived_from(fd, S):


    # 1. 주어진 fd에서 좌항 뽑아내 리스트(R형식으로 만들기)
    R = fd[0]
    # print("========fd to List========")
    # print(R)

    # 2. closure 함수 이용해 좌항에 있는 속성의 클로저 구하기
    f = io.StringIO() # closure()안에 있는 print안나오게 하려고 씀
    with redirect_stdout(f):
        X = closure(S, R)
    
    # print("========closure calcurate========")
    # print(R)

    # 3. X값 안에 fd의 우항이 있는지 확인
    derived = set(fd[1]).issubset(set(X))
    
    # 4. 결과 값 반환
    return derived



# (3)  3.2.7절 관련 주어진 FD 집합이 기존 FD집합의 basis인지 검사

# is_basis_of :: FD 집합 , FD 집합 -> Bool
# is_basis_of(B, S)는 B가 S의 basis이면 (minimal이 아닌 경우도 포함) 참, 그렇지 않으면 거짓
def is_basis_of(B, S):


    # 1. 두 FD들의 집합 B, S의 좌항에 있는 속성 뽑아 합집합으로 만듬
    lefts = []
    for FD in B:
        if(FD[0] not in lefts):
            lefts.append(FD[0])
    for FD in S:
        if(FD[0] not in lefts):
            lefts.append(FD[0])

    # print("========Combine leftS and leftB========")
    # print(lefts)

    # 2. B와 S각각 lefts의 속성들 closure함수 사용하여 비교 둘이 같으면 basis O 틀리면 X
    basis = True
    for attr in lefts:
        f = io.StringIO() # closure()안에 있는 print안나오게 하려고 씀
        with redirect_stdout(f):
            if closure(S, attr) != closure(B, attr):
                basis = False
                break
        
    # 3. 출력 및 반환
    return basis

# (4)  3.2.7절 관련 주어진 FD 집합이 minimal인지 검사
# is_minimal :: [FD] -> Bool             
# is_minimal(B)는 B가 3.2.7절에 제시된 기준에 따라 minimal하면 참, 그렇지 않으면 거짓

def is_minimal(B):
    answer = True

    # 1. 모든 FD의 우항이 singleton인지 검사
    for FD in B:
        if len(FD[1]) != 1:   # 2개 이상일 경우 minimal 아님
            answer = False
            return answer

    # 2. FD 하나라도 제거하면 동등성 깨지는지 검사
    for FD in B:
        temp = B.copy()
        temp.remove(FD)
        if is_basis_of(B, temp):  # 제거해도 동등하면 minimal 아님
            answer = False
            return answer

    # 3. FD의 좌항에서 속성을 줄이면 동등성 깨지는지 검사
    for FD in B:
        left = FD[0]
        right = FD[1]
        if len(left) >= 2:
            for attr in left:
                reduced_left = left.copy()
                reduced_left.remove(attr)
                temp = B.copy()
                temp.remove(FD)
                temp.append((reduced_left, right))
                if is_basis_of(B, temp):  # 왼쪽 속성 줄여도 동등하면 minimal 아님
                    answer = False
                    return answer

    return answer

# (5) 3.2.8절 Algorithm 3.12
# project_FDs :: [Attr] -> [FD] -> [Attr] -> [FD]
# project_FDs(L, S, L1)은
#   원래 관계(R)의 속성 집합을 L과 FD집합을 S라 할 때
#   L의 부분집합인 L1을 속성으로 하는 (즉, R의 프로젝션인 R1) 관계에서 성립하는 성질을 나타내는 FD집합(즉, S의 프로젝션인 S1)을 계산
def project_FDs(L, S, L1):
    T = []   # projection 결과 잠정 FD 집합

    # 1. L1의 각 속성에 대해 closure 계산 → L1 내부에서 유효한 FD 추가
    for attr in L1:
        X = closure(S, [attr])  
        for i in X:
            if i not in [attr] and i in L1:  # 자기 자신 제외, L1 내부 속성만
                T.append(([attr], [i]))

    # 2. 중복/비최소 FD 제거 (minimal form으로 정리)
    i = 0
    while i < len(T):
        temp = T.copy()
        FD = T[i]
        temp.remove(FD)
        if is_basis_of(T, temp):  # 제거해도 동등하면 삭제
            T.remove(FD)
            i = 0  # 다시 처음부터 확인
        else:
            i += 1

    return T



# ===============================TEST===============================

# (1)-1. closure함수 예제 1 (Example 3.8)
S = [
     (['A', 'B'], ['C']), 
     (['B', 'C'], ['A', 'D']), 
     (['C', 'F'], ['B']), 
     (['D'], ['E'])
     ]
R = ['A', 'B']
print("== Example for closure test 1=============================")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
X = closure(S, R)

print("{", end='')
for i in R:
    print(i, end=',' if R.index(i) != R.index(R[-1]) else "")
print("}+ = ", end='')

print("{", end='')
for i in X:
    print(i, end=',' if X.index(i) != X.index(X[-1]) else "")
print("}")
print("")

# (1)-2. closure함수 예제 2 (MyExample)
R = ['C', 'F']
print("== Example for closure test 2=============================")
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
X = closure(S, R)
print("{", end='')
for i in R:
    print(i, end=',' if R.index(i) != R.index(R[-1]) else "")
print("}+ = ", end='')

print("{", end='')
for i in X:
    print(i, end=',' if X.index(i) != X.index(X[-1]) else "")
print("}")
print("")


# (2)-1. is_drived_from함수 예제 1 (MyExample)
S = [
     (['A', 'B'], ['C']), 
     (['B', 'C'], ['A', 'D']), 
     (['C', 'F'], ['B']), 
     (['D'], ['E'])
     ]
fd = (['A', 'B'], ['C', 'D'])

print("== Example for is_derived_from test 1=====================")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
derived = is_derived_from(fd, S)
print("{" + ' '.join(fd[0]), "->", ' '.join(fd[1])+ "}", " is derived from S :", derived)
print("")

# (2)-2. is_drived_from함수 예제 2 (MyExample)
S = [
     (['A', 'B'], ['C']), 
     (['B', 'C'], ['A', 'D']), 
     (['C', 'F'], ['B']), 
     (['D'], ['E'])
     ]
fd = (['A', 'B'], ['C', 'D', 'F'])

print("== Example for is_derived_from test 2=====================")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
derived = is_derived_from(fd, S)
print("{" + ' '.join(fd[0]), "->", ' '.join(fd[1])+ "}", " is derived from S :", derived)
print("")


# (3)-1. is_basis_of 함수 예제 1 (example 3.11)
S = [
    (['A'], ['B']), 
    (['B'], ['C']), 
    (['C'], ['A'])
    ]

B = [
    (['A'], ['B']), 
    (['B'], ['A']), 
    (['B'], ['C']), 
    (['C'], ['B'])
    ]

print("== Example for is_basis_of test 1=====================")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")

print("B = {", end='')
for FD in B:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if B.index(FD) != B.index(B[-1]) else "")
print("}")
basis = is_basis_of(B, S)
print("B is basis of S :", basis)
print("")

# (3)-2. is_basis_of 함수 예제 2(MyExample)
S = [
    (['A'], ['B']), 
    (['B'], ['C']), 
    (['C'], ['A'])
    ]

B = [
    (['A'], ['B']), 
    (['B'], ['A']), 
    (['D'], ['F']), 
    (['C'], ['B'])
    ]

print("== Example for is_basis_of test 2=====================")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")

print("B = {", end='')
for FD in B:
    print(' '.join(FD[0]), "->",  ' '.join(FD[1]), end=", " if B.index(FD) != B.index(B[-1]) else "")
print("}")
basis = is_basis_of(B, S)
print("B is basis of S :", basis)
print("")


# (4)-1 is_minimal 함수 예제1 (example 3.11)
print("== Example for is_minimal test 1===========================")
B = [
    (['A'], ['B']), 
    (['B'], ['A']), 
    (['B'], ['C']), 
    (['C'], ['B'])
    ]
print("B = {", end='')
for FD in B:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if B.index(FD) != B.index(B[-1]) else "")
print("}")

minimal = is_minimal(B)
print("B is minimal :", minimal)
print("")

# (4)-2 is_minimal 함수 예제2 (my example)
print("== Example for is_minimal test 2===========================")
B = [
    (['A'], ['B']),
    (['B'], ['C']),
    (['A'], ['C'])
]
print("B = {", end='')
for FD in B:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if B.index(FD) != B.index(B[-1]) else "")
print("}")

minimal = is_minimal(B)
print("B is minimal :", minimal)
print("")

# (5)-1. project_FDs 함수 예제 (example 3.13)
L  = ['A', 'B', 'C', 'D']    
S  = [                       
    (['A'], ['B']),
    (['B'], ['C']),
    (['C'], ['D'])
]
L1 = ['A', 'C', 'D']

print("== Example for project_FDs test 1===========================")
print("L = {", end='')
for FD in L:
    print(FD, end="," if L.index(FD) != L.index(L[-1]) else "")
print("}")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
print("L1 = {", end='')
for FD in L1:
    print(FD, end="," if L1.index(FD) != L1.index(L1[-1]) else "")
print("}")

S1 = project_FDs(L, S, L1)
print("S1 = {", end='')
for FD in S1:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if S1.index(FD) != S1.index(S1[-1]) else "")
print("}")
print("")


# (5)-2. project_FDs 함수 예제 (my example)
L  = ['A', 'B', 'C', 'D', 'E']
S  = [
    (['A'], ['B']),
    (['A'], ['C']),
    (['B'], ['C']),
    (['C'], ['D']),
    (['D'], ['E'])
]
L1 = ['A', 'B', 'D']

print("== Example for project_FDs test 2===========================")
print("L = {", end='')
for FD in L:
    print(FD, end="," if L.index(FD) != L.index(L[-1]) else "")
print("}")
print("S = {", end='')
for FD in S:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if S.index(FD) != S.index(S[-1]) else "")
print("}")
print("L1 = {", end='')
for FD in L1:
    print(FD, end="," if L1.index(FD) != L1.index(L1[-1]) else "")
print("}")

S1 = project_FDs(L, S, L1)
print("S1 = {", end='')
for FD in S1:
    print(' '.join(FD[0]), "->", ' '.join(FD[1]), end=", " if S1.index(FD) != S1.index(S1[-1]) else "")
print("}")
print("")
