分治法求最大最小值

简介: 输入n个数,求出该序列的最大和最小值。 参考: http://blog.csdn.net/kennyrose/article/details/7376457 http://www.360doc.com/content/12/0903/10/1317564_233893635.

输入n个数,求出该序列的最大和最小值。

参考:

http://blog.csdn.net/kennyrose/article/details/7376457

http://www.360doc.com/content/12/0903/10/1317564_233893635.shtml

http://www.cnblogs.com/goodwin/archive/2010/11/01/1866391.html

 http://blog.csdn.net/yyywyr/article/details/8003366

 

分治算法通俗的讲就是把一个规模比较大的问题分成n个规模较小的问题来解决,再将每个小规模的问题进行合并,最后得到结果。通常问题规模比较大难以用普通的编程方法实现,或者不可能实现的时候采用分治算法,能够简化问题的解决。

一个例子:

求出一个数组中的最大值和最小值。

 

写的比较好的代码:

 1 #include<iostream>
 2 
 3 void dcMaxMin( int [], int, int, int *, int * );
 4 
 5 void dcMaxMin( int arr[], int low, int high, int *tmpMax, int *tmpMin )
 6 {
 7         if ( high - low <= 1)// 最多俩元素
 8         {
 9                 if (arr[low] <= arr[high] )
10                 {
11                         *tmpMax = arr[high];
12                         *tmpMin = arr[low];
13                         return; // 递归出口
14                 }
15                 else
16                 {
17                         *tmpMax = arr[low];
18                         *tmpMin = arr[high];
19                         return; //递归出口
20                 }
21         }
22 
23         int min1, min2, max1, max2;
24         int mid = ( high - low ) / 2 + low;
25         //递归开始
26         dcMaxMin(arr, low, mid, &max1, &min1);
27         dcMaxMin(arr, mid+1, high, &max2, &min2);
28 
29         *tmpMax = max1 < max2 ? max2 : max1;
30         *tmpMin = min1 < min2 ? min1 : min2;
31 
32 }
33 
34 int main()
35 {
36         int arr[] = {4, 1, 2, 9, 8, 3};
37         int len = sizeof ( arr ) / sizeof( int );
38         int max, min;
39         max = arr[0];
40         min = arr[0];
41         dcMaxMin(arr, 0, len-1, &max, &min);
42         std::cout << max << "\t" << min << std::endl;
43 
44         return 0;
45 }
View Code
 1 #include <iostream>
 2 using namespace std;
 3 void min_max(int a[],int i,int j,int &min,int &max)
 4 {
 5  int mid,max1,max2,min1,min2;
 6  if(i==j){
 7   max=a[i];
 8   min=a[i];
 9   return ;
10  }
11  if(j==i+1)
12  {
13   if(a[i]>a[j]){min=a[j];max=a[i];}
14   else{min=a[i];max=a[j];}
15  }
16  else
17  {
18   mid=(i+j)/2;
19   min_max(a,i,mid,min1,max1);
20   min_max(a,mid+1,j,min2,max2);
21   if(min1>min2)min=min2;
22   else min=min1;
23   if(max1>max2)max=max1;
24   else max=max2;
25  }
26 }
27 int main()
28 {
29  int n,m,a[100],min,max;
30  cin>>n;
31  while(n!=0)
32  {
33   cin>>m;
34   for(int i=1;i<=m;i++)cin>>a[i];
35   min_max(a,1,m,min,max);
36   cout<<min<<' '<<max<<endl;
37   n--;
38  }
39  return 0;
40 }
View Code
 1 package example;
 2  
 3 public class MaxAndMinValue {
 4  
 5     // 直接算法 得到最大值和最小值
 6     public static void main(String[] args) {
 7         int[] A = { -18, -16, 9, -5, 7, -40, 0, 35 };
 8         System.out.println(getMaxValue(A));
 9         System.out.println(getMinValue(A));
10         System.out.println(getMax(A, 0, A.length - 1));
11  
12     }
13  
14     // 直接算法求最大值
15     public static int getMaxValue(int[] array) {
16         int Max = 0;
17         for (int i = 0; i < (array.length - 1); i++) {
18             if (array[i] == array[i + 1]) {
19                 Max = array[i + 1];
20             }
21             if (array[i] < array[i + 1]) {
22                 Max = array[i + 1];
23             }
24             if (array[i] > array[i + 1]) {
25                 Max = array[i];
26                 array[i] = array[i + 1];
27                 array[i + 1] = Max;
28  
29             }
30         }
31         return Max;
32     }
33  
34     // 直接算法求最小值
35     public static int getMinValue(int[] array) {
36  
37         int Min = 0;
38         for (int i = 0; i < (array.length - 1); i++) {
39             if (array[i] == array[i + 1]) {
40                 Min = array[i + 1];
41             } else if (array[i] < array[i + 1]) {
42                 Min = array[i];
43                 array[i] = array[i + 1];
44                 array[i + 1] = Min;
45             } else if (array[i] > array[i + 1]) {
46                 Min = array[i + 1];
47             }
48         }
49         return Min;
50     }
51  
52     // 用分治法求最大最小值
53     public static int getMax(int[] array, int i, int j) {
54         int Max1 = 0;
55         int Max2 = 0;
56         if (i == j) {
57             return Max1 = Max2 = array[j];
58         } else if (i == (j - 1)) {
59             Max1 = array[i];
60             Max2 = array[j];
61             return Max1 > Max2 ? Max1 : Max2;
62         } else {
63             int mid = (i + j) / 2;
64             Max1 = getMax(array, i, mid);
65             Max2 = getMax(array, mid, j);
66             return Max1 > Max2 ? Max1 : Max2;
67         }
68     }
69 }
70  
71 
72 假设数组的大小为8,用直接的算法,最大值最小值总需要比较14次,而用分治算法可以一次性求出最大和最小,只需要10次比较。
View Code
 1 **
 2  * 
 3  * 分治法求数组的最小值和最大值
 4  */
 5 import java.util.Arrays;
 6 
 7 public class MinAndMaxArray {
 8     public static void main(String[] args) {
 9         int arr[] = { -2, -9, 0, 5, 2 };
10         int result[] = new int[2];
11         result = minMax(arr, 0, arr.length - 1);
12         System.out.println(Arrays.toString(result));
13     }
14 
15     public static int[] minMax(int[] arr, int l, int r) {
16         int min = 0;
17         int max = 0;
18         if (l == r) {
19             min = arr[l];
20             max = arr[l];
21         } else if (l + 1 == r) {
22             if (arr[l] < arr[r]) {
23                 min = arr[l];
24                 max = arr[r];
25             } else {
26                 min = arr[r];
27                 max = arr[l];
28             }
29         } else {
30             int mid = (l + r) / 2;
31             int[] preHalf = minMax(arr, l, mid);
32             int[] postHalf = minMax(arr, mid + 1, r);
33             min = preHalf[0] < postHalf[0] ? preHalf[0] : postHalf[0];
34             max = preHalf[1] > postHalf[1] ? preHalf[1] : postHalf[1];
35         }
36 
37         return new int[] { min, max };
38     }
39 }
View Code

我自己写的代码:(保留以作备份)——以后尽量使用引用,而不要使用指针来传递参数。

 1 #include<stdio.h>
 2 #include<time.h>
 3 #include<stdlib.h>
 4 int maxMin(int *a,int i,int j,int *max,int *min);
 5 int main()
 6 {
 7     int *a;
 8     int i,n;
 9     int max,min;
10     
11     scanf("%d",&n);
12     a=(int *)malloc(sizeof(int)*n);
13     srand((unsigned int)time(0));
14     for(i=0;i<n;i++) a[i]=rand()%91+10;
15     for(i=0;i<n;i++)
16     {
17         printf("%d ",a[i]);
18         if((i+1)%5==0) printf("\n");
19     }
20     printf("\n\n");
21     maxMin(a,0,n-1,&max,&min);
22     printf("%d %d\n",max,min);/**/
23     return 0;
24 }
25 int maxMin(int *a,int i,int j,int *max,int *min)
26 {
27     int mid;
28     int lmax,lmin,rmax,rmin;
29     if(j==i) { *max=a[i]; *min=a[i]; return 0;}
30     else if(j-i==1)
31     {
32         if(a[i]>a[j])  {*max=a[i];*min=a[j];return 0;}
33         else  {*max=a[j];*min=a[i];return 0;}
34     }
35     else
36     {
37         mid=i+(j-i)/2;
38         maxMin(a,i,mid,&lmax,&lmin);
39         maxMin(a,mid+1,j,&rmax,&rmin);
40         if(lmax>rmax)   *max=lmax;
41         else   *max=rmax;
42         if(lmin<rmin)     *min=lmin;
43         else   *min=rmin;
44     }
45     return 0;
46 }
View Code
相关文章
|
8月前
|
算法 C语言
分治法——找众数
分治法——找众数
|
5月前
|
算法 测试技术
【算法】二分算法——寻找旋转排序数组中的最小值
【算法】二分算法——寻找旋转排序数组中的最小值
|
8月前
|
机器学习/深度学习 算法 测试技术
【线段树】【区间更新】2916. 子数组不同元素数目的平方和 II
【线段树】【区间更新】2916. 子数组不同元素数目的平方和 II
【线段树】【区间更新】2916. 子数组不同元素数目的平方和 II
|
8月前
|
算法 测试技术 C#
【折半处理 二分查找】1755. 最接近目标值的子序列和
【折半处理 二分查找】1755. 最接近目标值的子序列和
|
8月前
|
算法
贪心算法:排列算式
贪心算法:排列算式
37 0
|
人工智能 BI
1236:区间合并 2020-12-27
1236:区间合并 2020-12-27
|
Java
寻找两个有序数组的中位数
寻找两个有序数组的中位数
118 0
|
人工智能 算法
有序数组的平方
有序数组的平方
|
算法
【leedcode】0004. 两个有序数组的中位数
【leedcode】0004. 两个有序数组的中位数
79 0