本篇文章只对奇数阶以及偶数阶中阶数n = 4K的魔方阵进行讨论.下面就让我们进入正题:

1 :魔方阵的相关信息:(百度百科)

https://baike.baidu.com/item/%E9%AD%94%E6%96%B9%E9%98%B5/10973743?fr=aladdin

2 :奇数阶和偶数阶魔方阵的排列规律.(源自百度百科)  (可跳至第三部分)

    2.1 :奇数阶魔方阵的排列规律如下:
  ⑴ :将1放在第一行中间一列;
  ⑵ :从2开始直到n×n止各数依次按下列规则存放;每一个数存放的行比前一个数的行数减1,列数加1(例如上面的三阶魔方阵,5在4的上一行后一列;
  ⑶ :如果上一个数的行数为1,则下一个数的行数为n(指最下一行);例如1在第一行,则2应放在最下一行,列数同样加1;
  ⑷ :当上一个数的列数为n时,下一个数的列数应为1,行数减去1。例如2在第3行最后一列,则3应放在第二行第一列;
  ⑸ :如果按上面规则确定的位置上已有数,或上一个数是第一行第n列时,则把下一个数放在上一个数的下面。例如按上面的规定,4应该放在第1行第2列,
       但该位置已经被占据,所以4就放在3的下面。
 
    2.2 :阶数为n = 4K 的魔方阵的排列规律如下:
  (1) :先将整个方阵划分成k*k个4阶方阵,然后在每个4阶方阵的对角线上做记号;
  (2) :由左而右、由上而下,遇到没有记号的位置才填数字,但不管是否填入数字,每移动一格数字都要加1;
  (3) :自右下角开始,由右而左、由下而上,遇到没有数字的位置就填入数字,但每移动一格数字都要加1。
 
3 :排列规律的解读,代码实现以及实验结果( ar 二维数组 ,ROW ==行,  COL ==  列)
 3.1 :奇数阶魔方阵(n = 2*K + 1)
 奇数阶魔方阵的排列规律主要有以下三点:
   3.1.1) : 数字1的位置是确定的,摆放在第一行中间的位置,即:  ar[0][COL/2] == 1;
 
   3.1.2)(重点)  :假设当前摆放的数行列下标为 [preRow][preCol] , 则下一个数的摆放的位置的下标应为[preRow - 1] [preCol + 1] ,注意在这里(preRow - 1)
       以及(preCol + 1)的值都有可能越界. 因此,需要对变化后的值进行操作.我们以ROW=COL=3为例,假设上一个值的下标为 [0][3] ,
          那么下一个值的下标应为[2][1] ,他们与(preRow - 1)以及(preCol + 1)
       即 [0 - 1][3 + 1]的关系,为[(-1 + 3)%3] [(3+1+3)%3], 即[(preRow - 1+ ROW)%ROW][(preCol + 1+ COL)%COL].
       图示:
 
         ,
      2的摆放位置的下标应为[(0 -1 + 3)%3][(1+1)%3] == [2][2] ,
      即:  
 
   3.1.3) : 若将要摆放的位置已有值存在,则将该值排放在上一个值的下一行.
      例如:以三阶魔方阵为例: 按照相应的规则摆放1,2,3
                       ,
      要摆放4的位置的下标通过计算可以得出是[0][1],但是该位置已有数值1,因此4要摆放在3的下一行,即[preRow - 1][preCol],
      同时用3.1.2的方法对[preRow-1]进行 防越界处理:   [(preRow - 1 + ROW)%ROW][preCol]. 
    代码如下:

#include<assert.h>
#include<stdio.h>
void Magic_Square_1()
{
#define ROW 3
#define COL ROW // 等价于 #define COL 3
assert(ROW % 2 != 0);
if (ROW % 2 == 0)//ROW & ! == 1 判断是否为奇数 &(只有全为1,才为1)
{
return;
}
int preRow = 0; //记录上一个数的行坐标
int preCol = 0; //记录上一个数的列坐标
int ar[ROW][COL] = {};
ar[0][COL / 2] = 1; //先放1
preRow = 0;
preCol = COL / 2;
for (int i = 2; i <= ROW * COL; i++)//1已经放入所以从2开始进行
{
if (ar[(preRow - 1 + ROW) % ROW][(preCol + 1) % COL] == 0)
//判断上一个数的上一行下一列是否有值若没有则放入当前的i
{
preRow = (preRow - 1 + ROW) % ROW;
preCol = (preCol + 1) % COL;
}
else
{ // 若有则i放在上一个数的下一行.
preRow = (preRow + 1) % ROW;
}
ar[preRow][preCol] = i;
}
for (int i = 0; i < ROW; i++)//打印
{
for (int j = 0; j < COL; j++)
{
printf("%3d", ar[i][j]);
}
printf("\n");
}
#undef ROW //取消定义
#undef COL
}
int main()
{
  Maagic_Square_1();
  return 0;
}

  运行结果:

  

 
3.2 :偶数阶魔方阵( n = 4*K )
 偶数阶魔方阵的排列规律主要有以下两点:(以8阶为例)
   3.2.1) : 先将 1 - ROW*COL 的值按从上到下,从左到右的顺序依次填入.接着将魔方阵分成k*k (在该例 k = 2)个4阶魔方阵,并将4阶魔方阵的对角线的数值取出(如下图带颜色部分的数值).

   3.2.2) : 将取出来的值按照从大到小的顺序(或从小到大的顺序)排好,从二维数组的[0][0]下标开始(从二维数组[ROW - 1][COL - 1]的位置开始) 依次填入到数组中空白的位置.

     (重点)排列n= 4*k 阶的魔方阵的关键是取出各个四阶魔方阵的对角线的元素.我们可以发现:所有在对角线的元素的行列下标只差都满足一定的规律:

    即 |row - col|(绝对值) % 4 == 0 (对应图中黄色部分)或者(row + col)%4 == 3(对应图中红色部分) .同时我们可以定义一个数组br[ROW*ROW/2],

    在第一步填入数据时对其行列下标进行判断,若满足对角线元素下标的特点,直接将数值存放到br中,同时对二维数组相应的位置赋零值.

    这样br 中的数就是从小到大排列的,只需要在第二步时从二维数组ar 的ar[ROW - 1][COL - 1]的位置开始,

依次将数组br中的值填入到数组中空白的位置.

      代码如下:

#include<stdio.h>
#include<assert.h>
void Magic_Square_4K()
{
#define ROW 8
#define COL ROW
int ar[ROW][COL] = {};
int br[ROW*ROW/2] = {};//用来存储4阶方阵对角线元素.
int num = 1;//从1开始填入
int k = 0;//数组br下标.
for (int i = 0; i < ROW; i++)
{
for (int j = 0; j < COL; j++)
{
if((i-j)%4==3||(i-j)%4==0|| (j - i) % 4 == 3 || (j - i) % 4 == 0)
{//先控制对角线元素为零,并将应该填在对角线的这些数记录到br中.
br[k] = num;
k += 1;
ar[i][j] = 0;
}
else
{
ar[i][j] = num;
}
num++;
}
}
int tag = 0;
for (int i = ROW - 1; i >= 0; i--)//将br中的数按照顺序,从ar[ROW-1][COl-1]开始对ar中为零的元素赋值.
{
for (int j = COL - 1; j >= 0; j--)
{
if (ar[i][j] == 0)
{
ar[i][j] = br[tag];
tag += 1;
}
}
} for (int i = 0; i < ROW; i++)//打印
{
for (int j = 0; j < COL; j++)
{
printf("%4d", ar[i][j]);
}
printf("\n");
}
#undef ROW
#undef COL
}
int main()
{
Magic_Square_4K();
return 0;
}

    运行结果:

  

未完待续...

神奇的魔方阵--(MagicSquare)(1)的更多相关文章

  1. 神奇的魔方阵--(MagicSquare)(2)

    在上一篇博客中,我们讨论了阶数为奇数,以及阶数为(4K)的魔方阵的排列规则,以及代码实现(详见:https://www.cnblogs.com/1651472192-wz/p/14640903.htm ...

  2. 任意阶魔方阵(幻方)的算法及C语言实现

    写于2012.10: 本来这是谭浩强那本<C程序设计(第四版)>的一道课后习题,刚开始做得时候去网上找最优的算法,结果发现奇数和双偶数(4的倍数)的情况下算法都比较简单,但是单偶数(2的倍 ...

  3. 魔方阵算法及C语言实现

    1 魔方阵概念 填充的,每一行.每一列.对角线之和均相等的方阵,阶数n = 3,4,5….魔方阵也称为幻方阵. 例如三阶魔方阵为: 魔方阵有什么的规律呢? 魔方阵分为奇幻方和偶幻方.而偶幻方又分为是4 ...

  4. C语言——打印魔方阵(每一行,每一列,对角线之和相等)

    <一>魔方阵说明: 魔方阵是一个N*N的矩阵: 该矩阵每一行,每一列,对角线之和都相等: <二>魔方阵示例: 三阶魔方阵: 8   1   6 3   5   7 4   9 ...

  5. 【2(2N+1)魔方阵 】

    /* 2(2N+1)魔方阵 */ #include<stdio.h> #include<stdlib.h> #define N 6 #define SWAP(x, y) {in ...

  6. n阶魔方阵(奇数阵)的输出

    需求 要求输出1~n²的自然数构成的魔方阵. STEP 1 什么是魔方阵? 魔方阵,古代又称“纵横图”,是指组成元素为自然数1.2…n2的平方的n×n的方阵,其中每个元素值都不相等,且每行.每列以及主 ...

  7. 算法:九宫格问题--奇数阶魔方(Magic-Square)

    一.魔方介绍 魔方(这里是简称,也可以叫幻方.魔术矩阵,Magic Square)是 n×n 正方形网格(n 为每侧的单元数),里面每个单元格填充了不同的正整数 1, 2, 3, ... , n2,并 ...

  8. Octave入门

    Octave/Matlab Tutorial Octave/Matlab Tutorial Basic Operations 你现在已经掌握不少机器学习知识了 在这段视频中 我将教你一种编程语言 Oc ...

  9. 吴恩达-coursera-机器学习-week2

    四.多变量线性回归(Linear Regression with Multiple Variables) 4.1 多维特征 4.2 多变量梯度下降 4.3 梯度下降法实践1-特征缩放 4.4 梯度下降 ...

随机推荐

  1. cURL all in one

    cURL all in one convert http request to curl online https://curlbuilder.com/ https://cdn.xgqfrms.xyz ...

  2. TypeScript & Object Error

    TypeScript & Object Error Element implicitly has an 'any' type because expression of type 'any' ...

  3. NGK公链有发展前景吗?

    最近网络中经常能看到一个新公链项目NGK的消息,很多朋友也都私下表示过,非常看好今年的NGK.对此,小编对NGK做了一些功课,发觉到NGK未来在商业Dapp应用的发展前景,下面就给大家分享一下我的理解 ...

  4. 云原生系列5 容器化日志之EFK

    上图是EFK架构图,k8s环境下常见的日志采集方式. 日志需求 1 集中采集微服务的日志,可以根据请求id追踪到完整的日志: 2 统计请求接口的耗时,超出最长响应时间的,需要做报警,并针对性的进行调优 ...

  5. 【HTB靶场系列】靶机Carrier的渗透测试

    出品|MS08067实验室(www.ms08067.com) 本文作者:大方子(Ms08067实验室核心成员) Hack The Box是一个CTF挑战靶机平台,在线渗透测试平台.它能帮助你提升渗透测 ...

  6. 调用Config.ini类

    private static string sPath = @Directory.GetCurrentDirectory() + "\\config.ini"; [DllImpor ...

  7. 学习String源码的部分方法

    先看构造器: private final char value[]; //char类型的数组 以下均会用到 private int hash; //缓存字符串的哈希值 //以下均会用到 public ...

  8. 微信小程序(七)-项目实例(原生框架 MINA转云开发)==02-云开发-配置

    云开发:1.就是用云函数的型式来使用云存储和云数据库完成各种操作!     2.只关注调什么函数,完成什么功能即可,无需关心HTTP请求哪一套!     3.此模式不代表没有服务器,只是部署在云环境中 ...

  9. vue子组件的样式没有加scoped属性会影响父组件的样式

    scoped是一个vue的指令,用来控制组件的样式生效区域,加上scoped,样式只在当前组件内生效,不加scoped,这个节点下的样式会全局生效. 需要注意的是:一个组件的样式肯定是用来美化自己组件 ...

  10. Linux安装jdk(两种方式)

    最近在研究大数据方面的东西,业务场景是从设备采集数据经过处理然后存放DB. 建设上面的环境第一步肯定是安装jdk,所以和大家一起学一下基本知识centos7.5安装jdk1.8. 安装jdk有两种方法 ...