神奇立方体

使用Base64(RFC 2045)的64个字符,每格一个字符,填满64*64*64格的立方体,要求:
【1】与x轴平行的每条线(64格)内的各字符均不同
【2】与y轴平行的每条线(64格)内的各字符均不同
【3】与z轴平行的每条线(64格)内的各字符均不同
【4】与x轴垂直的每个平面的64个(8*8格)小平面中,每个小平面内的各字符均不同
【5】与y轴垂直的每个平面的64个(8*8格)小平面中,每个小平面内的各字符均不同
【6】与z轴垂直的每个平面的64个(8*8格)小平面中,每个小平面内的各字符均不同
【7】4096个(4*4*4格)小立方体中,每个小立方体内的各字符均不同
以下为源程序 cubic.c :

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
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
// Copyright © 2022, 飞麦 <fitmap@qq.com>, All rights reserved.

#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>

#define LEN 64 // 立方体边长(格数, 每格一字符, 填满需要262144字符)
#define BLOCK_NUM (LEN * LEN) // 独立线(1*64)、面(8*8)、体(4*4*4)的个数,线互不重叠、面互不重叠、体互不重叠
#define PUZZLE_NUM (LEN * LEN * LEN) // 立方体所有格子的数量
#define TYPE_NUM 9 // 块的类型数: 与x/y/z轴平行线(3种)、与x/y/z轴垂直面(3种)、体(3种)
const char* CHAR_A = "ABCDEFGHIJKLMNOPQRSTUVWXYZ01234+abcdefghijklmnopqrstuvwxyz56789/"; // 填格之Base64字符
int32_t puzzle_no; // 立方体的一维下标(z*64*64 + y*64 + x)
int8_t x, y, z; // x轴坐标(0~63), y轴坐标(0~63), z轴坐标(0~63)
char puzzle_a[PUZZLE_NUM]; // 立方体所有字符(转为一维表示)
int8_t select_a[PUZZLE_NUM]; // 每个格子的待选择字符在 CHAR_A 中的序号(0~63)
int64_t occupy_a_a[BLOCK_NUM][TYPE_NUM]; // 每个(3种线、3种面、3种体)块的64字符占用情况(每bit代表一个字符)

// 根据立方体中的当前字符位置,生成各轴坐标
void gen_puzzle_coordinate() {
  z = (puzzle_no >> 12) & 0x3f; // (puzzle_no / BLOCK_NUM) % LEN
  y = (puzzle_no >> 6) & 0x3f; // (puzzle_no / LEN) % LEN
  x = puzzle_no & 0x3f; // puzzle_no % LEN
}

// 设置各初始值
void init() {
  for (puzzle_no = 0; puzzle_no < PUZZLE_NUM; puzzle_no++) {
    select_a[puzzle_no] = 0; // 立方体中每格在 CHAR_A 中的候选字符下标
  }
  for (int16_t block_no = 0; block_no < BLOCK_NUM; block_no++) {
    int64_t* occupy_a = occupy_a_a[block_no];
    for (int8_t type_no = 0; type_no < TYPE_NUM; type_no++) {
      occupy_a[type_no] = 0LL; // 初始化线、面、体的64字符占用情况, LL表示64位整数字面量
    }
  }
  puzzle_no = 0; // 立方体中字符索引
  gen_puzzle_coordinate(); // 根据 puzzle_no 生成 x轴坐标, y轴坐标, z轴坐标
}

// 根据坐标计算当前格所属线、面、体的块位置
int16_t calc_block_no(int8_t type_no) {
  switch (type_no) {
    case 0: // 当前坐标所属的与x轴平行的(1格*64格)线的位置: (z * 64) + y
      return (z << 6) | y;
    case 1: // 当前坐标所属的与y轴平行的(1格*64格)线的位置: (x * 64) + z
      return (x << 6) | z;
    case 2: // 当前坐标所属的与z轴平行的(1格*64格)线的位置: (y * 64) + x
      return (y << 6) | x;
    case 3: // 当前坐标所属的与x轴垂直的(8格*8格)面的位置: (x * 64) + ((z / 8) * 8) + (y / 8)
      return (x << 6) | (z & 0x38) | (y >> 3);
    case 4: // 当前坐标所属的与y轴垂直的(8格*8格)面的位置: (y * 64) + ((x / 8) * 8) + (z / 8)
      return (y << 6) | (x & 0x38) | (z >> 3);
    case 5: // 当前坐标所属的与z轴垂直的(8格*8格)面的位置: (z * 64) + ((y / 8) * 8) + (x / 8)
      return (z << 6) | (y & 0x38) | (x >> 3);
    case 6: // 当前坐标所属的(4格*4格*4格)体的位置: ((z / 4) * 256) + ((y / 4) * 16) + (x / 4)
      return ((z >> 2) << 8) | ((y >> 2) << 4) | (x >> 2);
    case 7: // 当前坐标所属的均匀分布的8个(2格*2格*2格)体的位置: ((z / 2 % 16) * 256) + ((y / 2 % 16) * 16) + (x / 2 % 16)
      return ((z >> 1 & 0x0f) << 8) | ((y >> 1 & 0x0f) << 4) | (x >> 1 & 0x0f);
    default: // 当前坐标所属的均匀分布的64个(1格*1格*1格)体的位置: ((z % 16) * 256) + ((y % 16) * 16) + (x % 16)
      return ((z & 0x0f) << 8) | ((y & 0x0f) << 4) | (x & 0x0f);
  }
}

// 根据类型获取当前格所属线、面、体的块
int64_t* got_block(int8_t type_no) {
  return occupy_a_a[calc_block_no(type_no)];
}

// 计算字符在 CHAR_A 中的索引
int8_t calc_occupy_bit_loc(char ch) {
  if (ch >= 'A' && ch <= 'Z') {
    return ch - 'A';
  }
  if (ch >= 'a' && ch <= 'z') {
    return 26 + ch - 'a';
  }
  if (ch >= '0' && ch <= '9') {
    return 52 + ch - '0';
  }
  if (ch == '+') {
    return 62;
  }
  return 63; // ch == '/'
}

// 计算字符在 CHAR_A 中的索引对应的二进制位
int64_t calc_occupy_bits(char ch) {
  return 1LL << calc_occupy_bit_loc(ch); // LL表示64位整数字面量
}

// 标记指定字符在所属线、面、体已被占用
void tag_occupy_one(int64_t bits, int64_t* occupy_a, int8_t type_no) {
  occupy_a[type_no] |= bits;
}

// 标记指定字符在所属线、面、体中未被占用
void untag_occupy_one(int64_t bits, int64_t* occupy_a, int8_t type_no) {
  occupy_a[type_no] &= ~bits;
}

// 判断指定位置的占位情况中的指定字符是否未被占用
bool one_occupy_free(int64_t bits, int64_t* occupy_a, int8_t type_no) {
  return (occupy_a[type_no] & bits) == 0LL; // LL表示64位整数字面量
}

// 标记立方体中当前字符的所属线、面、体中指定字符已被占用
void tag_occupy_all(char ch) {
  int64_t bits = calc_occupy_bits(ch);
  for (int8_t type_no = 0; type_no < TYPE_NUM; type_no++) {
    tag_occupy_one(bits, got_block(type_no), type_no);
  }
}

// 清除原字符的所属线、面、体的占位情况中该字符已被占用信息
void untag_occupy_all() {
  int64_t bits = calc_occupy_bits(puzzle_a[puzzle_no]);
  for (int8_t type_no = 0; type_no < TYPE_NUM; type_no++) {
    untag_occupy_one(bits, got_block(type_no), type_no);
  }
}

// 判断立方体中当前字符的所属线、面、体的占位情况中该字符是否均未被占用
bool all_occupy_free(char ch) {
  int64_t bits = calc_occupy_bits(ch);
  for (int8_t type_no = 0; type_no < TYPE_NUM; type_no++) {
    if (!one_occupy_free(bits, got_block(type_no), type_no)) {
      return false;
    }
  }
  return true;
}

// 寻找当前位置未被各所属线、面、体占用的字符并填入,同时更新占用情况
bool pick_free_and_tag_all() {
  int8_t select = select_a[puzzle_no];
  char ch;
  while (select < LEN) {
    ch = CHAR_A[select];
    if (all_occupy_free(ch)) {
      break;
    }
    select++;
  }
  if (select >= LEN) {
    select_a[puzzle_no] = 0;
    return false; // 若所有字符均已被各所属线、面、体占用,则返回当前位置失败,以便后续回退位置
  }
  puzzle_a[puzzle_no] = ch;
  select_a[puzzle_no] = select + 1;
  tag_occupy_all(ch);
  return true;
}

// 改变立方体中的当前字符位置, 并重新计算各轴坐标
void add_puzzle_no(int8_t change) {
  puzzle_no += change;
  if (puzzle_no >= 0 && puzzle_no < PUZZLE_NUM) {
    gen_puzzle_coordinate();
  }
}

// 挑选候选字符,若无可用字符则回退字符
void deal() {
  if (pick_free_and_tag_all()) {
    add_puzzle_no(1);
  } else { // 本程序情况下不会出现下列回退
    add_puzzle_no(-1);
    if (puzzle_no > 0) {
      untag_occupy_all();
    }
  }
}

// 求解每1*64线、每8*8面、每4*4*4体各Base64字符均不同的64*64*64立方体
bool solve() {
  while (puzzle_no >= 0 && puzzle_no < PUZZLE_NUM) {
    deal();
  }
  return puzzle_no >= 0; // 是否求解成功
}

// 输出立方体, 每行后换行, 每面后加一空行{所有面均恰巧为中心对称[z][y][x]==[z][63-y][63-x]}
void output() {
  for (z = 0; z < LEN; z++) { // 每面
    for (y = 0; y < LEN; y++) { // 每行
      for (x = 0; x < LEN; x++) { // 每列
        printf("%c", puzzle_a[z << 12 | y << 6 | x]);
      }
      printf("\n");
    }
    printf("\n");
  }
}

void main() {
  init(); // 初始化
  if (solve()) { // 生成立方体所有字符
    output(); // 输出立方体所有字符
  }
}
posted @ 2022-10-11 21:10  飞麦  阅读(104)  评论(0编辑  收藏  举报