逆FizzBuzz問題求最短序列

問題描述php

FizzBuzz問題:一個大於0的天然數能整除3,將輸出「Fizz」;能整除5,將輸出「Buzz」;能整除3和5,將輸出「FizzBuzz」;不然輸出本身。編程

逆FizzBuzz問題最短序列:已知一個FizzBuzz問題的非數字輸出序列,求能得到該序列的最短連續數字序列。如「Fizz」的最短序列是「3」,「Fizz Buzz」的最短序列是「9 10」,而不是「3 4 5」。this

 

編程實現spa

  1 <?php
  2 
  3 /**
  4  * @author cenze
  5  *
  6  * 反FizzBuzz問題求最短序列
  7  * 
  8  * $inverseFizzBuzz = new InverseFizzBuzz([
  9  *   'fizz',
 10  *   'fizz',
 11  *   'buzz',
 12  * ]);
 13  * $inverseFizzBuzz->sequence():
 14  * Array ( [0] => 6 [1] => 7 [2] => 8 [3] => 9 [4] => 10 )
 15  */
 16 class InverseFizzBuzz
 17 {
 18     // FizzBuzz問題賦值
 19     const FizzBuzz = [
 20         'fizz' => 3,
 21         'buzz' => 5,
 22         'fizzbuzz' => 15
 23     ];
 24     
 25     // 預約義問題區間
 26     private $range = [
 27         1,
 28         100
 29     ];
 30     
 31     // 待求序列
 32     private $list = [];
 33     
 34     // 待求序列包含項數統計
 35     private $listItemsCount = 0;
 36 
 37     public function __construct($list = [], $range = [])
 38     {
 39         $this->init($list, $range);
 40     }
 41 
 42     /**
 43      * 待求序列、區間初始化
 44      *
 45      * @param array $list
 46      *            待求序列
 47      * @param array $range
 48      *            問題區間
 49      * @param bool $check
 50      *            是否檢查屬性已完成初始化
 51      *            
 52      * @return void
 53      */
 54     private function init($list = [], $range = [], $check = false)
 55     {
 56         if ($list) {
 57             $this->list = $this->inverseList($list);
 58             $this->listItemsCount = count($this->list);
 59             if (! $this->listItemsCount) {
 60                 exit('Invalid list.');
 61             }
 62         }
 63         
 64         if ($range) {
 65             $this->range = $range;
 66         }
 67         
 68         if ($check) {
 69             if (! $this->list) {
 70                 exit('Property list uninitialized.');
 71             }
 72         }
 73     }
 74 
 75     /**
 76      * 將待求解序列反轉
 77      *
 78      * @param array $list
 79      *            待求解序列
 80      *            
 81      * @return mixed
 82      */
 83     private function inverseList($list)
 84     {
 85         $inverseList = [];
 86         foreach ($list as $item) {
 87             if (! array_key_exists($item, self::FizzBuzz)) {
 88                 return null;
 89             }
 90             $inverseList[] = self::FizzBuzz[
 91                 $item
 92             ];
 93         }
 94         
 95         return $inverseList;
 96     }
 97 
 98     /**
 99      * 搜索最短序列是否存在
100      *
101      * @param array $list
102      *            待求序列
103      * @param array $range
104      *            問題區間
105      *            
106      *            
107      * @return mixed
108      */
109     public function sequence($list = [], $range = [])
110     {
111         $this->init($list, $range, true);
112         
113         $sequences = $this->findSequences();
114         if (! $sequences) {
115             return 'Found no sequences.';
116         }
117         
118         $sequence = $this->findShortestSequence($sequences);
119         // 將最短序列序列化後返回
120         return range($sequence[0], $sequence[$this->listItemsCount - 1]);
121     }
122 
123     /**
124      * 搜索最短序列(未序列化)
125      *
126      * @param array $sequences
127      *            序列集合,包含最短序列
128      *            
129      * @return array
130      */
131     private function findShortestSequence($sequences)
132     {
133         // 默認第一條爲最短
134         $shortestSequence = $sequences[0];
135         // 把每個序列看作一條路徑,都有對應的長度。記錄最短的序列長度
136         $shortestSequenceLength = $sequences[0][$this->listItemsCount - 1] - $sequences[0][0];
137         // 爲array_reduce()定義一個callback,用來搜索最短序列
138         $findShortestSequence = function ($shortestSequence, $currentSequence) use(&$shortestSequenceLength) {
139             $currentSequenceLength = $currentSequence[$this->listItemsCount - 1] - $currentSequence[0];
140             if ($currentSequenceLength < $shortestSequenceLength) {
141                 $shortestSequenceLength = $currentSequenceLength;
142                 return $currentSequence;
143             } else {
144                 return $shortestSequence;
145             }
146         };
147         
148         return array_reduce($sequences, $findShortestSequence, $shortestSequence);
149     }
150 
151     /**
152      * 從指定位置開始搜索指定項目的序列
153      *
154      * @param int $item
155      *            指定項目
156      * @param int $start
157      *            起始搜索位置
158      *            
159      * @return int
160      */
161     private function findItemSequenceFromPosition($item, $start)
162     {
163         if ($start % $item == 0) {
164             return $start;
165         }
166         
167         return $start + ($item - $start % $item);
168     }
169 
170     /**
171      * 從指定位置開始搜索首個序列
172      *
173      * @param int $start
174      *            起始搜索位置
175      *            
176      * @return mixed
177      */
178     private function findSequenceFromPosition($start)
179     {
180         $listSequence = [];
181         for ($i = 0; $i < $this->listItemsCount; $i ++) {
182             $itemSequence = $this->findItemSequenceFromPosition($this->list[$i], $start);
183             if ($itemSequence > $this->range[1]) {
184                 return null;
185             } else {
186                 $listSequence[] = $itemSequence;
187                 $start = $itemSequence + 1;
188             }
189         }
190         
191         return $listSequence;
192     }
193 
194     /**
195      * 找到多個序列,其中包括最短序列
196      *
197      * @return void
198      */
199     private function findSequences()
200     {
201         $sequences = [];
202         // 以第一個項爲初始搜索位置在預約區間中搜索,且以第一個項爲遞增步長向後逐步搜索
203         for ($start = $this->list[0]; $start <= $this->range[1]; $start += $this->list[0]) {
204             // 只需找到指定位置開始的第一個序列便可
205             if ($sequence = $this->findSequenceFromPosition($start)) {
206                 $sequences[] = $sequence;
207             }
208         }
209         
210         return $sequences;
211     }
212 }
相關文章
相關標籤/搜索