組合數學-換方算法分析

註釋:原文轉自:http://www.cnblogs.com/panlijiao/archive/2012/05/11/2496757.html

 

一、幻方按照階數可成了三類,奇數階幻方雙偶階幻方單偶階幻方

二、奇數階幻方(勞伯法)

奇數階幻方最經典的填法是羅伯法。填寫的方法是:

1(或最小的數)放在第一行正中;按以下規律排列剩下的(n×n1)個數:
1每一個數放在前一個數的右上一格;

(2如果這個數所要放的格已經超出了頂行那麼就把它放在底行,仍然要放在右一列;

3如果這個數所要放的格已經超出了最右列那麼就把它放在最左列,仍然要放在上一行;

4如果這個數所要放的格已經超出了頂行且超出了最右列,那麼就把它放在底行且最左列;

5如果這個數所要放的格已經有數填入,那麼就把它放在前一個數的下一行同一列的格內。

例,用該填法獲得的5階幻方:

17

24

1

8

15

23

5

7

14

16

4

6

13

20

22

10

12

19

21

3

11

18

25

2

9

二、雙偶數階幻方(海爾法)

所謂雙偶階幻方就是當n可以被4整除時的偶階幻方,即4K階幻方。在說解法之前我們先說明一個“互補數”定義:就是在n階幻方中,如果兩個數的和等於幻方中最大的數與1的和(即n×n1),我們稱它們爲一對互補數。如在三階幻方中,每一對和爲10的數,是一對互補數 ;在四階幻方中,每一對和爲17的數,是一對互補數。

雙偶數階幻方最經典的填法是海爾法。填寫的方法是:

8階幻方爲例:
1先把數字按順序填。然後,按4×4把它分割成4塊(如圖)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

2每個小方陣對角線上的數字(如左上角小方陣部分),換成和它互補的數。

64

2

3

61

60

6

7

57

9

55

54

12

13

51

50

16

17

47

46

20

21

43

42

24

40

26

27

37

36

30

31

33

32

34

35

29

28

38

39

25

41

23

22

44

45

19

18

48

49

15

14

52

53

11

10

56

8

58

59

5

4

62

63

1



三、單偶數階幻方(斯特拉茲法)

所謂單偶階幻方就是當n不可以被4整除時的偶階幻方,即4K+2階幻方。如(n=61014……)的幻方。

 

單偶數階幻方最經典的填法是斯特拉茲法。填寫的方法是:

10階幻方爲例。這時,k=2
1)把魔方陣分爲ABCD四個象限,這樣每一個象限肯定是奇數階。用羅伯法,依次在A象限,D象限,B象限,C象限按奇數階幻方的填法填數。


2)在A象限的中間行、中間格開始,按自左向右的方向,標出k格。A象限的其它行則標出最左邊的k格。將這些格,和C象限相對位置上的數互換位置。


3)在B象限所有行的中間格,自右向左,標出k1格。(注:6階幻方由於k1=0,所以不用再作BD象限的數據交換),將這些格,和D象限相對位置上的數互換位置。

四、源代碼如下,已加詳細註釋

  1. #include<stdio.h>  
  2. #include<stdlib.h>  
  3.   
  4. int array[15][15];  
  5.   
  6. int init(int degree)                                  //初始化  
  7. {  
  8.     int i;  
  9.     int j;  
  10.       
  11.     for(i=0; i<=degree+1; i++)  
  12.     for(j=0; j<=degree+1; j++)  
  13.         array[i][j] = 0;  
  14.     return 0;  
  15. }  
  16.   
  17. int test_print(int x, int y, int w, int h)            //測試用的,輸出以(x,y)爲原點,寬爲w,高爲h,這個區域的數值  
  18. {  
  19.     int i;  
  20.     int j;  
  21.     for(i=y; i<=y+h-1; i++){  
  22.         for(j=x; j<=x+w-1; j++){  
  23.             printf("%2d ",array[i][j]);  
  24.         }  
  25.         printf("\n");  
  26.     }  
  27.     return 0;  
  28. }  
  29.   
  30. int lao_bo_er(int degree, int x, int y, int num)      //勞伯法  
  31. {  
  32.     int i;  
  33.     int j;  
  34.     int k;  
  35.       
  36.     i = y;  
  37.     j = degree/2 + x;  
  38.     for(k=num; k<=num+degree*degree-1; k++){  
  39.         array[i][j] = k;  
  40.         if((k-num+1)%degree == 0){            //如果這個數所要放的格已經有數填入  
  41.             i = (i-y+1)%degree+y;  
  42.         }  
  43.         else{                                 //每一個數放在前一個數的右上一格  
  44.             i = (i-y-1+degree)%degree+y;  
  45.             j = (j-x+1)%degree+x;  
  46.         }  
  47.     }  
  48.     return 0;  
  49. }  
  50.   
  51. int seq_range(int degree)                             //把數字按順序填  
  52. {  
  53.     int i;  
  54.     int j;  
  55.     int num;  
  56.       
  57.     num = 1;  
  58.     for(i=1; i<=degree; i++){  
  59.         for(j=1; j<=degree; j++){  
  60.             array[i][j] = num++;  
  61.         }  
  62.     }  
  63.     return 0;  
  64. }  
  65.   
  66. int si_te_la_zi(int degree, int x, int y, int num)    //斯特拉茲法  
  67. {  
  68.     int deg;  
  69.     int k;  
  70.     int temp;  
  71.     int i;  
  72.     int j;  
  73.       
  74.     deg = degree/2;  
  75.     lao_bo_er(deg, x, y, num);                    //用羅伯法,依次在A象限,D象限,B象限,C象限按奇數階幻方的填法填數  
  76.     lao_bo_er(deg, x+deg, y, num+2*deg*deg);  
  77.     lao_bo_er(deg, x, y+deg, num+3*deg*deg);  
  78.     lao_bo_er(deg, x+deg, y+deg, num+deg*deg);  
  79.   
  80.     k = (degree-2)/4;  
  81.     for(i=1; i<=deg; i++){                        //A象限和C象限對換數據  
  82.         for(j=1; j<=k; j++){  
  83.             temp = array[i][j];  
  84.             array[i][j] = array[i+deg][j];  
  85.             array[i+deg][j]=temp;  
  86.         }  
  87.         for(j=deg+deg/2+1; j>=deg+deg/2-k+3; j--){  
  88.             temp = array[i][j];  
  89.             array[i][j] = array[i+deg][j];  
  90.             array[i+deg][j]=temp;  
  91.         }  
  92.     }  
  93.       
  94.     for(i=j=1; j<=deg/2+k; j++){                  //B象限和D象限對換數據  
  95.         temp = array[i+deg/2][j];  
  96.         array[i+deg/2][j] = array[i+deg+deg/2][j];  
  97.         array[i+deg+deg/2][j]=temp;  
  98.     }  
  99.       
  100.     return 0;  
  101. }  
  102.   
  103. int hai_er_fa(int degree)                             //海爾法  
  104. {  
  105.     int i;  
  106.     int j;  
  107.     int complement;  
  108.     int deg;  
  109.       
  110.     seq_range(degree);  
  111.       
  112.     complement = degree*degree+1;  
  113.     deg = degree/4;  
  114.     for(i=0; i<deg; i++){  
  115.         for(j=0; j<deg; j++){                 //對角線上的數字換成和它互補的數  
  116.             array[i*4+1][j*4+1] = complement - array[i*4+1][j*4+1];  
  117.             array[i*4+1][j*4+4] = complement - array[i*4+1][j*4+4];  
  118.             array[i*4+4][j*4+1] = complement - array[i*4+4][j*4+1];  
  119.             array[i*4+4][j*4+4] = complement - array[i*4+4][j*4+4];  
  120.               
  121.             array[i*4+2][j*4+2] = complement - array[i*4+2][j*4+2];  
  122.             array[i*4+2][j*4+3] = complement - array[i*4+2][j*4+3];  
  123.             array[i*4+3][j*4+2] = complement - array[i*4+3][j*4+2];  
  124.             array[i*4+3][j*4+3] = complement - array[i*4+3][j*4+3];  
  125.         }  
  126.     }  
  127.     return 0;  
  128. }  
  129.   
  130. int main()  
  131. {  
  132.     int degree;  
  133.     printf("please input the degree\n");  
  134.     scanf("%d",&degree);  
  135.     init(degree);  
  136.     if(degree%2 == 1){                            //奇數階幻方  
  137.         lao_bo_er(degree,1,1,1);  
  138.         test_print(1,1,degree,degree);  
  139.     }  
  140.     else if(degree%4 == 2){                       //雙偶階幻方  
  141.         si_te_la_zi(degree, 1, 1, 1);  
  142.         test_print(1,1,degree,degree);  
  143.     }  
  144.     else{                                         //單偶階幻方  
  145.         hai_er_fa(degree);  
  146.         test_print(1,1,degree,degree);  
  147.     }  
  148.       
  149.     return 0;  
  150. }  

 

 

 

 

發佈了32 篇原創文章 · 獲贊 5 · 訪問量 16萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章