Quine 블로그 시리즈의 이전 모험에서 우리는 Quines와 Intron을 직접 작성하는 방법을 탐색했습니다. 오늘은 QuineRelays가 무엇인지, Introns를 활용하여 어떻게 QuineRelays를 만들 수 있는지 알아보겠습니다.
각 프로그램이 서로 다른 언어로 작성된 일련의 프로그램을 상상해 보세요. 이 원의 각 프로그램에는 단일 목적이 있습니다. 즉, 시퀀스에서 다음 프로그램의 소스 코드를 인쇄하는 것입니다. 그런 다음 마지막 프로그램은 첫 번째 프로그램의 소스 코드를 인쇄하여 루프를 닫습니다.
기본적으로 QuineRelays는 n개의 다양한 언어로 제공되는 n개의 프로그램 세트입니다.
개념을 더 잘 이해하기 위해 몇 가지 예를 살펴보겠습니다.
이 2차 QuineRelay는 Python 프로그램을 인쇄한 다음 원래 JavaScript 프로그램을 다시 인쇄하는 JavaScript 프로그램을 특징으로 합니다. 자기복제의 다이나믹한 듀오입니다.
자바스크립트 → 파이썬 ⥀
JavaScript: 온라인으로 사용해 보세요!
console.log((q=_=>`print(${JSON.stringify(`console.log((q=${q+[]})())`)})`)())
Python: 온라인에서 사용해 보세요!
print("console.log((q=_=>`print(${JSON.stringify(`console.log((q=${q+[]})())`)})`)())")
3차 QuineRelay를 사용하면 상황이 더욱 흥미로워집니다. 이것은 Python 프로그램을 출력하고 Ruby 프로그램을 출력하는 Haskell 프로그램으로 시작하고 마지막으로 Ruby 프로그램은 원래 Haskell 프로그램으로 루프백됩니다.
하스켈 → Python2 → Ruby ⥀
하스켈: 온라인에서 사용해 보세요!
q a b c=putStrLn $ b ++ [toEnum 10,'q','('] ++ show b ++ [','] ++ show c ++ [','] ++ show a ++ [')'] main=q "q a b c=putStrLn $ b ++ [toEnum 10,'q','('] ++ show b ++ [','] ++ show c ++ [','] ++ show a ++ [')']" "def q(a,b,c):print b+chr(10)+'q('+repr(b)+','+repr(c)+','+repr(a)+')'" "def e(x) return 34.chr+x+34.chr end;def q(a,b,c) print b+10.chr+'main=q '+e(b)+' '+e(c)+' '+e(a)+' '+10.chr end"
Python2: 온라인으로 사용해 보세요!
def q(a,b,c):print b+chr(10)+'q('+repr(b)+','+repr(c)+','+repr(a)+')' q("def q(a,b,c):print b+chr(10)+'q('+repr(b)+','+repr(c)+','+repr(a)+')'","def e(x) return 34.chr+x+34.chr end;def q(a,b,c) print b+10.chr+'main=q '+e(b)+' '+e(c)+' '+e(a)+' '+10.chr end","q a b c=putStrLn $ b ++ [toEnum 10,'q','('] ++ show b ++ [','] ++ show c ++ [','] ++ show a ++ [')']")
루비: 온라인으로 체험해 보세요!
def e(x) return 34.chr+x+34.chr end;def q(a,b,c) print b+10.chr+'main=q '+e(b)+' '+e(c)+' '+e(a)+' '+10.chr end q("def e(x) return 34.chr+x+34.chr end;def q(a,b,c) print b+10.chr+'main=q '+e(b)+' '+e(c)+' '+e(a)+' '+10.chr end","q a b c=putStrLn $ b ++ [toEnum 10,'q','('] ++ show b ++ [','] ++ show c ++ [','] ++ show a ++ [')']","def q(a,b,c):print b+chr(10)+'q('+repr(b)+','+repr(c)+','+repr(a)+')'")
다음은 4차 QuineRelay입니다: Ruby → Java → C# → Python ⥀
GitHub
문화 전반에 걸쳐 몇 가지 공유된 상징이 있습니다. 그 중 하나는 이집트, 그리스, 로마, 힌두교, 시베리아, 북유럽, 아프리카 및 남미 신화에 언급된 우로보로스입니다. 우로보로스(Ouroboros)는 자신의 꼬리를 먹고 있는 뱀이나 용을 묘사한 고대 상징으로, 창조와 파괴의 영원한 순환을 상징합니다.
노르드 신화에서 요르문간드(Jörmungandr)는 로키(Loki)와 거인 앙그르보다(Angrboða)의 둘째 아들인 거대한 뱀입니다. 오딘은 미드가르드(인간의 영역)를 둘러싼 바다에 요르문간드를 던졌고, 그곳에서 뱀은 너무 거대해져서 세상을 둘러싸 자신의 꼬리를 잡았습니다. 미드가르드(지구)를 둘러싸고 있기 때문에 세계 뱀(World Serpent), 우로보로스(Ouroboros)라고 불립니다. 요르문간드가 꼬리를 놓는 것은 라그나로크(세계 최후의 전쟁)의 시작을 알리는 신호 중 하나입니다.
퀸 릴레이는 이러한 상징을 아주 깔끔하게 구현하고 있는데, 주기의 각 프로그램이 다음 프로그램을 생성하고 결국 다시 태어나기 때문에 우로보로스 프로그램이라는 별명이 붙습니다.
자리를 지켜주세요. 다음은 128차 Ouroboros QuineRelay입니다. 예, 당신이 읽은 것이 맞습니다. 128 !!!
Ruby → Rust → Scala → ... (다른 120개) ... → Python → R → Ratfor → rc → REXX ⥀
GitHub
이것만으로는 부족했는지 부활절 달걀이 포함되어 있습니다. 축소한 원본 Ruby 코드에는 우로보로스 드래곤이 포함되어 있습니다!
앞서 작성한 Python 인트론을 시작하여 2차 QuineRelay로 바꿔 보겠습니다.
Python Intron: 온라인에서 사용해 보세요!
intron = 'wubbalubbadubdub' data = "print('intron =', repr(intron)); print('data =', repr(data)); print(data)" print('intron =', repr(intron)); print('data =', repr(data)); print(data)
인트론의 마법을 사용하여 이제 다른 언어의 형제 Quine의 코드 부분을 인트론에 쉽게 넣을 수 있습니다. 다음 형식의 프로그램 제작:
파이썬:
intron = "code part of sibling" data = "code part of self" print('intron =', repr(intron)); print('data =', repr(data)); print(data)
각 변수는 단지 다른 Quine의 데이터 역할을 할 뿐입니다. data와 intron의 이름을 각각 d1과 d2로 바꾸겠습니다.
파이썬:
d1 = "code part of self" d2 = "code part of sibling" print('d1 =', repr(d1)); print('d2 =', repr(d2)); print(d1)
이제 d2는 인트론 역할을 하지만 위 프로그램은 여전히 self의 코드 부분을 인쇄하려고 시도했습니다. 다음의 소스를 인쇄하려면 마지막에 print(d1) 대신 print(d2)를 사용합시다.
파이썬:
d1 = "code part of self" d2 = "code part of sibling" print('d1 =', repr(d1)); print('d2 =', repr(d2)); print(d2)
우리는 이미 d1의 내용이 3행의 복사본이라는 것을 알고 있습니다. 하지만 아직 d2의 내용은 없습니다.
예를 들어, JavaScript로 QuineRelay를 만들고 싶었습니다. JS에서도 비슷한 인트론을 작성해 보겠습니다.
자바스크립트:
d1 = "code part of sibling" d2 = "code part of self" console.log(`d1 = ${JSON.stringify(d1)}`); console.log(`d2 = ${JSON.stringify(d2)}`); console.log(d1);
자 위 JS 인트론의 3번째 줄이 바로 우리가 원하던 형제 프로그램의 코드입니다!
서로의 코드를 상대방의 인트론으로 붙여넣으세요.
메모. 일부 인용 불일치를 방지하려면 js에 d1 + ''를 추가해야 합니다
Python: Try it online!
d1 = "print('d1 =', repr(d1)); print('d2 =', repr(d2)); print(d2)" d2 = "console.log(`d1 = ${JSON.stringify(d1)}`); console.log(`d2 = ${JSON.stringify(d2)}`); console.log(d1 + '');" print('d1 =', repr(d1)); print('d2 =', repr(d2)); print(d2)
JavaScript: Try it online!
d1 = "print('d1 =', repr(d1)); print('d2 =', repr(d2)); print(d2)" d2 = "console.log(`d1 = ${JSON.stringify(d1)}`); console.log(`d2 = ${JSON.stringify(d2)}`); console.log(d1 + '');" console.log(`d1 = ${JSON.stringify(d1)}`); console.log(`d2 = ${JSON.stringify(d2)}`); console.log(d1 + '');
There you go. It's a proper 2nd order QuineRelay! A Python program, that prints a JavaScript program, that prints the original Python program back in a cycle.
Creating a QuineRelay is an exercise in creative coding and understanding how different languages represent and manipulate strings. It involves weaving together introns from various programs, each containing the code to replicate its next neighbour.
At its core, an nth-order relay is a game of n clever ways to escape quotes across n programming languages.
Stay tuned for the next post on MultiQuine!
Sources and references:
위 내용은 QuineRelay: Quine 뱀의 탄생의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!