結果

問題 No.274 The Wall
ユーザー KazunKazun
提出日時 2021-08-24 15:38:09
言語 PyPy3
(7.3.15)
結果
AC  
実行時間 195 ms / 2,000 ms
コード長 4,684 bytes
コンパイル時間 323 ms
コンパイル使用メモリ 82,428 KB
実行使用メモリ 79,052 KB
最終ジャッジ日時 2024-11-14 14:32:17
合計ジャッジ時間 3,708 ms
ジャッジサーバーID
(参考情報)
judge2 / judge3
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 38 ms
54,504 KB
testcase_01 AC 39 ms
54,580 KB
testcase_02 AC 38 ms
53,624 KB
testcase_03 AC 53 ms
66,168 KB
testcase_04 AC 39 ms
53,624 KB
testcase_05 AC 38 ms
53,032 KB
testcase_06 AC 38 ms
54,500 KB
testcase_07 AC 40 ms
53,300 KB
testcase_08 AC 39 ms
53,840 KB
testcase_09 AC 38 ms
53,340 KB
testcase_10 AC 38 ms
54,564 KB
testcase_11 AC 53 ms
66,536 KB
testcase_12 AC 76 ms
69,608 KB
testcase_13 AC 56 ms
65,808 KB
testcase_14 AC 96 ms
76,668 KB
testcase_15 AC 125 ms
77,784 KB
testcase_16 AC 56 ms
68,856 KB
testcase_17 AC 54 ms
67,108 KB
testcase_18 AC 52 ms
66,732 KB
testcase_19 AC 165 ms
78,044 KB
testcase_20 AC 180 ms
78,356 KB
testcase_21 AC 191 ms
78,748 KB
testcase_22 AC 179 ms
78,048 KB
testcase_23 AC 182 ms
78,404 KB
testcase_24 AC 195 ms
79,052 KB
testcase_25 AC 193 ms
79,024 KB
権限があれば一括ダウンロードができます

ソースコード

diff #

class Two_SAT:
    def __init__(self,N):
        """ N 変数の 2-SAT を定義する.

        N: int
        """

        self.N=N
        self.var_num=N

        self.arc=[[] for _ in range(2*N)]
        self.rev=[[] for _ in range(2*N)]

    def var_to_index(self,v):
        if v>=0:
            return 2*v
        else:
            return 2*(-v-1)+1

    def index_to_var(self,i):
        if i%2:
            return -(i+1)//2
        else:
            return i//2

    def add_variable(self,k):
        """ 新たに変数 k 個を加える.
        """

        m=self.var_num
        self.var_num+=k

        self.arc+=[[] for _ in range(2*k)]
        self.rev+=[[] for _ in range(2*k)]

        return list(range(m,m+k))

    def __add_clause(self,i,j):
        self.arc[self.var_to_index(i)].append(self.var_to_index(j))
        self.rev[self.var_to_index(j)].append(self.var_to_index(i))

    def add_imply(self,i,j):
        """ X_i -> X_j を加える.
        """
        self.__add_clause(i,j)
        self.__add_clause(~j,~i)

    def add_or(self,i,j):
        """ X_i or X_j を加える.
        """

        self.add_imply(~i,j)

    def add_nand(self,i,j):
        """ not (X_i and X_j) を加える.
        """

        self.add_imply(i,~j)

    def add_equal(self,*I):
        """ I=[i_0, ..., i_{k-1}] に対して, X_{i_0}=...=X_{i_{k-1}} を追加する.
        """

        k=len(I)

        if k<=1:
            return

        for j in range(k-1):
            self.add_imply(I[j],I[j+1])
        self.add_imply(I[-1],I[0])

    def add_not_equal(self,i,j):
        """ X_i != X_j を追加する.
        """
        self.add_equal(i,~j)

    def set_true(self,i):
        """ 変数 X_i を True にする.
        """

        self.__add_clause(~i,i)

    def set_false(self,i):
        """ 変数 X_i を False にする.
        """

        self.__add_clause(i,~i)

    def at_most_one(self,*I):
        """ X_i (i in I) を満たすような i は高々1つだけという条件を追加する.
        """

        k=len(I)

        if k<=1:
            return

        A=self.add_variable(k)

        self.add_imply(I[0],A[0])
        for i in range(1,k):
            self.add_imply(A[i-1],A[i])
            self.add_imply(I[i],A[i])
            self.add_nand(A[i-1],I[i])

    def is_satisfy(self,Mode=0):
        """ Two-SAT は充足可能?

        Mode:
        0 (Defalt): 充足可能?
        1: 充足可能ならば,その変数の割当を与える (不可能なときはNone).
        2: 充足不能の原因である変数を全て挙げる.
        """
        N=self.var_num
        Group=[0]*(2*N)
        Order=[]

        for s in range(2*N):
            if Group[s]:continue

            S=[s]
            Group[s]=-1

            while S:
                u=S.pop()
                for v in self.arc[u]:
                    if Group[v]:continue
                    Group[v]=-1
                    S.append(u);S.append(v)
                    break
                else:
                    Order.append(u)

        K=0
        for s in Order[::-1]:
            if Group[s]!=-1:continue

            S=[s]
            Group[s]=K

            while S:
                u=S.pop()
                for v in self.rev[u]:
                    if Group[v]!=-1:continue

                    Group[v]=K
                    S.append(v)
            K+=1

        if Mode==0:
            for i in range(N):
                if Group[2*i]==Group[2*i+1]:
                    return False
            return True
        elif Mode==1:
            T=[0]*N
            for i in range(N):
                if Group[2*i]>Group[2*i+1]:
                    T[i]=1
                elif Group[2*i]==Group[2*i+1]:
                    return None
            return T[:self.N]
        elif Mode==2:
            return [i for i in range(self.N) if Group[2*i]==Group[2*i+1]]
#================================================
import sys
input=sys.stdin.readline

N,M=map(int,input().split())
L =[0]*N;R =[0]*N
IL=[0]*N;IR=[0]*N

for i in range(N):
    L[i],R[i]=map(int,input().split())
    IL[i],IR[i]=M-1-R[i],M-1-L[i]

T=Two_SAT(N)
for i in range(N):
    for j in range(i+1,N):
        cnt=0
        if not(R[i]<L[j] or R[j]<L[i]):
            cnt+=1
            T.add_or(i,j)

        if not(IR[i]<L[j] or R[j]<IL[i]):
            cnt+=1
            T.add_or(~i,j)

        if not(R[i]<IL[j] or IR[j]<L[i]):
            cnt+=1
            T.add_or(i,~j)

        if not(IR[i]<IL[j] or IR[j]<IL[i]):
            cnt+=1
            T.add_or(~i,~j)

        if cnt==4:
            break
    if cnt==4:
        break

print("YES" if T.is_satisfy() else "NO")
0