ํ’€์Šคํƒ ์›น๐ŸŒ ๊ฐœ๋ฐœ์ž ์ง€๋ง์ƒ ๐Ÿง‘๐Ÿฝโ€๐Ÿ’ป
โž• ์ธ๊ณต์ง€๋Šฅ ๊ด€์‹ฌ ๐Ÿค–


Categories


Recent views

  • 1
  • 2
  • 3
  • 4
  • 5

Python for algorithms-Libraries

Python for Algorithms-Libraries

๊ฐ„ํŽธํ•˜๊ฒŒ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋Š” ํŒŒ์ด์ฌ ์ฝ”๋“œ๋“ค

Template.py

  • ์‹œ๊ฐ„ ๋น„๊ต, ํ…Œ์ŠคํŠธ ์ผ€์ด์Šค ์ž…๋ ฅ๊ณผ ์ถœ๋ ฅ, ์ •๋‹ต ๋น„๊ต ๋“ฑ์„ ํ•œ ํŒŒ์ผ์— ๋‹ด์€ template.
โœ Template.py
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
{: #coding-utf-8}
# ----------------------------------------------------------------------------
{: #}
# Created By  : Junseok Yun(markkorea@naver.com)
{: #created-by- -junseok-yun-markkorea-naver-com}
# Created Date: 2022-08-10
{: #created-date-2022-08-10}
# ---------------------------------------------------------------------------
{: #}
""" Python template file for algorithm competition. """
# ---------------------------------------------------------------------------
{: #}
import sys
import time

# input = sys.stdin.readline 
{: #input-sys-stdin-readline}# Change `input` func to `readline` func.
# print(sys.getrecursionlimit()) 
{: #print-sys-getrecursionlimit}# Print maximum recursion limit.
sys.setrecursionlimit(987654321)   # Change maximum recursion limit.

arrayOfInput = [
    # ์ž…๋ ฅ๊ฐ’ ๋ฆฌ์ŠคํŠธ
]

arrayOfAnswer = [
    # ์ •๋‹ต ๋ฆฌ์ŠคํŠธ
]

if len(arrayOfInput) != len(arrayOfAnswer):
    raise Exception(
        f"Error: (len(arrayOfInput)={len(arrayOfInput)}) != (len(arrayOfAnswer)={len(arrayOfAnswer)})")  

def solution(
    # your arguments
):
    result = []
    return result  

print("---------------------------------------------------------------------------------------------------------------------------------------")
print("|case|               Input               |                correct               |              submitted              | result|time(ms)|")
print("=======================================================================================================================================")  

for case, (example_input, correct_answer) in enumerate(zip(arrayOfInput, arrayOfAnswer)):
    startTime = time.time() * 1000
    answer = solution(*example_input)
    responseTime = time.time() * 1000 - startTime
    print(f"|{case:^4}|{str(example_input):^35.35}|{str(correct_answer):^37.37}|{str(answer):^37.37}|{'OOO' if answer == correct_answer else 'XXX':*^7}|{responseTime:^8.3f}|")
    print("---------------------------------------------------------------------------------------------------------------------------------------")

functools

๊ณ ์ฐจ ํ•จ์ˆ˜์™€ ์ฝœ๋Ÿฌ๋ธ” ๊ฐ์ฒด์— ๋Œ€ํ•œ ์—ฐ์‚ฐ, ์ฆ‰ ํ•จ์ˆ˜๋ฅผ ์ž…๋ ฅ ๊ฐ’ ํ˜น์€ ์ถœ๋ ฅ ๊ฐ’์œผ๋กœ ์ฃผ๋กœ ๋‹ค๋ฃจ๋Š” ํ•จ์ˆ˜ ๋“ค์ด๋‹ค.

์–ด๋ ต๊ฒŒ ๊ตฌํ˜„ํ•ด์•ผ ํ•  ์ฝ”๋“œ๋“ค์ด ์‰ฝ๊ฒŒ ์“ธ ์ˆ˜ ์žˆ๊ฒŒ ๋˜์–ด ์žˆ๋‹ค.

๋งŒ์•ฝ ํ‘œ์ค€ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋ฅผ ์‚ฌ์šฉํ•  ์ˆ˜ ์—†๋Š” ์‹œํ—˜์— ๋Œ€๋น„ํ•˜์—ฌ ํ•ด๋‹น ๊ตฌํ˜„ ์ฝ”๋“œ๋“ค์„ ์˜ฌ๋ ค๋†“์•˜๋‹ค.

cmp_to_key()

sorted, sort ํ•จ์ˆ˜๋ฅผ ์ด์šฉํ•ด ์ •๋ ฌ ์‹œ ๋น„๊ต ๊ธฐ์ค€์„ ์ค„ ์ˆ˜ ์žˆ์Œ

๐Ÿงพ๏ธ example

title:cmp_to_key()์˜ ํ™œ์šฉ ์˜ˆ์‹œ
collapse: true

from functools import cmp_to_key

points = [{'x': 1, 'y': 2}, {'x': 2, 'y': 2},
          {'x': 2, 'y': 1}, {'x': 1, 'y': 1}]
          
# ๋‹จ์ˆœ ๋น„๊ต ์„ค์ •์€ lambda๋ฅผ ์ด์šฉํ•ด ๊ฐ€๋Šฅ
{: #๋‹จ์ˆœ-๋น„๊ต-์„ค์ •์€-lambda๋ฅผ-์ด์šฉํ•ด-๊ฐ€๋Šฅ}
points.sort(key=lambda x: x['x']) 

def cmp(a, b):
	# ํ•„์š” ๋ฆฌํ„ด ๊ฐ’
	# 1 : ์ขŒ์ธก์ด ๋” ํผ
	# 0 : ๋‘ ๊ฐ’์ด ๊ฐ™์Œ
	# -1 : ์šฐ์ธก์ด ๋” ํผ
	if a['y'] == b['y']:
        if a['x'] > b['x']:
			return 1
		elif a['x'] == b['x']:
			return 0
		elif a['x'] < b['x']:
			return -1
	return 1 if a['y'] > b['y'] else -1

points.sort(key=cmp_to_key(cmp))
sorted(points, key=cmp_to_key(cmp), reverse=True)

print(points)
# [{'x': 1, 'y': 1}, {'x': 2, 'y': 1}, {'x': 1, 'y': 2}, {'x': 2, 'y': 2}] 'y': 2}]
{: #x-1-y-1-x-2-y-1-x-1-y-2-x-2-y-2-y-2}
โž• cmp_to_key()์˜ ๊ตฌํ˜„
def cmp_to_key(mycmp):
    'Convert a cmp= function into a key= function'
    class K(object):
        def __init__(self, obj, *args):
            self.obj = obj
        def __lt__(self, other):
            return mycmp(self.obj, other.obj) < 0
        def __gt__(self, other):
            return mycmp(self.obj, other.obj) > 0
        def __eq__(self, other):
            return mycmp(self.obj, other.obj) == 0
        def __le__(self, other):
            return mycmp(self.obj, other.obj) <= 0
        def __ge__(self, other):
            return mycmp(self.obj, other.obj) >= 0
        def __ne__(self, other):
            return mycmp(self.obj, other.obj) != 0
    return K
  

points = [{'x': 1, 'y': 2}, {'x': 2, 'y': 2},
          {'x': 2, 'y': 1}, {'x': 1, 'y': 1}]

def cmp(a, b):
    if a['y'] == b['y']:
        if a['x'] > b['x']:
            return 1
        elif a['x'] == b['x']:
            return 0
        elif a['x'] < b['x']:
            return -1
    return 1 if a['y'] > b['y'] else -1  

points.sort(key=cmp_to_key(cmp))
print(points)
โž• ์‚ฌ์šฉํ•ด๋ณธ ์•Œ๊ณ ๋ฆฌ์ฆ˜ ๋ฌธ์ œ

๋ฐฑ์ค€ 11650๋ฒˆ ๋ฌธ์ œ

  • ๋‹ค๋ฅธ ์‚ฌ๋žŒ์€ x์™€ y๊ฐ’์˜ ํ•ฉ์„ ํ†ตํ•ด ์ •๋ ฌํ•˜๋˜, y์— ๊ฐ€์ค‘์น˜๋ฅผ ์ ๊ฒŒ ์ฃผ์–ด (x์˜ ์ตœ๋Œ€๊ฐ’์œผ๋กœ ๋‚˜๋ˆˆ ์ˆ˜์ค€์˜ ๊ฐ€์ค‘์น˜) ๊ตฌ๋ณ„ํ•˜๋Š” ๋ฐฉ๋ฒ•์œผ๋กœ ๋น ๋ฅด๊ฒŒ ์ •๋ ฌํ–ˆ๋‹ค.

reduce()

์ž๋ฃŒ ๊ตฌ์กฐ ๋‚ด์˜ ๊ฐ ์š”์†Œ๋“ค์„ ์ˆœ์„œ๋Œ€๋กœ ํ•จ์ˆ˜๋ฅผ ์ ์šฉํ•ด ๋ˆ„์ ๋œ ๊ฐ’์„ ๋ณด์—ฌ์ค€๋‹ค.

๐Ÿ’ก accumulate ํ•จ์ˆ˜์™€ ๋‹ฌ๋ฆฌ ๊ฒฐ๊ณผ๊ฐ’์ด iterator๊ฐ€ ์•„๋‹Œ ๊ฐ’์ด๋‹ค.
from itertools import accumulate
from operator import add

print(list(accumulate([1, 2, 3, 4, 5], add))) # [1, 3, 6, 10, 15]
# iterator๋กœ ๋‚˜์˜ค๋ฏ€๋กœ list ํ•จ์ˆ˜๋ฅผ ํ†ตํ•ด list๋กœ ๋ฐ”๊ฟˆ
{: #iterator๋กœ-๋‚˜์˜ค๋ฏ€๋กœ-list-ํ•จ์ˆ˜๋ฅผ-ํ†ตํ•ด-list๋กœ-๋ฐ”๊ฟˆ}
# add ํ•จ์ˆ˜๋ฅผ ๋งŒ๋“œ๋Š” ๋Œ€์‹  operator๋กœ ์ด์šฉ ๊ฐ€๋Šฅ
{: #add-ํ•จ์ˆ˜๋ฅผ-๋งŒ๋“œ๋Š”-๋Œ€์‹ -operator๋กœ-์ด์šฉ-๊ฐ€๋Šฅ}
๐Ÿงพ๏ธ reduce()์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from functools import reduce

print(reduce(lambda accumulated, element: accumulated + element, [1, 2, 3, 4, 5])) #15 
โž• reduce()์™€ accumulate()์˜ ๊ตฌํ˜„
def reduce(function, iterable, initializer=None):
    it = iter(iterable)
    if initializer is None:
        value = next(it)
    else:
        value = initializer
    for element in it:
        value = function(value, element)
    return value

def accumulate(iterable, func=operator.add, *, initial=None):
    it = iter(iterable)
    total = initial
    if initial is None:
        try:
            total = next(it)
        except StopIteration:
            return
    yield total
    for element in it:
        total = func(total, element)
        yield total

@cache

ํ•จ์ˆ˜ ์œ„์— ๋ฐ์ฝ”๋ ˆ์ดํ„ฐ๋กœ ์ด์šฉํ•˜์—ฌ ์†์‰ฝ๊ฒŒ Memoization ์„ ๊ตฌํ˜„ํ•  ์ˆ˜ ์žˆ๋‹ค.

โš ๏ธ ์ตœ์‹  ๋ฒ„์ „(3.9++) ํŒŒ์ด์ฌ์˜ ํ•จ์ˆ˜์ด๋ฏ€๋กœ ์ง€์›ํ•˜์ง€ ์•Š๋Š” ์•Œ๊ณ ๋ฆฌ์ฆ˜ ์‹œํ—˜๋„ ๋งŽ๋‹ค.
๐Ÿงพ๏ธ @cache ๋ฐ์ฝ”๋ ˆ์ดํ„ฐ์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from functools import cache
recursiveCall = 0

@cache
def factorial(n):
    global recursiveCall
    recursiveCall = recursiveCall + 1  # ์žฌ๊ท€ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋  ๋•Œ๋งˆ๋‹ค ์ฆ๊ฐ€
    return n * factorial(n-1) if n else 1

print("value: ", factorial(10))  # 3628800
print("recursive called : ", recursiveCall)  # 11

recursiveCall = 0
print("value: ", factorial(5))  # 120
print("recursive called : ", recursiveCall)  # 0
# ์•ž์„œ ์‹คํ–‰ํ•œ ๊ฒฐ๊ณผ๊ฐ€ ์ €์žฅ๋˜์–ด ๋ฐ”๋กœ ๊ฐ’์„ ๊ฐ€์ ธ์˜ด  
{: #์•ž์„œ-์‹คํ–‰ํ•œ-๊ฒฐ๊ณผ๊ฐ€-์ €์žฅ๋˜์–ด-๋ฐ”๋กœ-๊ฐ’์„-๊ฐ€์ ธ์˜ด}

recursiveCall = 0
print("value: ", factorial(12))  # 479001600
print("recursive called : ", recursiveCall)  # 2
# ์•ž์„œ ์‹คํ–‰ํ•œ ๊ฒฐ๊ณผ์— ์ถ”๊ฐ€๋กœ ๋‘๋ฒˆ๋งŒ ๋” ์‹คํ–‰ํ•ด ๊ฐ’์„ ๊ฐ€์ ธ์˜ด
{: #์•ž์„œ-์‹คํ–‰ํ•œ-๊ฒฐ๊ณผ์—-์ถ”๊ฐ€๋กœ-๋‘๋ฒˆ๋งŒ-๋”-์‹คํ–‰ํ•ด-๊ฐ’์„-๊ฐ€์ ธ์˜ด}

itertools

ํšจ์œจ์ ์ธ ๋ฃจํ•‘์„ ์œ„ํ•œ ์ดํ„ฐ๋ ˆ์ดํ„ฐ๋ฅผ ๋งŒ๋“œ๋Š” ํ•จ์ˆ˜

cycles()

๊ฐ™์€ ๊ฐ’์„ ๋ฌดํ•œํžˆ ๋ฐ˜๋ณตํ•˜์—ฌ ๋Œ๋ ค์ฃผ๋Š” iterator๋ฅผ ์ƒ์„ฑ

๐Ÿงพ๏ธ cycles()์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from itertools import cycle
   
   for i in cycle("ABCD"):
        print(i) # ABCD ๋ฌดํ•œ ๋ฐ˜๋ณต
   
   dispenser = cycle("ABCD")
   peoples = 10
   for i in range(peoples):
       print(dispenser.__next__())
   
   # A B C D A B C D A B ์ถœ๋ ฅ
โž• cycles()์˜ ๊ตฌํ˜„
def cycle(iterable):
    saved = []
    for element in iterable:
        yield element
        saved.append(element)
    while saved:
        for element in saved:
            yield element

for i in cycle("ABCD"):
    print(i)  # ABCD ๋ฌดํ•œ ๋ฐ˜๋ณต

compress()

True๋กœ ํŒ๋ช…๋˜๋Š” ์ธ๋ฑ์Šค์˜ ์›์†Œ๋งŒ ์ถ”์ถœํ•˜์—ฌ ์ดํ„ฐ๋ ˆ์ดํ„ฐ๋กœ ์ƒ์„ฑ.
data ๋ฆฌ์ŠคํŠธ๋‚˜ selector ๋ฆฌ์ŠคํŠธ ๋‘˜ ์ค‘์— ํ•˜๋‚˜๋งŒ ์†Œ์ง„๋˜์–ด๋„ ์ค‘๋‹จ๋จ

๐Ÿงพ๏ธ compress()์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from itertools import compress  

a = [1, 2, 3, 4, 5]
b = [True, False, False, True, True]
# filter ํ•จ์ˆ˜ ๋“ฑ์„ ์ด์šฉํ•ด ์ƒ์„ฑํ•œ selector ๋ฆฌ์ŠคํŠธ
{: #filter-ํ•จ์ˆ˜-๋“ฑ์„-์ด์šฉํ•ด-์ƒ์„ฑํ•œ-selector-๋ฆฌ์ŠคํŠธ}

print(list(compress(a, b)))  # [1, 4, 5]
# selector ๋ฆฌ์ŠคํŠธ์—์„œ true ์˜€๋˜ ๊ฐ’๋“ค๋งŒ ์ถœ๋ ฅ๋จ
{: #selector-๋ฆฌ์ŠคํŠธ์—์„œ-true-์˜€๋˜-๊ฐ’๋“ค๋งŒ-์ถœ๋ ฅ๋จ}
โž• compress()์˜ ๊ตฌํ˜„
def compress(data, selectors):
    return (d for d, s in zip(data, selectors) if s)  

a = [1, 2, 3, 4, 5]

b = [True, False, False, True, True]  

print(list(compress(a, b))); # [1, 4, 5]

์กฐํ•ฉ๊ณผ ์ˆœ์—ด์„ ์œ„ํ•œ ํ•จ์ˆ˜๋“ค

product, permutations, combinations, combinations_wtih_replacement() ํ•จ์ˆ˜๋“ค์€ ์•Œ๊ณ ๋ฆฌ์ฆ˜ ๋ฌธ์ œ๋กœ ์ž์ฃผ ์ถœ์ œ๋˜๋Š” ์กฐํ•ฉ๊ณผ ์ˆœ์—ด์„ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๋Š” ํ•จ์ˆ˜๋“ค์ด๋‹ค.

๐Ÿงพ๏ธ ์กฐํ•ฉ๊ณผ ์ˆœ์—ด ํ•จ์ˆ˜๋“ค์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from itertools import product, permutations, combinations, combinations_with_replacement

print(list(product('ABCD', repeat=2))) # ์ค‘์ฒฉ๋œ for ๋ฃจํ”„, ๋ชจ๋“  ์กฐํ•ฉ๊ณผ ์ˆœ์—ด์„ ์ƒ์„ฑํ•˜๊ฒŒ ๋จ.
# [('A', 'A'), ('A', 'B'), ('A', 'C'), ('A', 'D'), ('B', 'A'), ('B', 'B'), ('B', 'C'), ('B', 'D'), ('C', 'A'), ('C', 'B'), ('C', 'C'), ('C', 'D'), ('D', 'A'), ('D', 'B'), ('D', 'C'), ('D', 'D')]
{: #a-a-a-b-a-c-a-d-b-a-b-b-b-c-b-d-c-a-c-b-c-c-c-d-d-a-d-b-d-c-d-d}
print(list(permutations('ABCD', 2))) # ์ˆœ์—ด ์ƒ์„ฑ, 
# [('A', 'B'), ('A', 'C'), ('A', 'D'), ('B', 'A'), ('B', 'C'), ('B', 'D'), ('C', 'A'), ('C', 'B'), ('C', 'D'), ('D', 'A'), ('D', 'B'), ('D', 'C')]
{: #a-b-a-c-a-d-b-a-b-c-b-d-c-a-c-b-c-d-d-a-d-b-d-c}
print(list(combinations('ABCD', 2))) # ์กฐํ•ฉ ์ƒ์„ฑ
# [('A', 'B'), ('A', 'C'), ('A', 'D'), ('B', 'C'), ('B', 'D'), ('C', 'D')]
{: #a-b-a-c-a-d-b-c-b-d-c-d}
print(list(combinations_with_replacement('ABCD', 2))) # n๋ฒˆ ์ค‘๋ณต ๊ฐ€๋Šฅํ•œ ์กฐํ•ฉ ์ƒ์„ฑ
# [('A', 'A'), ('A', 'B'), ('A', 'C'), ('A', 'D'), ('B', 'B'), ('B', 'C'), ('B', 'D'), ('C', 'C'), ('C', 'D'), ('D', 'D')]
{: #a-a-a-b-a-c-a-d-b-b-b-c-b-d-c-c-c-d-d-d}
โž• ์กฐํ•ฉ, ์ˆœ์—ด ํ•จ์ˆ˜๋“ค์˜ ๊ตฌํ˜„
def product(*args, repeat=1):
    pools = [tuple(pool) for pool in args] * repeat
    result = [](){: .wikilink}{:target=\"_blank\"}
    for pool in pools:
        result = [x+[y] for x in result for y in pool]
    for prod in result:
        yield tuple(prod)  

def permutations(iterable, r=None):
    pool = tuple(iterable)
    n = len(pool)
    r = n if r is None else r
    for indices in product(range(n), repeat=r):
        if len(set(indices)) == r:
            yield tuple(pool[i] for i in indices)  

def permutations_standalone(iterable, r=None):
    pool = tuple(iterable)
    n = len(pool)
    r = n if r is None else r
    if r > n:
        return
    indices = list(range(n))
    cycles = list(range(n, n-r, -1))
    yield tuple(pool[i] for i in indices[:r])
    while n:
        for i in reversed(range(r)):
            cycles[i] -= 1
            if cycles[i] == 0:
                indices[i:] = indices[i+1:] + indices[i:i+1]
                cycles[i] = n - i
            else:
                j = cycles[i]
                indices[i], indices[-j] = indices[-j], indices[i]
                yield tuple(pool[i] for i in indices[:r])
                break
        else:
            return  

def combinations(iterable, r):
    pool = tuple(iterable)
    n = len(pool)
    for indices in permutations(range(n), r):
        if sorted(indices) == list(indices):
            yield tuple(pool[i] for i in indices)  

def combinations_standalone(iterable, r):
    pool = tuple(iterable)
    n = len(pool)
    if r > n:
        return
    indices = list(range(r))
    yield tuple(pool[i] for i in indices)
    while True:
        for i in reversed(range(r)):
            if indices[i] != i + n - r:
                break
        else:
            return
        indices[i] += 1
        for j in range(i+1, r):
            indices[j] = indices[j-1] + 1  

def combinations_with_replacement(iterable, r):
    pool = tuple(iterable)
    n = len(pool)
    for indices in product(range(n), repeat=r):
        if sorted(indices) == list(indices):
            yield tuple(pool[i] for i in indices)

def combinations_with_replacement_standalone(iterable, r):
    pool = tuple(iterable)
    n = len(pool)
    if not n and r:
        return
    indices = [0] * r
    yield tuple(pool[i] for i in indices)
    while True:
        for i in reversed(range(r)):
            if indices[i] != n - 1:
                break
        else:
            return
        indices[i:] = [indices[i] + 1] * (r - i)
        yield tuple(pool[i] for i in indices)
        
print(list(product('ABCD', repeat=2)))
print(list(permutations('ABCD', 2))) 
print(list(combinations('ABCD', 2))) 
print(list(combinations_with_replacement('ABCD', 2)))

bisect

์ด์ง„ ํƒ์ƒ‰์„ ์ด์šฉํ•˜์—ฌ ๋ฆฌ์ŠคํŠธ ๋‚ด์˜ ์›์†Œ์˜ ์œ„์น˜๋ฅผ ํƒ์ƒ‰ํ•˜๊ฑฐ๋‚˜ ์‚ฝ์ž…ํ•˜๋Š” ํ•จ์ˆ˜๋ฅผ ๊ฐ€์ง€๊ณ  ์žˆ๋‹ค.
์ด์ง„ ํƒ์ƒ‰์„ ์ง์ ‘ ๊ตฌํ˜„ํ•˜์ง€ ์•Š๊ณ  ๋†’์€ ์„ฑ๋Šฅ์„ ๋‚ผ ์ˆ˜ ์žˆ๋‹ค.

๐Ÿงพ๏ธ bisect์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from bisect import bisect_left, bisect_right, insort_left, insort_right  

a = [2, 1, 4, 3]
a.sort()  # ๋ฐฐ์—ด์€ ์ •๋ ฌ๋˜์žˆ์–ด์•ผ ํ•œ๋‹ค.  

print(bisect_right(a, 3))  # 3 [1, 2, 3, #3๋ฒˆ ์ธ๋ฑ์Šค ์ž๋ฆฌ#, 4]
print(bisect_left(a, 3))  # 2 [1, 2, #2๋ฒˆ ์ธ๋ฑ์Šค ์ž๋ฆฌ#, 3, 4]

a = [1, 2, 3, 4]
print(insort_left(a, 3))  # ํ•จ์ˆ˜ ์ž์ฒด๋Š” None์„ ๋ฆฌํ„ด # None
print(a)  # ๋Œ€์‹  ๋ฐฐ์—ด์— ์ถ”๊ฐ€๋˜์–ด ์žˆ์Œ # [1,2,3,3,4]

a = [1, 2, 3, 4]
print(insort_right(a, 3))  # None
print(a)  # [1,2,3,3,4]
๐Ÿงพ๏ธ bisect๋ฅผ ์ด์šฉํ•œ ํƒ์ƒ‰ ํ•จ์ˆ˜ ๊ตฌํ˜„
def index(a, x):
    'Locate the leftmost value exactly equal to x'
    i = bisect_left(a, x)
    if i != len(a) and a[i] == x:
        return i
    raise ValueError

def find_lt(a, x):
    'Find rightmost value less than x'
    i = bisect_left(a, x)
    if i:
        return a[i-1]
    raise ValueError

def find_le(a, x):
    'Find rightmost value less than or equal to x'
    i = bisect_right(a, x)
    if i:
        return a[i-1]
    raise ValueError

def find_gt(a, x):
    'Find leftmost value greater than x'
    i = bisect_right(a, x)
    if i != len(a):
        return a[i]
    raise ValueError

def find_ge(a, x):
    'Find leftmost item greater than or equal to x'
    i = bisect_left(a, x)
    if i != len(a):
        return a[i]
    raise ValueError
โž• bisect ํ•จ์ˆ˜๋“ค์˜ ๊ตฌํ˜„
def insort_right(a, x, lo=0, hi=None, *, key=None):
    if key is None:
        lo = bisect_right(a, x, lo, hi)
    else:
        lo = bisect_right(a, key(x), lo, hi, key=key)
    a.insert(lo, x)  

def bisect_right(a, x, lo=0, hi=None, *, key=None):
    if lo < 0:
        raise ValueError('lo must be non-negative')
    if hi is None:
        hi = len(a)
    if key is None:
        while lo < hi:
            mid = (lo + hi) // 2
            if x < a[mid]:
                hi = mid
            else:
                lo = mid + 1
    else:
        while lo < hi:
            mid = (lo + hi) // 2
            if x < key(a[mid]):
                hi = mid
            else:
                lo = mid + 1
    return lo  

def insort_left(a, x, lo=0, hi=None, *, key=None):
    if key is None:
        lo = bisect_left(a, x, lo, hi)
    else:
        lo = bisect_left(a, key(x), lo, hi, key=key)
    a.insert(lo, x) 

def bisect_left(a, x, lo=0, hi=None, *, key=None):
    if lo < 0:
        raise ValueError('lo must be non-negative')
    if hi is None:
        hi = len(a)
    if key is None:
        while lo < hi:
            mid = (lo + hi) // 2
            if a[mid] < x:
                lo = mid + 1
            else:
                hi = mid
    else:
        while lo < hi:
            mid = (lo + hi) // 2
            if key(a[mid]) < x:
                lo = mid + 1
            else:
                hi = mid
    return lo

Collections

ํŒŒ์ด์ฌ์˜ ์ž๋ฃŒ ๊ตฌ์กฐ์˜ ํŠน์ˆ˜ํ•œ ํ˜•ํƒœ์˜ ๋Œ€์•ˆ๋“ค

defaultdict

์ดˆ๊ธฐ ๊ฐ’์„ ์„ค์ •ํ•  ์ˆ˜ ์žˆ๋Š” dictionary

๐Ÿงพ๏ธ defaultdict์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from collections import defaultdict  

int_dict = defaultdict(int)
print(int_dict['no_value'])  # 0
int_dict = defaultdict(lambda: 1)  # ๊ธฐ๋ณธ๊ฐ’ ์ง€์ •
print(int_dict['no_value2'])  # 1
print(int_dict['str_value'])
int_dict['str_value'] = "str"
print(int_dict) # ๋ฌผ๋ก  ๋‹ค๋ฅธ ์ž๋ฃŒํ˜•์œผ๋กœ ์ง€์ • ๋˜ํ•œ ๊ฐ€๋Šฅ

OrderedDict

dictionary์˜ ํ‚ค,๊ฐ’ ํŽ˜์–ด์— ์ถ”๊ฐ€๋กœ ์ˆœ์„œ๋ฅผ ์ ์šฉํ•œ ์ž๋ฃŒ๊ตฌ์กฐ

๐Ÿงพ๏ธ OrderedDict์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from collections import OrderedDict

ordered_dict = OrderedDict()
ordered_dict["1"] = "a"
ordered_dict[2] = 2
ordered_dict["three"] = "3"

print(ordered_dict) # OrderedDict([('1', 'a'), (2, 2), ('three', '3')]
print(ordered_dict.popitem()) # ๋งจ ๋’ค์˜ ํ‚ค ๊ฐ’์„ pop # ('three', '3')
print(ordered_dict) # OrderedDict([('1', 'a'), (2, 2)])
ordered_dict.move_to_end('1') # ํ‚ค 1์˜ ๊ฐ’์„ ๋งจ ๋’ค๋กœ
print(ordered_dict) # OrderedDict([(2, 2), ('1', 'a')])
print(list(ordered_dict.items())[0]) # ์ฒซ๋ฒˆ์งธ ๊ฐ’ ๊ฐ€์ ธ์˜ค๊ธฐ # (2, 2)

namedtuple

ํŠœํ”Œ ๋‚ด๋ถ€์— ๋‚ด๋ถ€์— ํ‚ค ๊ฐ’์˜ ์„œ๋ธŒ ํด๋ž˜์Šค๋ฅผ ์ƒ์„ฑํ•˜์—ฌ ์ ‘๊ทผ ๊ฐ€๋Šฅ

๐Ÿงพ๏ธ namedtuple์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from collections import namedtuple  

Point = namedtuple('Point', ['x', 'y'])
p = Point(11, y=22)  # instantiate with positional or keyword arguments
print(p[0] + p[1])  # indexing like plain tuple (11, 22)
# 33
{: #33}
x, y = p  # unpack like a regular tuple
print(x, y)
# (11, 22)
{: #11-22}
print(p.x + p.y)  # fields also accessible by name
# 33
{: #33}
print(p)

ChainMap

๋‘ dictionary๋ฅผ ๋น ๋ฅด๊ฒŒ ํ•ฉ์น˜๋Š”๋ฐ ์‚ฌ์šฉ

๐Ÿงพ๏ธ ChainMap์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from collections import ChainMap, OrderedDict

dict1 = {"x": 1, "2": "a"}
dict2 = {"y": 3, "three": "b"}
dict3 = OrderedDict({"z": 4, "1": "one"})  

chained = ChainMap(dict1, dict2, dict3)  

print(chained) # ChainMap({'x': 1, '2': 'a'}, {'y': 3, 'three': 'b'}, OrderedDict([('z', 4), ('1', 'one')]))
print(chained['2']) # a
print(chained['y']) # 3
print(chained['1'])# one # ๊ฐ๊ธฐ ๋‹ค๋ฅธ ํ‚ค๋“ค์— ํ•œ๊บผ๋ฒˆ์— ์ ‘๊ทผ ๊ฐ€๋Šฅ

์ค‘์š”ํ•˜๊ณ  ์ž์ฃผ ์‚ฌ์šฉ๋˜๋Š” ๋‹ค๋ฅธ ์ž๋ฃŒ ๊ตฌ์กฐ๋“ค

๋‹จ์ˆœํžˆ ํŽธ๋ฆฌํ•œ ์ˆ˜์ค€์ด ์•„๋‹Œ ํ•ต์‹ฌ์ ์œผ๋กœ ์‚ฌ์šฉ๋˜๋Š” ์ž๋ฃŒ ๊ตฌ์กฐ๋“ค

deque

์Šคํƒ + ํ ๊ธฐ๋Šฅ์ด ํ•ฉ์ณ์ง„ ๋ฆฌ์ŠคํŠธ
๊ธฐ๋ณธ ๋ฆฌ์ŠคํŠธ์™€ ๋‹ฌ๋ฆฌ pop()๊ณผ insert() ์—ฐ์‚ฐ์„ ์ƒ์ˆ˜ ์‹œ๊ฐ„ ๋งŒ์— ๋๋‚ผ ์ˆ˜ ์žˆ๋‹ค.

๐Ÿงพ๏ธ deque์˜ ํ™œ์šฉ ์˜ˆ์‹œ
from collections import deque

d = deque('12345')
print(d)  # deque(['1', '2', '3', '4', '5'])  

d.append('6')
d.appendleft('0')
print(d)  # deque(['0', '1', '2', '3', '4', '5', '6'])  

print(d.pop())  # 6
print(d)  # deque(['0', '1', '2', '3', '4', '5'])

print(d.popleft())  # 0
print(d)  # deque(['1', '2', '3', '4', '5']) 

d.extend('abc')
print(d)  # deque(['1', '2', '3', '4', '5', 'a', 'b', 'c'])

d.extendleft('efg')
print(d)  # deque(['g', 'f', 'e', '1', '2', '3', '4', '5', 'a', 'b', 'c'])

d.rotate(1)
print(d)  # deque(['c', 'g', 'f', 'e', '1', '2', '3', '4', '5', 'a', 'b'])

d.rotate(-1)
print(d)  # deque(['g', 'f', 'e', '1', '2', '3', '4', '5', 'a', 'b', 'c'])  

# d[0] : ์ตœ์ขŒ์ธก peek
{: #d-0-์ตœ์ขŒ์ธก-peek}
# d[-1] : ์ตœ์šฐ์ธก peek
{: #d-1-์ตœ์šฐ์ธก-peek}

heapq

์ด์ง„ ํŠธ๋ฆฌ๋ฅผ ์ด์šฉํ•ด ์ตœ์†Œ ํž™์„ ๊ตฌํ˜„, ์ตœ๋Œ€ ํž™์„ ๊ตฌํ˜„ ํ•˜๊ธฐ ์œ„ํ•ด์„œ๋Š” ๊ฐ’์— -์„ ๋ถ™์—ฌ ๋„ฃ์–ด์ค˜์•ผ ํ•œ๋‹ค.

๐Ÿงพ๏ธ heapq์˜ ํ™œ์šฉ ์˜ˆ์‹œ
import heapq

a = [2, 4, 6]

# heap์— ์‚ฝ์ž…
{: #heap์—-์‚ฝ์ž…}
heapq.heappush(a, 3)
print(a)  # [2, 3, 6, 4]  

# ์ตœ์†Œ๊ฐ’ pop
{: #์ตœ์†Œ๊ฐ’-pop}
print(heapq.heappop(a))  # 2
print(a)  # [3, 4, 6]  

b = [3, 1, 6]
heapq.heapify(b) # b๋ฅผ ํž™์„ ์ด์šฉํ•ด ์ •๋ ฌ
print(b)  # [1, 3, 6]
๐Ÿ’ก heapq ํ‚ค๊ฐ’ ๋น„๊ต ๋ฐฉ๋ฒ• ๋ฐ”๊พธ๊ธฐ

์ถœ์ฒ˜ here

heapq๋ฅผ ์ด์šฉํ•˜์—ฌ ์šฐ์„ ์ˆœ์œ„ ํ ๋“ฑ์„ ๋งŒ๋“ค ์‹œ ์•„๋ž˜์™€ ๊ฐ™์ด __lt__ ๋ฉ”์†Œ๋“œ๋ฅผ ์˜ค๋ฒ„๋ผ์ด๋”ฉํ•˜์—ฌ ํ‚ค๊ฐ’์˜ ๋น„๊ต ๋ฐฉ๋ฒ•์„ ์ •ํ•ด์ค„ ์ˆ˜ ์žˆ๋‹ค.

import heapq  

class Node(object):
    def __init__(self, val: int):
        self.val = val 

    def __repr__(self):
        return f'Node value: {self.val}'  

    def __lt__(self, other):
        return self.val < other.val  

heap = [Node(2), Node(0), Node(1), Node(4), Node(2)]
heapq.heapify(heap)

# output: [Node value: 0, Node value: 2, Node value: 1, Node value: 4, Node value: 2]
{: #output-node-value-0-node-value-2-node-value-1-node-value-4-node-value-2}
print(heap) 

heapq.heappop(heap)
# output: [Node value: 1, Node value: 2, Node value: 2, Node value: 4]
{: #output-node-value-1-node-value-2-node-value-2-node-value-4}
print(heap)

์žฌ๊ท€ํ•จ์ˆ˜

์žฌ๊ท€ํ•จ์ˆ˜ ๋ฌธ์ œ

JAVA

์กฐํ•ฉ๊ณผ ์ˆœ์—ด

๐Ÿงพ๏ธ example
import java.util.ArrayList;
import java.util.List; 

public class CombinationPermutationGenerator {
    /**
     * Generates all combinations of size k from an array of n elements.
     * @param arr the input array
     * @param k the size of the combinations to generate
     * @return a list of int arrays representing all combinations
     */

    public static List<int[]> combinations(int[] arr, int k) {
        List<int[]> result = new ArrayList<>();
        combinationsHelper(arr, k, 0, new int[k], 0, result);
        return result;
    }

    private static void combinationsHelper(int[] arr, int k, int start, int[] combination, int index, List<int[]> result) {
        if (index == k) {
            result.add(combination.clone());
            return;
        }
        for (int i = start; i <= arr.length - k + index; i++) {
            combination[index] = arr[i];
            combinationsHelper(arr, k, i + 1, combination, index + 1, result);
        }
    }

    /**
     * Generates all permutations of an array of n elements.
     * @param arr the input array
     * @return a list of int arrays representing all permutations
     */
    public static List<int[]> permutations(int[] arr) {
        List<int[]> result = new ArrayList<>();
        permutationsHelper(arr, new boolean[arr.length], new int[arr.length], 0, result);
        return result;
    }

    private static void permutationsHelper(int[] arr, boolean[] used, int[] permutation, int index, List<int[]> result) {
        if (index == arr.length) {
            result.add(permutation.clone());
            return;
        }
        for (int i = 0; i < arr.length; i++) {
            if (!used[i]) {
                used[i] = true;
                permutation[index] = arr[i];
                permutationsHelper(arr, used, permutation, index + 1, result);
                used[i] = false;
            }
        }
    }

    /**
     * Generates all permutations with replacement of an array of n elements.
     * @param arr the input array
     * @param k the number of selections for each position
     * @return a list of int arrays representing all permutations with replacement
     */
    public static List<int[]> permutationsWithReplacement(int[] arr, int k) {
        List<int[]> result = new ArrayList<>();
        permutationsWithReplacementHelper(arr, k, new int[k], 0, result);
        return result;
    }

    private static void pesssrmutationsWithReplacementHelper(int[] arr, int k, int[] permutation, int index, List<int[]> result) {
        if (index == k) {
            result.add(permutation.clone());
            return;
        }
        for (int i = 0; i < arr.length; i++) {
            permutation[index] = arr[i];
            permutationsWithReplacementHelper(arr, k, permutation, index + 1, result);
        }
    }
}

higher order functions

๐Ÿงพ๏ธ example
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
List<Integer> squares = numbers.stream().map(x -> x * x).collect(Collectors.toList());



List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6);
List<Integer> evenNumbers = numbers.stream()
                                    .filter(n -> n % 2 == 0)
                                    .collect(Collectors.toList());
System.out.println(evenNumbers); // Output: [2, 4, 6]

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
int sum = numbers.stream().reduce(0, (a, b) -> a + b);

bisect

๐Ÿงพ๏ธ example
import java.util.Arrays;

public class Bisect {
    public static int bisectLeft(int[] arr, int x) {
        int index = Arrays.binarySearch(arr, x);
        return index >= 0 ? index : -index - 1;
    }
    
    public static int bisectRight(int[] arr, int x) {
        int index = Arrays.binarySearch(arr, x);
        return index >= 0 ? index + 1 : -index - 1;
    }
    
    public static void main(String[] args) {
        int[] arr = {1, 2, 2, 3, 3, 3, 4, 5};
        int x = 3;
        int leftIndex = bisectLeft(arr, x);
        int rightIndex = bisectRight(arr, x);
        System.out.println(leftIndex);  // Output: 3
        System.out.println(rightIndex);  // Output: 6
    }
}

์ •๋ ฌ

๐Ÿงพ๏ธ stream์„ ํ†ตํ•œ ์ •๋ ฌ
String[] arr = { "apple", "banana", "cherry", "date" };
Arrays.sort(arr, new Comparator<String>() {
    public int compare(String s1, String s2) {
        return s2.compareTo(s1);
    }
});

List<String> list = Arrays.asList("apple", "banana", "cherry", "date");
List<String> sortedList = list.stream()
    .sorted()
    .collect(Collectors.toList());

heap

๐Ÿงพ๏ธ PriorityQueue, ๊ธฐ๋ณธ ์ตœ์†Œํž™์ด๋‹ค.
PriorityQueue<int[]> hq = new PriorityQueue<>(new Comparator<int[]>() {
    public int compare(int[] s1, int[] s2) {
        if (s1[1] < s2[1]){
            return 1;
        }
        return -1;
   }
});
hq.add(new int[]{start, 0});