저번 포스팅에서 제가 올해 Advent of Code에 참여한다고 간략하게 언급한 적이 있습니다. 공교롭게도 퍼즐 중 하나, 특히 5일차에 게시된 퍼즐에서는 목록의 페이지 순서를 수정하는 작업이 포함됩니다. 정렬 알고리즘 구현에 대한 글을 올린 지 얼마 되지 않아서 글을 써야겠다는 생각이 들었습니다.
일부 정렬 알고리즘을 묘사한 귀여운 이미지
Advent of Code에 대해 아직 들어보지 못한 분들을 위해 Eric Wastl이 주최하는 연례 행사입니다. 매년 휴가철을 배경으로 한 이야기를 들려주는데, 올해는 모든 대형 크리스마스 썰매 출시에서 중요한 인물일 수 있는 수석 역사가를 찾는 이야기입니다. 이 챌린지는 매년 12월 1일부터 25일까지 진행됩니다. 매일 플롯이 진행되며 여기에는 프로그래밍 퍼즐이 포함되어 있습니다(입력도 함께 제공됩니다).
스토리 내레이션 내에서 퍼즐은 일반적으로 명확하게 정의되며 테스트 사례가 포함됩니다. 모든 퍼즐은 두 부분으로 나누어져 있으며, 두 번째 부분은 첫 번째 답변을 제출한 후에만 표시됩니다.
참여자는 파생된 답변이 일치하는 한 모든 언어로 모든 알고리즘을 구현하거나 프로그래밍을 완전히 건너뛸 수도 있습니다. 올해 저는 Python으로 솔루션을 코딩하려고 시도하고 있는데, 9일 후에는 여행 내내 꽤 많은 것을 배운 것 같습니다.
5일째, 사연은 안전 매뉴얼 인쇄를 도와달라는 부탁을 받았습니다. 입력에는 페이지 규칙과 엘프가 인쇄하려고 했던 페이지 목록이 모두 포함되어 있습니다.
47|53 97|13 97|61 97|47 75|29 61|13 75|53 29|13 97|29 53|29 61|53 97|53 61|29 47|13 75|47 97|75 47|61 75|61 47|29 75|13 53|13 75,47,61,53,29 97,61,53,29,13 75,29,13 75,97,47,61,53 61,13,29 97,13,75,29,47
입력 구문 분석부터 시작하겠습니다.
def parse( input: str, ) -> tuple[tuple[tuple[int, int], ...], tuple[tuple[int, ...], ...]]: def inner( current, incoming ) -> tuple[tuple[tuple[int, int], ...], tuple[tuple[int, ...], ...]]: rules, pages = current if "|" in incoming: return rules + ( tuple(int(item) for item in incoming.strip().split("|")), ), pages else: return rules, pages + ( tuple(int(item) for item in incoming.strip().split(",")), ) return reduce( inner, filter(lambda line: line.strip(), input.strip().splitlines()), ((), ()) )
함수는 입력을 input이라는 문자열로 받고, 이를 .splitlines()를 사용하여 여러 줄로 나누어 내부 함수로 전송하여 두 개의 튜플(페이지 규칙용 튜플과 페이지 시퀀스용 튜플)을 생성합니다. 코드는 구분 기호 |를 통해 두 가지 유형의 정의를 구별합니다. 페이지 규칙의 경우 , 페이지의 경우
퍼즐의 첫 번째 부분에서는 페이지가 순서대로 되어 있는지 확인하라는 이야기가 나왔습니다. 작업을 수행하는 함수를 구현하는 것부터 시작해 보겠습니다.
def check_pair(rules: tuple[tuple[int, int], ...], alpha: int, beta: int) -> bool: return (beta, alpha) not in rules
그리고 페이지의 모든 조합(조합((1,2,3), 2)은 1,2, 1,3 및 2,3을 반환)을 보내는 또 다른 함수는 다음과 같습니다.
from itertools import combinations def check_pages(rules: tuple[tuple[int, int], ...], pages: tuple[int, ...]) -> bool: return all( check_pair(rules, alpha, beta) for alpha, beta in combinations(pages, 2) )
이 두 가지 기능을 개별 기능으로 분리한 주된 이유는 각 부분을 최대한 작게 유지하고 싶기 때문입니다. 내 경험에 따르면, 항목을 충분히 작게 유지하면 테스트가 가능해질 뿐만 아니라 일반적으로 최종 입력(일반적으로 비합리적으로 큼)을 디버깅하는 데에도 도움이 됩니다.
많은 경우 파트 2는 놀랍게도 파트 1의 코드 디자인 수정이 필요한 경우가 많습니다. 구현한 항목에 대한 작은 변형일 수도 있고 다른 기능이 필요할 수도 있습니다. 다른 목표에 대한 호출 순서 등 저는 직장에서 (댓글 대신) 짧은 함수를 작성하는 습관을 유지하고 있습니다.
이런 작은 기능은 이름이 좋을 때만 작동하므로 이름 지정에 주의를 기울여야 합니다. 여기에는 연습이 필요하지만 일단 익숙해지면 이 접근 방식을 통해 코드를 자체적으로 문서화할 수 있습니다. 더 큰 규모의 기능은 마치 이야기처럼 읽을 수 있으며, 독자는 필요할 때 어떤 기능을 더 자세히 살펴보고 싶은지 선택할 수 있습니다.
Martin Fowler가 작성한 Function Length라는 기사에서 인용
퍼즐로 돌아갑니다.
마지막에 퍼즐은 페이지가 제대로 정렬된 모든 경우에 대해 중간 페이지 번호의 합을 요구했습니다.
47|53 97|13 97|61 97|47 75|29 61|13 75|53 29|13 97|29 53|29 61|53 97|53 61|29 47|13 75|47 97|75 47|61 75|61 47|29 75|13 53|13 75,47,61,53,29 97,61,53,29,13 75,29,13 75,97,47,61,53 61,13,29 97,13,75,29,47
매우 간단합니다. 모든 작업을 올바르게 수행했다면 목록 이해만 하면 됩니다(Python 개발자는 맵/필터보다 이것을 선호하기 때문입니다).
다음은 정렬 알고리즘입니다.
1부에 이어 두 번째 부분에서는 중간 페이지의 합계를 원했지만 페이지 순서가 제대로 지정되지 않은 경우가 있었습니다. 중간 페이지 번호를 검색하기 전에 순서를 수정하라는 지시도 나와 있습니다.
동료들이 완전한 정렬 알고리즘 없이 문제를 해결하는 동안 저는 페이지 규칙을 설명하는 섹션에서 앞서 설명한 퍼즐과 똑같은 방식으로 문제를 해결하기로 결정했습니다. 이미 비교 부분(check_pair)이 완료되었으므로 이제 요소를 이동하는 함수가 필요합니다.
def parse( input: str, ) -> tuple[tuple[tuple[int, int], ...], tuple[tuple[int, ...], ...]]: def inner( current, incoming ) -> tuple[tuple[tuple[int, int], ...], tuple[tuple[int, ...], ...]]: rules, pages = current if "|" in incoming: return rules + ( tuple(int(item) for item in incoming.strip().split("|")), ), pages else: return rules, pages + ( tuple(int(item) for item in incoming.strip().split(",")), ) return reduce( inner, filter(lambda line: line.strip(), input.strip().splitlines()), ((), ()) )
1,2,3,4,5가 있고 함수가 수신 번호를 현재 번호 바로 앞으로 이동한다고 가정합니다. 전류 = 2, 수신 = 4라고 가정하면 그 대가로 1,4,2,3,5를 얻게 됩니다(증가하는 숫자 값에 따라 정렬한다고 가정).
친구에게 알고리즘을 설명하려 했으나 실패했습니다
다음은 제가 직접 작성한 초안에 표시된 알고리즘을 실제 코드로 바꾸는 것입니다.
def check_pair(rules: tuple[tuple[int, int], ...], alpha: int, beta: int) -> bool: return (beta, alpha) not in rules
네, 안타깝게도 재귀 상태입니다. 읽기에 더 친숙할 수 있는 첫 번째 버전을 게시해야 합니다.
from itertools import combinations def check_pages(rules: tuple[tuple[int, int], ...], pages: tuple[int, ...]) -> bool: return all( check_pair(rules, alpha, beta) for alpha, beta in combinations(pages, 2) )
두 가지 모두 본질적으로 동일하며 최종 기능 버전은 약간 최적화되었습니다. 초안 스크린샷을 참고하면 포인터가 2개 있는데 노란색 밑줄은 코드에서 포인터라는 이름으로, 파란색 밑줄은 입력됩니다.
알고리즘은 다음과 같이 작동합니다.
- 첫 번째 요소에 포인터를 설정하는 것부터 시작됩니다.
- 처음에 들어오는 요소는 항상 옆에 있는 요소입니다.
- 들어오는 포인터는 한 번에 한 요소씩 진행하며 규칙을 위반하는 경우 값을 현재 바로 앞의 값으로 이동합니다.
- 이런 일이 발생하면 수신 포인터가 재설정되고 현재의 다음으로 다시 이동합니다.
- 현재 포인터는 위치를 변경하지 않지만 이제 이전 단계에서 삽입된 새 요소를 가리키고 있습니다.
들어오는 포인터가 변경 사항을 적용하지 않고 목록의 나머지 부분을 단계별로 진행하는 경우 현재 포인터를 전진시키고(그리고 들어오는 포인터는 옆 위치로 다시 초기화됨) 프로세스를 다시 반복합니다.
마지막 2개 요소를 비교하는 알고리즘이 완료된 후 프로세스가 종료되고, 정렬된 페이지가 결과로 반환됩니다. 그런 다음 2부에서 사용할 수 있는 모든 것을 조립할 수 있습니다.
47|53 97|13 97|61 97|47 75|29 61|13 75|53 29|13 97|29 53|29 61|53 97|53 61|29 47|13 75|47 97|75 47|61 75|61 47|29 75|13 53|13 75,47,61,53,29 97,61,53,29,13 75,29,13 75,97,47,61,53 61,13,29 97,13,75,29,47
두 부분의 코드는 비슷합니다. part1을 약간 수정하고 필터 절을 약간 변형한 것뿐입니다. get_middle은 대신 정렬된 목록을 받습니다. 본질적으로 if는 약간 다른 조합으로 함수 형태의 빌딩 블록에서 답을 조립하는 것과 같습니다.
시간 복잡도가 O(n^2)에 가깝기 때문에 이는 여전히 효율적인 알고리즘은 아닙니다. 윈드서핑의 캐스케이드 AI 동반자에 따르면 알고리즘은 어떤 면에서는 삽입 정렬과 유사합니다(예, 알고리즘에 대한 설명을 제공하는 AI 도구가 유용한 경우입니다).
오늘은 여기까지입니다. 알고리즘이 잘 작동해서 다행입니다. 비록 현재 내 생활은 엉망이지만(자금 문제로 인해 프로젝트에서 제외되었습니다). 시간이 지날수록 상황이 좋아지길 바라면서 다음주에 다시 글을 쓰겠습니다.
위 내용은 코드 4의 출현을 위한 정렬 알고리즘을 코딩하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

Tomergelistsinpython, youcanusethe operator, extendmethod, listcomprehension, oritertools.chain, 각각은 각각의 지위를 불러 일으킨다

Python 3에서는 다양한 방법을 통해 두 개의 목록을 연결할 수 있습니다. 1) 작은 목록에 적합하지만 큰 목록에는 비효율적입니다. 2) 메모리 효율이 높지만 원래 목록을 수정하는 큰 목록에 적합한 확장 방법을 사용합니다. 3) 원래 목록을 수정하지 않고 여러 목록을 병합하는 데 적합한 * 운영자 사용; 4) 메모리 효율이 높은 대형 데이터 세트에 적합한 itertools.chain을 사용하십시오.

join () 메소드를 사용하는 것은 Python의 목록에서 문자열을 연결하는 가장 효율적인 방법입니다. 1) join () 메소드를 사용하여 효율적이고 읽기 쉽습니다. 2)주기는 큰 목록에 비효율적으로 운영자를 사용합니다. 3) List Comprehension과 Join ()의 조합은 변환이 필요한 시나리오에 적합합니다. 4) READE () 방법은 다른 유형의 감소에 적합하지만 문자열 연결에 비효율적입니다. 완전한 문장은 끝납니다.

pythonexecutionissprocessoftransformingpythoncodeintoExecutableInstructions.1) the -interreadsTheCode, ConvertingItintoByTecode, thethepythonVirtualMachine (pvm)을 실행합니다

Python의 주요 특징은 다음과 같습니다. 1. 구문은 간결하고 이해하기 쉽고 초보자에게 적합합니다. 2. 개발 속도 향상, 동적 유형 시스템; 3. 여러 작업을 지원하는 풍부한 표준 라이브러리; 4. 광범위한 지원을 제공하는 강력한 지역 사회와 생태계; 5. 스크립팅 및 빠른 프로토 타이핑에 적합한 해석; 6. 다양한 프로그래밍 스타일에 적합한 다중-파라 디그 지원.

Python은 해석 된 언어이지만 편집 프로세스도 포함됩니다. 1) 파이썬 코드는 먼저 바이트 코드로 컴파일됩니다. 2) 바이트 코드는 Python Virtual Machine에 의해 해석되고 실행됩니다. 3)이 하이브리드 메커니즘은 파이썬이 유연하고 효율적이지만 완전히 편집 된 언어만큼 빠르지는 않습니다.

USEAFORLOOPHENTERATINGOVERASERASERASPECIFICNUMBEROFTIMES; USEAWHILLOOPWHENTINUTIMONDITINISMET.FORLOOPSAREIDEALFORKNOWNSEDINGENCENCENS, WHILEWHILELOOPSSUITSITUATIONS WITHERMINGEDERITERATIONS.

Pythonloopscanleadtoerrors likeinfiniteloops, modifyinglistsdizeration, off-by-by-byerrors, zero-indexingissues, andnestedloopineficiencies.toavoidthese : 1) aing'i


핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

맨티스BT
Mantis는 제품 결함 추적을 돕기 위해 설계된 배포하기 쉬운 웹 기반 결함 추적 도구입니다. PHP, MySQL 및 웹 서버가 필요합니다. 데모 및 호스팅 서비스를 확인해 보세요.

에디트플러스 중국어 크랙 버전
작은 크기, 구문 강조, 코드 프롬프트 기능을 지원하지 않음

Dreamweaver Mac版
시각적 웹 개발 도구

Atom Editor Mac 버전 다운로드
가장 인기 있는 오픈 소스 편집기

WebStorm Mac 버전
유용한 JavaScript 개발 도구