找回密码
 用户注册

QQ登录

只需一步,快速开始

查看: 4030|回复: 0

幻方算法(Magic Square)

[复制链接]
发表于 2011-12-27 12:33:04 | 显示全部楼层 |阅读模式

一、幻方按照阶数可成了三类,
奇数阶幻方双偶阶幻方单偶阶幻方
二、奇数阶幻方(劳伯法)
奇数阶幻方最经典的填法是罗伯法。填写的方法是:
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. int array[15][15];
  4. int init(int degree)                                  //初始化
  5. {
  6.         int i;
  7.         int j;
  8.        
  9.         for(i=0; i<=degree+1; i++)
  10.         for(j=0; j<=degree+1; j++)
  11.                 array[i][j] = 0;
  12.         return 0;
  13. }
  14. int test_print(int x, int y, int w, int h)            //测试用的,输出以(x,y)为原点,宽为w,高为h,这个区域的数值
  15. {
  16.         int i;
  17.         int j;
  18.         for(i=y; i<=y+h-1; i++){
  19.                 for(j=x; j<=x+w-1; j++){
  20.                         printf("%2d ",array[i][j]);
  21.                 }
  22.                 printf("\n");
  23.         }
  24.         return 0;
  25. }
  26. int lao_bo_er(int degree, int x, int y, int num)      //劳伯法
  27. {
  28.         int i;
  29.         int j;
  30.         int k;
  31.        
  32.         i = y;
  33.         j = degree/2 + x;
  34.         for(k=num; k<=num+degree*degree-1; k++){
  35.                 array[i][j] = k;
  36.                 if((k-num+1)%degree == 0){            //如果这个数所要放的格已经有数填入
  37.                         i = (i-y+1)%degree+y;
  38.                 }
  39.                 else{                                 //每一个数放在前一个数的右上一格
  40.                         i = (i-y-1+degree)%degree+y;
  41.                         j = (j-x+1)%degree+x;
  42.                 }
  43.         }
  44.         return 0;
  45. }
  46. int seq_range(int degree)                             //把数字按顺序填
  47. {
  48.         int i;
  49.         int j;
  50.         int num;
  51.        
  52.         num = 1;
  53.         for(i=1; i<=degree; i++){
  54.                 for(j=1; j<=degree; j++){
  55.                         array[i][j] = num++;
  56.                 }
  57.         }
  58.         return 0;
  59. }
  60. int si_te_la_zi(int degree, int x, int y, int num)    //斯特拉兹法
  61. {
  62.         int deg;
  63.         int k;
  64.         int temp;
  65.         int i;
  66.         int j;
  67.        
  68.         deg = degree/2;
  69.         lao_bo_er(deg, x, y, num);                    //用罗伯法,依次在A象限,D象限,B象限,C象限按奇数阶幻方的填法填数
  70.         lao_bo_er(deg, x+deg, y, num+2*deg*deg);
  71.         lao_bo_er(deg, x, y+deg, num+3*deg*deg);
  72.         lao_bo_er(deg, x+deg, y+deg, num+deg*deg);
  73.         k = (degree-2)/4;
  74.         for(i=1; i<=deg; i++){                        //A象限和C象限对换数据
  75.                 for(j=1; j<=k; j++){
  76.                         temp = array[i][j];
  77.                         array[i][j] = array[i+deg][j];
  78.                         array[i+deg][j]=temp;
  79.                 }
  80.                 for(j=deg+deg/2+1; j>=deg+deg/2-k+3; j--){
  81.                         temp = array[i][j];
  82.                         array[i][j] = array[i+deg][j];
  83.                         array[i+deg][j]=temp;
  84.                 }
  85.         }
  86.        
  87.         for(i=j=1; j<=deg/2+k; j++){                  //B象限和D象限对换数据
  88.                 temp = array[i+deg/2][j];
  89.                 array[i+deg/2][j] = array[i+deg+deg/2][j];
  90.                 array[i+deg+deg/2][j]=temp;
  91.         }
  92.        
  93.         return 0;
  94. }
  95. int hai_er_fa(int degree)                             //海尔法
  96. {
  97.         int i;
  98.         int j;
  99.         int complement;
  100.         int deg;
  101.        
  102.         seq_range(degree);
  103.        
  104.         complement = degree*degree+1;
  105.         deg = degree/4;
  106.         for(i=0; i<deg; i++){
  107.                 for(j=0; j<deg; j++){                 //对角线上的数字换成和它互补的数
  108.                         array[i*4+1][j*4+1] = complement - array[i*4+1][j*4+1];
  109.                         array[i*4+1][j*4+4] = complement - array[i*4+1][j*4+4];
  110.                         array[i*4+4][j*4+1] = complement - array[i*4+4][j*4+1];
  111.                         array[i*4+4][j*4+4] = complement - array[i*4+4][j*4+4];
  112.                        
  113.                         array[i*4+2][j*4+2] = complement - array[i*4+2][j*4+2];
  114.                         array[i*4+2][j*4+3] = complement - array[i*4+2][j*4+3];
  115.                         array[i*4+3][j*4+2] = complement - array[i*4+3][j*4+2];
  116.                         array[i*4+3][j*4+3] = complement - array[i*4+3][j*4+3];
  117.                 }
  118.         }
  119.         return 0;
  120. }
  121. int main()
  122. {
  123.         int degree;
  124.         printf("please input the degree\n");
  125.         scanf("%d",°ree);
  126.         init(degree);
  127.         if(degree%2 == 1){                            //奇数阶幻方
  128.                 lao_bo_er(degree,1,1,1);
  129.                 test_print(1,1,degree,degree);
  130.         }
  131.         else if(degree%4 == 2){                       //双偶阶幻方
  132.                 si_te_la_zi(degree, 1, 1, 1);
  133.                 test_print(1,1,degree,degree);
  134.         }
  135.         else{                                         //单偶阶幻方
  136.                 hai_er_fa(degree);
  137.                 test_print(1,1,degree,degree);
  138.         }
  139.        
  140.         return 0;
  141. }
复制代码

作者:xuyuanfan77 发表于2011-12-27 1:06:42 原文链接
您需要登录后才可以回帖 登录 | 用户注册

本版积分规则

Archiver|手机版|小黑屋|ACE Developer ( 京ICP备06055248号 )

GMT+8, 2024-5-8 00:01 , Processed in 0.036214 second(s), 5 queries , Redis On.

Powered by Discuz! X3.5

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表