1、遞歸簡介java
2、兩個基礎遞歸問題:算法
1.計算階乘編程
階乘的定義:優化
0!=1;this
n!=n*(n-1)*...*2*1=n*(n-1)!; (n>0)spa
1 import java.util.Scanner; 2 public class ComputeFactorial{ 3 public static void main(String[] args){ 4 Scanner input=new Scanner(System.in); 5 System.out.print("Enter a nonnegtive integer:"); 6 int n=input.nextInt(); 7 System.out.println(n+"!="+factorial(n)); 8 } 9 private static int factorial(int n){ 10 if(n==0) 11 return 1; 12 else 13 return n*factorial(n-1); 14 } 15 }
2.計算斐波那契數設計
斐波那契數的定義:3d
fib(0)=0;code
fib(1)=1;blog
fib(index)=fib(index-1)+fib(index-2); index>=2
1 import java.util.Scanner; 2 public class ComputeFibonacci{ 3 public static void main(String[] args){ 4 Scanner input=new Scanner(System.in); 5 System.out.print("Enter an index for Fobonacci number:"); 6 int n=input.nextInt(); 7 System.out.println(fib(n)); 8 } 9 private static int fib(int n){ 10 if(n==0) 11 return 0; 12 else if(n==1) 13 return 1; 14 else 15 return fib(n-1)+fib(n-2); 16 } 17 }
以上兩個例子只是用來演示遞歸的概念,實際上用循環來實現效率更高,代碼以下:
1 //階乘 2 import java.util.Scanner; 3 public class ComputeFactorial{ 4 public static void main(String[] args){ 5 Scanner input=new Scanner(System.in); 6 System.out.print("Enter a nonnegtive integer:"); 7 int n=input.nextInt(); 8 int n0=n; 9 int s=1; 10 while(n>0) 11 { 12 s*=n--; 13 } 14 System.out.println(n0+"!="+ s); 15 } 16 }
1 //斐波那契數 2 import java.util.Scanner; 3 public class ComputeFibonacci{ 4 public static void main(String[] args){ 5 Scanner input=new Scanner(System.in); 6 System.out.print("Enter an index for Fobonacci number:"); 7 int n=input.nextInt(); 8 int f0=0,f1=1,currentFib=0; 9 for(int i=2;i<=n;i++){ 10 currentFib=f0+f1; 11 f0=f1; 12 f1=currentFib; 13 } 14 System.out.println(currentFib); 15 } 16 }
3、遞歸的輔助方法
1.遞歸的輔助方法
用遞歸的思路解決迴文字符串問題:
1 import java.util.Scanner; 2 public class Palindrome{ 3 public static void main(String[] args){ 4 Scanner input=new Scanner(System.in); 5 System.out.print("Enter a string:"); 6 String text=input.nextLine(); 7 System.out.println(isPalindrome(text)); 8 } 9 private static boolean isPalindrome(String text){ 10 if(text.length()<=1) 11 return true; 12 else if(text.charAt(0)!=text.charAt(text.length()-1)) 13 return false; 14 else 15 return isPalindrome(text.substring(1,text.length()-1)); 16 } 17 18 }
因爲每次遞歸時都要建立一個新的字符串,所以不夠高效。爲避免建立新字符串,可使用low和high下標來代表子串的範圍。
1 import java.util.Scanner; 2 public class Palindrome{ 3 public static void main(String[] args){ 4 Scanner input=new Scanner(System.in); 5 System.out.print("Enter a string:"); 6 String text=input.nextLine(); 7 System.out.println(isPalindrome(text)); 8 } 9 private static boolean isPalindrome(String text){ 10 return isPalindrome(text,0,text.length()-1); 11 } 12 private static boolean isPalindrome(String text,int low,int high){ 13 if(low>=high) 14 return true; 15 else if(text.charAt(low)!=text.charAt(high)) 16 return false; 17 else 18 return isPalindrome(text,low+1,high-1); 19 } 20 21 }
程序中定義了兩個重載的isPalindrome方法,第一個方法將low和high的值傳遞給第二個方法,第二個方法採用遞歸調用,檢測不斷縮小的子串是否迴文。在遞歸程序設計中定義第二個方法來接收附加的參數是一種經常使用的技巧,稱爲遞歸的輔助方法。
2.遞歸的輔助方法舉例
選擇排序
1 public class RecursiveSeletionSort{ 2 public static void sort(double[] list){ 3 sort(list,0,list.length-1); 4 } 5 public static void sort(double[] list,int low,int high){ 6 if(low<high){ 7 int indexOfMin=low; 8 double min=list[low]; 9 for(int i=low+1;i<=high;i++){ 10 if(min<list[i]){ 11 min=list[i]; 12 indexOfMin=i; 13 } 14 } 15 //Swap 16 list[indexOfMin]=list[low]; 17 list[low]=min; 18 19 sort(list,low+1,high); 20 } 21 } 22 }//只是算法,並不是完整程序
二分查找
1 public class RecursiveBinarySearch{ 2 public static int binarySearch(int[] list,int key){ 3 int low=0; 4 int high=list.length-1; 5 return binarySearch(list,key,low,high); 6 } 7 public static int binarySearch(int[] list,int key,int low,int high){ 8 if(low>high) 9 return -low-1; 10 int mid=(low+high)/2; 11 if(key<list[mid]) 12 return binarySearch(list,key,low,mid-1); 13 else if(key=list[mid]) 14 return mid; 15 else 16 return binarySearch(list,key,mid+1,high); 17 } 18 }//只是算法,並不是完整程序
4、尾遞歸
若是從遞歸調用返回時沒有待定的操做要完成,這個遞歸就稱爲尾遞歸(tail recursive)。某些編譯器會優化尾遞歸以減小桟空間。可使用輔助參數將非尾遞歸轉化爲尾遞歸。
將階乘的例子改成尾遞歸
5、更多例子
1.求出目錄的大小
1 package book; 2 3 import java.io.File; 4 import java.util.Scanner; 5 6 public class DirectorySize{ 7 public static void main(String[] args){ 8 System.out.print("Enter a directory or a file:"); 9 Scanner input=new Scanner(System.in); 10 String directory=input.nextLine(); 11 12 System.out.println(getSize(new File(directory))+"bytes"); 13 14 input.close(); 15 } 16 public static long getSize(File file){ 17 long size=0; 18 if(file.isDirectory()){ 19 File[] files=file.listFiles(); 20 for(int i=0;i<files.length;i++){ 21 size+=getSize(files[i]); 22 } 23 } 24 else 25 size+=file.length(); 26 27 return size; 28 } 29 30 }
2.漢諾塔問題
1 import java.util.Scanner; 2 public class TowersOfHanoi{ 3 public static void main(String[] args){ 4 Scanner input=new Scanner(System.in); 5 System.out.print("Enter number of disks:"); 6 int n=input.nextInt(); 7 System.out.println("The moves are:"); 8 moveDisks(n,'A','B','C'); 9 } 10 private static void moveDisks(int n,char fromTower,char toTower,char auxTower){ 11 if(n==1) 12 System.out.println("Move disk "+n+" from "+fromTower+" to "+toTower); 13 else{ 14 moveDisks(n-1,fromTower,auxTower,toTower); 15 System.out.println("Move disk "+n+" from "+fromTower+" to "+toTower); 16 moveDisks(n-1,auxTower, toTower,fromTower); 17 18 } 19 20 } 21 22 }
3.八皇后問題
1 import java.awt.*; 2 import javax.swing.*; 3 public class EightQueens extends JApplet{ 4 /** 5 * 6 */ 7 private static final long serialVersionUID = 5660619297259886927L; 8 public static final int SIZE=8; 9 private int[] queens=new int[SIZE]; 10 11 public EightQueens(){ 12 search(0); 13 add(new ChessBord(),BorderLayout.CENTER); 14 } 15 16 private boolean isValid(int row,int cloumn){ 17 for(int i=1;i<=row;i++){ 18 if(queens[row-i]==cloumn 19 ||queens[row-i]==cloumn-1 20 ||queens[row-i]==cloumn+1) 21 22 return false; 23 } 24 return true; 25 } 26 27 private boolean search(int row){ 28 if(row==SIZE) 29 return true; 30 31 for(int cloumn=0;cloumn<SIZE;cloumn++){ 32 queens[row]=cloumn; 33 if(isValid(row,cloumn)&&search(row+1)) 34 return true; 35 } 36 37 return false; 38 } 39 40 class ChessBord extends JPanel{ 41 /** 42 * 43 */ 44 private static final long serialVersionUID = 4151757353625344600L; 45 private Image queenImage=new ImageIcon("image/queen.jpg").getImage(); 46 47 ChessBord(){ 48 this.setBorder(BorderFactory.createLineBorder(Color.BLACK,2)); 49 } 50 protected void paintComponent(Graphics g){ 51 super.paintComponent(g); 52 53 //Paint the queens 54 for(int i=0;i<SIZE;i++){ 55 int j=queens[i]; 56 g.drawImage(queenImage,j*getWidth()/SIZE,i*getHeight()/SIZE,getWidth()/SIZE,getHeight()/SIZE,this); 57 } 58 59 //Draw the horizontal and vertical lines 60 for(int i=0;i<SIZE;i++){ 61 g.drawLine(0,i*getHeight()/SIZE,getWidth(),i*getHeight()/SIZE); 62 g.drawLine(i*getWidth()/SIZE,0,i*getWidth()/SIZE,getHeight()); 63 } 64 65 } 66 } 67 }