# RubicCube **Repository Path**: arrivezenith/RubicCube ## Basic Information - **Project Name**: RubicCube - **Description**: c#三阶魔方实现及自动还原 - **Primary Language**: C# - **License**: GPL-2.0 - **Default Branch**: master - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 11 - **Forks**: 3 - **Created**: 2019-06-17 - **Last Updated**: 2024-08-12 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README # README 魔方是一种很受欢迎的益智玩具,同时魔方运动也被称为“手指三大极限运动”之一。本篇博文将就如何使用程序实现一个三阶魔方的旋转提出我自己的思路 ## 魔方的结构 一个普通的三阶魔方具有六个面、二十六个块。其中,有六个中心块,十二个棱块,八个角块。中心块的相对位置是不会变的,无论你如何旋转魔方,六个中心块的位置始终是不会变化的。棱块处于两个面交界的位置,拥有两个颜色,因此也有两种朝向。所以一个棱块的位置可能有12×2=24种情况。角块处于三个面交界的位置,有三个颜色,因此有三种朝向。一个角块的位置可能有8×3=24种情况。 ## 思考一个问题 如何确定一个魔方?按照日常经验,只要魔方的六个面都确定了,那么这个魔方的情况就完全确定了。这的确是一种思路。我们可以考虑使用一个二维数组来存放魔方六个面的情况: ``` c++ #define WHITE 0 #define YELLOW 1 #define RED 2 #define ORANGE 3 #define BLUE 4 #define GREEN 5 int[6][9] = {0}; void init_cube() { for(int i=0; i<6; i++) { for(int j=0; j<9; j++) { int[i][j] = i; } } } ``` 用数字0代表白色,数字1代表黄色...这样我们便有了一个6*9的数组,代表每个面的每个色块分别是什么颜色,这样的一个数组就可以表示一个魔方。但是这种方式真的合适吗?仔细考虑一下,一方面,一个魔方面与面的颜色之间是存在关联的,比如,一个棱块上的两个颜色,必然不可能是两个相对面的颜色,一个角块上的三个颜色,必然是三个相邻的面的颜色。而我们分别储存六个面的颜色信息,这些数据之间就缺少了由魔方的结构所带来的约束关系,因此我们必然会写更多的代码来弥补这种缺失的关联。另一方面,魔方是一个三维空间中的结构,我们用这种方式表示一个魔方,相当于是将魔方六个面展开,降成了二维。如此一来,在处理三维的旋转时,必然会遇到困难。因此这种方式虽然很容易想到,也比较简单,但是后续的实现会相当复杂,因此是不合适的。 ## 魔方结构的数字化 综合以上分析,我们对魔方结构的表示一定要注意几个关键点: - 要以三维的形式来表示 - 不能丢失魔方结构决定的约束 - 易于处理旋转等操作 所以,我们可以考虑以魔方的块为单位,用26个方块的位置和朝向来决定一个魔方。说到位置,我们很自然地就能想到利用控件坐标系中的一组向量来表示方块的位置。首先,建立空间坐标系。我们选择以立方体的中心为原点,向右为X轴正向,向前为Y轴正向,向上为Z轴正向: ![坐标系建立][img1] 这样一来,魔方中所有方块的位置就都可以用一组向量来表示了。比如下图中,红色标明的中心块可以表示为 $\{0,1,0\}^T$,蓝色标明的棱块位置可以表示为 $\{-1,0,1\}^T$,绿色标明的角块位置可以表示为 $\{1,-1,1\}^T$。 ![坐标系示例][img2] 接下来,我们考虑一下如何表示每个方块的朝向。所谓朝向,无非就是哪个方向对应哪个颜色。因此,我们也可以使用与位置类似的方法,用一组向量来表示方块的朝向。我们用R代表红色,O代表橙色,W代表白色,Y代表红色,B代表蓝色,G代表绿色。考虑到棱块只有两个方向有颜色,中心块只有一个方向有颜色,为了统一,我们引入X,代表某个方向没有颜色。如下图所示,中心块的颜色可以表示为 $\{X,B,Z\}^T$,右边棱块的颜色可以表示为 $\{R,G,X\}^T$,右上角角块的颜色可以表示为 $\{G,Y,R\}^T$。 ![颜色坐标示例][img3] 这样,我们就可以用两个向量来表示魔方中一个块的位置和颜色。比如上图中的角块可以表示为: $$ \begin{cases} 位置:\{1,1,1\}^T\\ 颜色:\{G,Y,R\}^T \end{cases} $$ 这样,只需要26组向量,我们便可以用数据表示出一个魔方。 ## 魔方的旋转 魔方的旋转可以理解为空间中的坐标变换,而我们刚才已经可以用用向量来表示魔方的结构,因此直接使用旋转矩阵,即可完成对魔方的旋转。向量旋转公式: $$ \begin{Bmatrix} x'\\y'\\z' \end{Bmatrix} =R* \begin{Bmatrix} x\\y\\z \end{Bmatrix} $$ 绕$x,y,z$轴旋转时,$R$的值分别如下: $$ R_x= \begin{Bmatrix} 1 & 0 & 0\\ 0 & cos\theta & -sin\theta\\ 0 & sin\theta & cos\theta \end{Bmatrix} $$ $$ R_y= \begin{Bmatrix} cos\theta & 0 & sin\theta\\ 0 & 1 & 0\\ -sin\theta & 0 & cos\theta \end{Bmatrix} $$ $$ R_z= \begin{Bmatrix} cos\theta & -sin\theta & 0\\ sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \end{Bmatrix} $$ 其中$\theta$是顺旋转的角度。取$\theta=90\degree$,即为顺时针旋转90度,反之为逆时针旋转90度。 让我们来测试一下。假如将上图角块所在的顶层沿顺时针方向旋转90度,那么旋转前后该角块的数据: $$ \begin{cases} 位置:\{1,1,1\}^T\\ 颜色:\{G,Y,R\}^T \end{cases}\tag前 $$ $$ \begin{cases} 位置:\{-1,1,1\}^T\\ 颜色:\{Y,G,R\}^T \end{cases}\tag后 $$ 如果我们用旋转矩阵进行变换,得到以下结果: $$ \begin{Bmatrix} cos\theta & -sin\theta & 0\\ sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \end{Bmatrix} *\begin{Bmatrix} 1\\1\\1 \end{Bmatrix}= \begin{Bmatrix} -1\\1\\1 \end{Bmatrix} $$ $$ \begin{Bmatrix} cos\theta & -sin\theta & 0\\ sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \end{Bmatrix} *\begin{Bmatrix} G\\Y\\R \end{Bmatrix}= \begin{Bmatrix} -Y\\G\\R \end{Bmatrix} $$ 颜色向量中的符号代表颜色面向的方向与坐标轴方向相反。舍弃掉符号,运算所得的结果与魔方实际旋转后的结果完全一致,这就证明了我们的方法是正确的。这样一来,我们就解决了魔方用数据来表示这一问题,也解决了魔方的旋转。你可以顺着这个思路,尝试自己编写代码来实现。 [img1]: [img2]: [img3]: