The Fibonacci sequence is defined using the following recursive formula:
F(0) = 0
F(1) = 1
F(M) = F(M - 1) + F(M - 2) if M >= 2
A small frog wants to get to the other side of a river. The frog is initially located at one bank of the river (position −1) and wants to get to the other bank (position N). The frog can jump over any distance F(K), where F(K) is the K-th Fibonacci number. Luckily, there are many leaves on the river, and the frog can jump between the leaves, but only in the direction of the bank at position N.
The leaves on the river are represented in an array A consisting of N integers. Consecutive(連續的) elements of array A represent consecutive positions from 0 to N − 1 on the river. Array A contains only 0s and/or 1s:
0 represents a position without a leaf; 1 represents a position containing a leaf. The goal is to count the minimum number of jumps in which the frog can get to the other side of the river (from position −1 to position N). The frog can jump between positions −1 and N (the banks of the river) and every position containing a leaf.
For example, consider array A such that:
A[0] = 0
A[1] = 0
A[2] = 0
A[3] = 1
A[4] = 1
A[5] = 0
A[6] = 1
A[7] = 0
A[8] = 0
A[9] = 0
A[10] = 0
The frog can make three jumps of length F(5) = 5, F(3) = 2 and F(5) = 5.
Write a function:
func Solution(A []int) int
that, given an array A consisting of N integers, returns the minimum number of jumps by which the frog can get to the other side of the river. If the frog cannot reach the other side of the river, the function should return −1.
For example, given:
A[0] = 0
A[1] = 0
A[2] = 0
A[3] = 1
A[4] = 1
A[5] = 0
A[6] = 1
A[7] = 0
A[8] = 0
A[9] = 0
A[10] = 0
the function should return 3, as explained above.
Write an efficient algorithm for the following assumptions:
N is an integer within the range [0..100,000]; each element of array A is an integer that can have one of the following values: 0, 1. Copyright 2009–2021 by Codility Limited. All Rights Reserved. Unauthorized copying, publication or disclosure prohibited.
一只小青蛙想到對岸。它開始位於河的另一邊 位置-1, 想要到對面的河岸 位置N . 青蛙可以跳任意距離 F(K). 其中F(K)是第K個斐波那契數. 且河上有許多樹葉 A[0] = 0 代表位置 0 沒有樹葉, 1 代表有樹葉 青蛙可以在樹葉之間跳, 但只能朝河岸 N 的方向跳 找出最小跳的次數
廣度優先搜尋 (Breadth-First Search, BFS) 問題. 對於河上有樹葉的位置index, 則遍歷比index小的斐波那契數f, 只要 index - f 這個位置可以達到, 這index的位置就可以經過一次跳躍長度為f
package fibfrog
/**
* @description: 產生不大於n的斐波那契數的列表
* @param {int} N
* @return {*}
*/
func Fib(N int) (fibArr []int) {
fibArr = append(fibArr, 0)
fibArr = append(fibArr, 1)
fibArr = append(fibArr, 1)
i := 2
for fibArr[i] < N {
i = i + 1
fibArr = append(fibArr, fibArr[i-1]+fibArr[i-2])
}
return fibArr
}
func Solution(A []int) int {
// 終點
A = append(A, 1)
N := len(A)
fibArr := Fib(N)
// 一次就可以從 -1 跳到 N
if fibArr[len(fibArr)-1] == N {
return 1
}
fibArr = fibArr[1 : len(fibArr)-1]
// fmt.Println(fibArr)
// get the leafs that can be reached from the starting shore
reachable := make([]int, N)
for _, v := range fibArr {
if A[v-1] == 1 {
// 找到樹葉
reachable[v-1] = 1
}
}
// 一開始只能跳到 index: 4 , fib是 [1 1 2 3 5 8], 會使用到 5
// fmt.Println("re", reachable) // [0 0 0 0 1 0 0 0 0 0 0 0]
// iterate all the positions until you reach the other shore
for i := 0; i < N; i++ {
// 忽略不是葉子或已經找過的path
if A[i] != 1 || reachable[i] > 0 {
continue
}
// get the optimal jump count to reach this leaf
if A[i] == 1 {
// 有樹葉
// 遍歷斐波那契數列, 尋找最少的跳躍次數
minJump := i + 1
canJump := false
for _, f := range fibArr {
previousIdx := i - f
if previousIdx < 0 || reachable[previousIdx] == 0 {
// fmt.Printf("[No] %d :previousIdx = %d reachable = %v \n", i, previousIdx, reachable)
continue
}
if minJump > reachable[previousIdx] {
// 此 previousIdx 位置可以到達
// fmt.Printf("%d :previousIdx = %d reachable = %v \n", i, previousIdx, reachable)
minJump = reachable[previousIdx]
canJump = true
}
}
if canJump {
reachable[i] = minJump + 1
}
}
// fmt.Printf("i=%d , reachable = %v \n", i, reachable)
}
if reachable[len(reachable)-1] == 0 {
return -1
}
return reachable[len(reachable)-1]
}