2016-11-02 5 views
1

저는 알고리즘 선택 (a.k.a 결정론적인 선택)을 구현하고 있습니다. 작은 배열/목록에 대한 작업을 가지고 있지만 배열 크기가 26 이상이되면 다음 오류로 중단됩니다. "RuntimeError : 최대 재귀 깊이 초과". 배열 크기가 25 이하이면 문제가 없습니다.중앙값의 중간 값 선택 파이썬

나의 궁극적 인 목표는 크기가 500 인 배열을 실행하고 많은 반복을 수행하는 것입니다. 반복은 문제가되지 않습니다. 나는 이미 StackOverflow를 연구했고 기사를 보았습니다 : Python implementation of "median of medians" algorithm 다른 많은 것들 중에서. 내 무작위로 생성 된 배열에 중복이 문제를 일으킬 수도 있지만 그럴 것 같지 않은 직감이있었습니다. 어떤 도움을 주시면 감사하겠습니다

import math 
import random 

# Insertion Sort Khan Academy video: https://www.youtube.com/watch?v=6pyeMmJTefg&list=PL36E7A2B75028A3D6&index=22 

def insertion_sort(A): # Sorting it in place 
    for index in range(1, len(A)):# range is up to but not including len(A) 
     value = A[index] 
     i = index - 1   # index of the item that is directly to the left 
     while i >= 0: 
     if value < A[i]: 
      A[i + 1] = A[i] 
      A[i] = value 
      i = i - 1 
     else: 
      break 

timeslo = 0 # I think that this is a global variable 

def partition(A, p): 
    global timeslo 
    hi = [] #hold things larger than our pivot 
    lo = [] # "  " smaller " " " 
    for x in A:  # walk through all the elements in the Array A. 
    if x <p: 
     lo = lo + [x] 
     timeslo = timeslo + 1 #keep track no. of comparisons 
    else: 
     hi = hi + [x] 
    return lo,hi,timeslo 

def get_chunks(Acopy, n): 
            # Declare some empty lists to hold our chunks 
    chunk = [] 
    chunks = [] 
            # Step through the array n element at a time 
    for x in range(0, len(Acopy), n): # stepping by size n starting at the beginning 
            # of the array 
    chunk = Acopy[x:x+n]   # Extract 5 elements       
            # sort chunk and find its median 
    insertion_sort(chunk) # in place sort of chunk of size 5 
    # get the median ... (i.e. the middle element) 
    # Add them to list 



mindex = (len(chunk)-1)/2 # pick middle index each time 

    chunks.append(chunk[mindex]) 
#  chunks.append(chunk)      # assuming subarrays are size 5 and we want the middle 
                # this caused some trouble because not all subarrays were size 5 
          # index which is 2. 
    return chunks 


def Select(A, k): 

    if (len(A) == 1): # if the array is size 1 then just return the one and only element 
    return A[0] 
    elif (len(A) <= 5): # if length is 5 or less, sort it and return the kth smallest element 
    insertion_sort(A) 
    return A[k-1] 
    else: 
    M = get_chunks(A, 5) # this will give you the array of medians,,, don't sort it....WHY ??? 



    m = len(M)   # m is the size of the array of Medians M. 

    x = Select(M, m/2)# m/2 is the same as len(A)/10 FYI 

    lo, hi, timeslo = partition(A, x) 

    rank = len(lo) + 1 

    if rank == k: # we're in the middle -- we're done 
     return x, timeslo # return the value of the kth smallest element 
    elif k < rank: 
     return Select(lo, k) # ??????????????? 
    else: 
     return Select(hi, k-rank) 

################### TROUBLESHOOTING ################################ 
# Works with arrays of size 25 and 5000 iterations 
# Doesn't work with  " 26 and 5000 " 
# 
# arrays of size 26 and 20 iterations breaks it ????????????????? 

# A = [] 
Total = 0 
n = input('What size of array of random #s do you want?: ') 
N = input('number of iterations: ') 

# n = 26 
# N = 1 

for x in range(0, N): 
    A = random.sample(range(1,1000), n) # make an array or list of size n 
    result = Select(A, 2)  #p is the median of the medians, 2 means the 3rd smallest element 
    Total = Total + timeslo    # the total number of comparisons made 
print("the result is"), result 
print("timeslo = "), timeslo 
print("# of comparisons = "), Total 

# A = [7, 1, 3, 5, 9, 2, 83, 8, 4, 13, 17, 21, 16, 11, 77, 33, 55, 44, 66, 88, 111, 222] 
# result = Select(A, 2) 
# print("Result = "), result 

:

여기 내 코드입니다.

답변

1

변경 당신은 결국 그것을 인쇄하여 timeslo를 얻을 수 있습니다
return x # return the value of the kth smallest element

에이 라인
return x, timeslo # return the value of the kth smallest element
. 이 매개 변수 p 난 그냥 빨리이 시도 x = Select(M, m/2)

+0

이전 문에서 평균 수치로 배열을 분할하는 partition(A, p)에 사용되며 그 일 때문에 timeslox 반환하는 것은 올바르지 않습니다! 이것은 환상적입니다, 나는 방금 내 생명을 구하기위한 버그를 찾을 수 없었습니다. 브라보, 고마워! – Effectsmeister