||[[TableOfContents]]|| === Approach === 맨 처ìŒì— ë¬¸ì œë¥¼ ì½ê³ 대략 ì—°ìŠµìž¥ì— ë¬¸ì œì—ì˜ ê° ë³€ìˆ˜ë“¤ì´ ë 만한 ë¶€ë¶„ë“¤ì„ ë³´ì•˜ë‹¤. ì¼ë‹¨ ì†Œìˆ˜ë“¤ì˜ í•©ì´ë¼ í•˜ê³ , 4ìžë¦¬ì˜ í•©ì´ë¼ê³ 한다. 대략 pseudo code 를 다ìŒì™€ ê°™ì´ ìž‘ì„±í•´ë³´ì•˜ë‹¤. {{{~cpp for eachGenNumberFour in GenNumbersFour isAllPrimeNumber: print eachGenNumberFour return print "impossible" GenNumbersFour = í•©ì´ x ì¸ ìˆ˜ 조합리스트 }}} í•©ì´ x ì¸ ìˆ˜ ì¡°í•©ë¦¬ìŠ¤íŠ¸ì— ëŒ€í•´ 어떻게 êµ¬í• ê¹Œ ê¶ë¦¬í•˜ë˜ì¤‘, 소수리스트를 ë¨¼ì € ë§Œë“¤ê³ ì†Œìˆ˜ë¦¬ìŠ¤íŠ¸ì—ì„œ 4개를 골ë¼ì„œ í•©ì„ êµ¬í•œ 결과가 n ì¸ì§€ë¥¼ 비êµí•˜ëŠ” ë°©ë²•ì´ ë” ë¹¨ë¦¬ êµ¬í˜„í• ìˆ˜ ìžˆê² êµ¬ë‚˜ë¼ëŠ” ìƒê°ì´ 들었다. ì´ì— 대해서 TDDë¡œ 진행. {{{~cpp class PrimeNumberList: def __init__(self,aNum): self.size = aNum self.resultTable = [] self.createList() def createList(self): for i in range(2,self.size+1): if self.isPrimeNumber(i): self.resultTable.append(i) def isPrimeNumber(self,aNum): for eachValue in self.resultTable: if aNum % eachValue == 0: return False return True def getList(self): return self.resultTable class PrimeNumberTest(unittest.TestCase): def testPrimeNumber(self): self.assertEquals([2,], PrimeNumberList(2).getList()) self.assertEquals([2,3], PrimeNumberList(3).getList()) self.assertEquals([2,3,5], PrimeNumberList(5).getList()) self.assertEquals([2,3,5,7,11], PrimeNumberList(12).getList()) self.assertEquals([2,3,5,7,11,13,17,19], PrimeNumberList(20).getList()) }}} ê·¸ë¦¬ê³ ì†Œìˆ˜ë¦¬ìŠ¤íŠ¸ë¡œë¶€í„° 4개를 구하는 ë°©ë²•ì— ëŒ€í•´ ìƒê°í•˜ë‹¤. 맨 처ìŒì— 대해서는 ì¤‘ë³µì„ í—ˆìš©í•˜ë©´ 안ë˜ëŠ” 줄 ì•Œê³ êµ¬í˜„í•˜ì˜€ë‹¤. 그러다가 ë¬¸ì œì—ì„œ ì¤‘ë³µì„ í—ˆìš©í•œë‹¤ëŠ” ì‚¬ì‹¤ì„ ì•Œê³ ë‹¤ì‹œ 구현. {{{~cpp def selectionFour(aList): sizeOfList = len(aList) for i in range(sizeOfList): for j in range(sizeOfList): for k in range(sizeOfList): for l in range(sizeOfList): yield [aList[i],aList[j],aList[k],aList[l]] }}} === full source === {{{~cpp from __future__ import generators import psyco class PrimeNumberList: def __init__(self,aNum): self.size = aNum self.resultTable = [] self.createList() def createList(self): for i in range(2,self.size+1): if self.isPrimeNumber(i): self.resultTable.append(i) def isPrimeNumber(self,aNum): for eachValue in self.resultTable: if aNum % eachValue == 0: return False return True def getList(self): return self.resultTable def selectionFour(aList): sizeOfList = len(aList) for i in range(sizeOfList): for j in range(sizeOfList): for k in range(sizeOfList): for l in range(sizeOfList): yield [aList[i],aList[j],aList[k],aList[l]] def main(): n = int(raw_input('input number : ')) primeNumberList = PrimeNumberList(n) for eachPrimeNumberSeq in selectionFour(primeNumberList.getList()): if sum(eachPrimeNumberSeq) == n: print eachPrimeNumberSeq return print "Impossible" if __name__=="__main__": #unittest.main(argv=('','-v')) psyco.full() main() }}} === ë¬¸ì œì - 4ì´ˆ ì´ë‚´ì— ë‹µì´ ë‚˜ì˜¤ì§€ 않는다. === ìŠ¤íŽ™ìƒ 10000000 ê°’ ë‚´ì—ì„œ 4ì´ˆ ì´ë‚´ì— ë‹µì´ ë‚˜ì™€ì•¼ 한다. ì´ì— 대해서 í˜„ìž¬ì˜ ë³‘ëª©ì§€ì ì— ëŒ€í•´ profiling ì„ í•´ 보았다. {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 10000 [2, 2, 23, 9973] 21065 function calls in 6.387 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 1 0.075 0.075 1.201 1.201 summationoffourprimes.py:13(createList) 0 0.000 0.000 profile:0(profiler) 1 5.103 5.103 6.387 6.387 summationoffourprimes.py:88(main) 1 0.000 0.000 1.201 1.201 summationoffourprimes.py:8(__init__) 9999 1.126 0.000 1.126 0.000 summationoffourprimes.py:18(isPrimeNumber) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:24(getList) 11061 0.083 0.000 0.083 0.000 summationoffourprimes.py:79(selectionFour) 1 0.000 0.000 6.387 6.387 <string>:1(?) }}} {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 50000 [2, 2, 3, 49993] 60269 function calls in 24.926 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 1 0.428 0.428 19.348 19.348 summationoffourprimes.py:13(createList) 0 0.000 0.000 profile:0(profiler) 1 5.492 5.492 24.923 24.923 summationoffourprimes.py:88(main) 1 0.000 0.000 19.348 19.348 summationoffourprimes.py:8(__init__) 49999 18.920 0.000 18.920 0.000 summationoffourprimes.py:18(isPrimeNumber) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:24(getList) 10265 0.083 0.000 0.083 0.000 summationoffourprimes.py:79(selectionFour) 1 0.003 0.003 24.926 24.926 <string>:1(?) Hit any key to close this window... }}} ëŒ€ë¶€ë¶„ì˜ ì‹œê°„ì€ ì†Œìˆ˜í…Œì´ë¸”ì„ ìž‘ì„±í•˜ëŠ” ë¶€ë¶„ì´ ëœë‹¤. 그래서 ì´ ë¶€ë¶„ì— ëŒ€í•´ì„œ Seminar:PrimeNumber 를 ì°¸ê³ , 최ì í™”ëœ ì•Œê³ ë¦¬ì¦˜ìœ¼ë¡œ ìˆ˜ì •í•˜ì˜€ë‹¤. ê·¸ë¦¬ê³ ì—ì‹œ psyco 를 ì´ìš©í•˜ì˜€ë‹¤. ê·¸ ê²°ê³¼, 10000000 기준 10ì´ˆ. 기존 ì•Œê³ ë¦¬ì¦˜ì˜ ê²½ìš° 50000번 기준 24ì´ˆ ì´ìƒ. {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 10000 [2, 2, 23, 9973] 11067 function calls in 5.417 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 0 0.000 0.000 profile:0(profiler) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:48(getList) 1 0.000 0.000 5.417 5.417 <string>:1(?) 1 0.000 0.000 0.013 0.013 summationoffourprimes.py:36(createList) 1 0.000 0.000 0.013 0.013 summationoffourprimes.py:31(__init__) 11061 0.094 0.000 0.094 0.000 summationoffourprimes.py:103(selectionFour) 1 5.309 5.309 5.416 5.416 summationoffourprimes.py:112(main) 1 0.013 0.013 0.013 0.013 summationoffourprimes.py:8(prime2) Hit any key to close this window... }}} {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 50000 [2, 2, 3, 49993] 10271 function calls in 7.878 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 0 0.000 0.000 profile:0(profiler) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:48(getList) 1 0.002 0.002 7.878 7.878 <string>:1(?) 1 0.002 0.002 0.090 0.090 summationoffourprimes.py:36(createList) 1 0.000 0.000 0.090 0.090 summationoffourprimes.py:31(__init__) 10265 0.098 0.000 0.098 0.000 summationoffourprimes.py:103(selectionFour) 1 7.688 7.688 7.876 7.876 summationoffourprimes.py:112(main) 1 0.088 0.088 0.088 0.088 summationoffourprimes.py:8(prime2) Hit any key to close this window... }}} 최ì í™” ì´í›„ 50000번 기준으로는 ë³‘ëª©ì— ëŒ€í•œ ë³€ë³„ë ¥ì´ ì—†ë‹¤. {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 1000000 [2, 2, 13, 999983] 470994 function calls in 26.768 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 0 0.000 0.000 profile:0(profiler) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:48(getList) 1 0.040 0.040 26.768 26.768 <string>:1(?) 1 0.062 0.062 2.708 2.708 summationoffourprimes.py:36(createList) 1 0.000 0.000 2.708 2.708 summationoffourprimes.py:31(__init__) 470988 5.244 0.000 5.244 0.000 summationoffourprimes.py:103(selectionFour) 1 18.777 18.777 26.728 26.728 summationoffourprimes.py:112(main) 1 2.646 2.646 2.646 2.646 summationoffourprimes.py:8(prime2) Hit any key to close this window... }}} 여기서 selectionFour ì˜ ê²½ìš°ëŠ” percall ì— ê±¸ë¦¬ëŠ” ì‹œê°„ì€ ì 다. 하지만, call 횟수 ìžì²´ê°€ 470988 번으로 많다. ì´ ë¶€ë¶„ì— ëŒ€í•´ì„œ ì¼ì¢…ì˜ inlining ì„ í•˜ì˜€ë‹¤. {{{~cpp C:\WINDOWS\system32\cmd.exe /c python SummationOfFourPrimes.py input number : 1000000 (2, 2, 13, 999983) 6 function calls in 16.671 CPU seconds Random listing order was used ncalls tottime percall cumtime percall filename:lineno(function) 0 0.000 0.000 profile:0(profiler) 1 0.000 0.000 0.000 0.000 summationoffourprimes.py:48(getList) 1 13.736 13.736 16.540 16.540 summationoffourprimes.py:121(main2) 1 0.067 0.067 2.804 2.804 summationoffourprimes.py:36(createList) 1 2.738 2.738 2.738 2.738 summationoffourprimes.py:8(prime2) 1 0.131 0.131 16.671 16.671 <string>:1(?) 1 0.000 0.000 2.804 2.804 summationoffourprimes.py:31(__init__) Hit any key to close this window... }}} 10000000 ê±´ì— ëŒ€í•´ì„œëŠ” 7.49 ì´ˆ ê¸°ë¡ (profiler 를 ì´ìš©í• 경우 ì†ë„ì €í•˜ê°€ 있기 ë•Œë¬¸ì— profiler 를 ë„ê³ psyco 만으로 실행) === 최ì í™” 2ì°¨ : ë¬¸ì œ ì˜ì—ì„ ì¤„ì´ê¸° === ë¹„ë¡ inlining ì„ ì‹œì¼°ì§€ë§Œ ì—¬ì „í•œ ê²ƒì€ selectionFour 부분ì´ë‹¤. selectionFour ê°€ 실행ë˜ëŠ” ê²½ìš°ì˜ ìµœì•…ì˜ ê²½ìš°ëŠ” (n ê¹Œì§€ì˜ ì†Œìˆ˜ê°¯ìˆ˜)^4 ê°€ ëœë‹¤. ì´ë¥¼ 좀 ë” smart 하게 í• ë°©ë²•ì´ ì—†ì„까. 소수와 ê´€ë ¨í•˜ì—¬ 좀 ë” ë˜‘ë˜‘í•˜ê²Œ ê²€ìƒ‰í• ë°©ë²•ì´ ì¡´ìž¬í•˜ë¦¬ë¼ ìƒê°í•œë‹¤. 가장 간단한 ì•„ì´ë””ì–´ë¡œ, 4번째 ê°’ì— ëŒ€í•´ì„œëŠ” 바로 ê³„ì‚°ì„ í• ìˆ˜ê°€ ìžˆê² ë‹¤. selectionFour 를 다ìŒê³¼ ê°™ì´ ìˆ˜ì •í•˜ì˜€ë‹¤. {{{~cpp def selectionFour(aList,aNum): sizeOfList = len(aList) for i in range(sizeOfList): for j in range(sizeOfList): for k in range(sizeOfList): for l in range(sizeOfList): expectedLastValue = aNum-(aList[i]+aList[j]+aList[k]) if expectedLastValue in aList: yield [aList[i],aList[j],aList[k],expectedLastValue] else: break }}} inlining ì„ í•˜ì§€ 않았ìŒì—ë„ 6.4초대를 기ë¡í•˜ì˜€ë‹¤. inlining ì„ í•˜ë©´ 5.7초대를 기ë¡í•œë‹¤. === 최ì í™” 3ì°¨ : PyRex === PyRex 는 Python 코드를 C 코드로 ì „í™˜í•´ì¤€ë‹¤. ì´ë¥¼ ì´ìš©, C 모듈로 만들어 컴파ì¼í•˜ê³ 결과를 실행. ì˜ì™¸ë¡œ 7.x 대로, PsyCo 를 ì¼ì„때보다 ì˜¤ížˆë ¤ ì„±ëŠ¥ì´ ë–¨ì–´ì¡Œë‹¤. C 코드를 보니 웬걸. ì „í˜€ 알아볼수가 없는 코드다. ì°¨ë¼ë¦¬ 깨ë—하게 ì§ì ‘ 작성해주는게 ì„±ëŠ¥í–¥ìƒ ìƒìœ¼ë¡œëŠ” ìœ ë¦¬í•˜ê² ë‹¤ëŠ” ìƒê°. === í•œ ì¼ ëŒ€ë¹„ ëŠë‚€ì === * ì´ì „ì—ë„ ëŠë‚€ ì ì´ì§€ë§Œ, 한가지 ë¬¸ì œë¥¼ 아주 깊게 í’€ì–´ë³´ë ¤ê³ í•˜ëŠ” ê²ƒë„ ì—¬ëŸ¬ê°€ì§€ë¡œ í•™ìŠµì´ ëœë‹¤. * ê³¼ì •ì— ëŒ€í•œ 기ë¡. 무언가 잘 관리가 안ë˜ëŠ” ìƒí™©ì´ 오래ë ë•Œ, í•œ ì¼ë“¤ê³¼ ì‚¬ì‹¤ë“¤ì„ ê¸°ë¡í•´ë³´ëŠ” ê²ƒì€ ìƒë‹¹ížˆ ë„ì›€ì´ ëœë‹¤. === ë” í–ˆì—ˆìœ¼ë©´ 하는 ì . 보완 === * PrimeNumber ì˜ ìµœì í™”ì— ëŒ€í•´ì„œ. ê¸°ì¡´ì— ìžˆëŠ” ì•Œê³ ë¦¬ì¦˜ì´ ì•„ë‹Œ, ì§ì ‘ 최ì 화를 ì‹œë„해보는 것으로 ë” ë§Žì€ ê²ƒì„ í•™ìŠµí• ìˆ˜ 있으리ë¼. ì´ë²ˆì˜ 경우는 2시간 작업으로 계íšì„ ìž¡ì•˜ë˜ ê´€ê³„ë¡œ. * ì´ëŸ¬í•œ ë¬¸ì œì˜ ê²½ìš° íŠ¹ì • ì•Œê³ ë¦¬ì¦˜ì˜ ì•„ì£¼ 최ì í™” ëœ ê²°ê³¼ë¬¼ì´ ë‹µì´ê¸° 보다는, 무언가 다른 ì°¨ì›ì—ì„œ ë´¤ì„ë•Œ 너무나 빨리 ë‹µì´ ë‚˜ì˜¤ê²Œ ë˜ëŠ” 경우ì¼ê²ƒì´ë¼ 추측. ì „í˜€ 다른 ë°©ë²•ì˜ ì–´í”„ë¡œì¹˜ë„ ìƒê°í•´ë³´ê³ 싶다. ---- SummationOfFourPrimes