魔方状态

题目描述

二阶魔方就是只有2层的魔方,只由8个小块组成。

如图p1.png所示。

小明很淘气,他只喜欢3种颜色,所有把家里的二阶魔方重新涂了颜色,如下:

前面:橙色

右面:绿色

上面:黄色

左面:绿色

下面:橙色

后面:黄色

请你计算一下,这样的魔方被打乱后,一共有多少种不同的状态。

如果两个状态经过魔方的整体旋转后,各个面的颜色都一致,则认为是同一状态。

请提交表示状态数的整数,不要填写任何多余内容或说明文字。

开始拿到这道题没什么思路,笔算算不来,模拟判重感觉太麻烦。大神说burnside引理可以做,学渣表示看不懂。。网上基本没有求解的,有的也答案不一。最后还是模拟判重这么写了。

我的答案:229878

测试:全同色魔方状态为1,正确。正常二阶魔方状态3674160,正确。

思路:其实就是空间状态搜索。模拟操作+判重。关于操作,二阶魔方只做U(顶层顺时针) R(右层顺时针) F(前层顺时针)就可以得到所有状态了。判重需要旋转整个魔方去比较。(判重小白现在只会用set)。

然后是,怎么去表示一个二阶魔方。二阶魔方8个块,一个块6面(看不见的作黑色考虑),所以我用了char st[8][7]去表示一个魔方。块的顺序如下:

上面的初始状态表示就是{{“oybbgb”},{“oygbbb”},{“bygbby”},{“bybbgy”},{“obbogb”},{“obgobb”},{“bbgoby”},{“bbbogy”}}

o表示橙色,b表示黑色,g表示绿色,y表示黄色。

对于一个小块,6个面的颜色定义顺序如下:

所以,比如说,上面题目给的魔方,前面一层,左上角的橙黄绿块,表示就是oybbgb

博主还是个小白,只能找来C++的代码,还望 Java大佬及时写出

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. typedef char st[8][7];
  4. st state[2000000];
  5. set<string> all;
  6. st begin={{"oybbgb"},{"oygbbb"},{"bygbby"},{"bybbgy"},{"obbogb"},{"obgobb"},{"bbgoby"},{"bbbogy"}};
  7. //st begin={{"oooooo"},{"oooooo"},{"oooooo"},{"oooooo"},{"oooooo"},{"oooooo"},{"oooooo"},{"oooooo"}};
  8. //只有一个颜色的魔方 ans=1
  9. //st begin={{"rykkbk"},{"rygkkk"},{"kygkko"},{"kykkbo"},{"rkkwbk"},{"rkgwkk"},{"kkgwko"},{"kkkwbo"}};
  10. //正常2阶魔方状态 r红 y黄 b蓝 g绿 w白 o橙 k黑(红对橙,白对黄,蓝对绿,颜色相近的相对)这里白为底 前为红
  11. //需要将state大小改为4000000
  12. //这个测试用例跑了20分钟左右 560M内存 ans=3674160 与实际二阶魔方状态数相同 见下截图
  13. int front, tail;
  14. void ucell(char *a){swap(a[0], a[2]); swap(a[2], a[5]); swap(a[5], a[4]);}
  15. void rcell(char *a){swap(a[1], a[0]); swap(a[0], a[3]); swap(a[3], a[5]);}
  16. void fcell(char *a){swap(a[2], a[1]); swap(a[1], a[4]); swap(a[4], a[3]);}
  17. void u(st &s)//顶层顺时针旋转
  18. {
  19. ucell(s[0]);
  20. ucell(s[1]);
  21. ucell(s[2]);
  22. ucell(s[3]);
  23. swap(s[1], s[0]);
  24. swap(s[2], s[1]);
  25. swap(s[3], s[2]);
  26. }
  27. void uwhole(st &s)//整个魔方从顶部看 顺时针转 用于判重
  28. {
  29. u(s);
  30. ucell(s[4]);
  31. ucell(s[5]);
  32. ucell(s[6]);
  33. ucell(s[7]);
  34. swap(s[5], s[4]);
  35. swap(s[6], s[5]);
  36. swap(s[7], s[6]);
  37. }
  38. void f(st &s)//前面一层 顺时针转
  39. {
  40. fcell(s[0]);
  41. fcell(s[1]);
  42. fcell(s[4]);
  43. fcell(s[5]);
  44. swap(s[1], s[5]);
  45. swap(s[0], s[1]);
  46. swap(s[4], s[0]);
  47. }
  48. void fwhole(st &s)//整个魔方从前面看 顺时针转 用于判重
  49. {
  50. f(s);
  51. fcell(s[2]);
  52. fcell(s[6]);
  53. fcell(s[7]);
  54. fcell(s[3]);
  55. swap(s[2], s[6]);
  56. swap(s[3], s[2]);
  57. swap(s[7], s[3]);
  58. }
  59. void r(st &s)//魔方右层顺时针转
  60. {
  61. rcell(s[1]);
  62. rcell(s[2]);
  63. rcell(s[6]);
  64. rcell(s[5]);
  65. swap(s[2], s[1]);
  66. swap(s[5], s[1]);
  67. swap(s[6], s[5]);
  68. }
  69. void rwhole(st &s)//整个魔方从右边看 顺时针转 用于判重
  70. {
  71. r(s);
  72. rcell(s[0]);
  73. rcell(s[3]);
  74. rcell(s[4]);
  75. rcell(s[7]);
  76. swap(s[3], s[7]);
  77. swap(s[0], s[3]);
  78. swap(s[4], s[0]);
  79. }
  80. string convert(st &s)//魔方状态二维字符数组 转化为string
  81. {
  82. string ss;
  83. for(int i=0; i<8; i++)ss+=s[i];
  84. return ss;
  85. }
  86. bool try_to_insert(int tail)//判重
  87. {
  88. st k;
  89. memcpy((void*)k, (void*)state[tail], sizeof(state[tail]));
  90. for(int i=0; i<4; i++)
  91. {
  92. fwhole(k);
  93. for(int j=0; j<4; j++)
  94. {
  95. uwhole(k);
  96. for(int q=0; q<4; q++)
  97. {
  98. rwhole(k);
  99. if(all.count(convert(k))==1)
  100. {
  101. return false;
  102. }
  103. }
  104. }
  105. }
  106. all.insert(convert(k));
  107. return true;
  108. }
  109. int main()
  110. {
  111. front=0,tail=1;
  112. all.insert(convert(begin));
  113. memcpy((void*)state[0],(void*)begin,sizeof(begin));
  114. while(front!=tail)
  115. {
  116. //对当前状态分别模拟三种操作U R F 然后判重
  117. for(int i=0; i<3; i++)
  118. {
  119. memcpy((void*)state[tail], (void*)state[front], sizeof(state[front]));
  120. if(i==0)
  121. {
  122. u(state[tail]);
  123. if(try_to_insert(tail))tail++;
  124. }
  125. else if(i==1)
  126. {
  127. r(state[tail]);
  128. if(try_to_insert(tail))tail++;
  129. }
  130. else if(i==2)
  131. {
  132. f(state[tail]);
  133. if(try_to_insert(tail))tail++;
  134. }
  135. }
  136. front++;
  137. }
  138. cout<<front<<endl;
  139. return 0;
  140. }
  141. //ans 229878

Java实现第八届蓝桥杯魔方状态的更多相关文章

  1. Java实现第八届蓝桥杯购物单

    购物单 题目描述 小明刚刚找到工作,老板人很好,只是老板夫人很爱购物.老板忙的时候经常让小明帮忙到商场代为购物.小明很厌烦,但又不好推辞. 这不,XX大促销又来了!老板夫人开出了长长的购物单,都是有打 ...

  2. Java实现第八届蓝桥杯图形排版

    标题:图形排版 小明需要在一篇文档中加入 N 张图片,其中第 i 张图片的宽度是 Wi,高度是 Hi. 假设纸张的宽度是 M,小明使用的文档编辑工具会用以下方式对图片进行自动排版: 1. 该工具会按照 ...

  3. java实现第八届蓝桥杯生命游戏

    生命游戏 题目描述 康威生命游戏是英国数学家约翰·何顿·康威在1970年发明的细胞自动机. 这个游戏在一个无限大的2D网格上进行. 初始时,每个小方格中居住着一个活着或死了的细胞. 下一时刻每个细胞的 ...

  4. Java实现第八届蓝桥杯国赛 数字划分

    标题:数字划分 w星球的长老交给小明一个任务: 1,2,3-16 这16个数字分为两组. 要求: 这两组数字的和相同, 并且,两组数字的平方和也相同, 并且,两组数字的立方和也相同. 请你利用计算机的 ...

  5. Java实现第八届蓝桥杯青蛙跳杯子

    青蛙跳杯子 题目描述 X星球的流行宠物是青蛙,一般有两种颜色:白色和黑色. X星球的居民喜欢把它们放在一排茶杯里,这样可以观察它们跳来跳去. 如下图,有一排杯子,左边的一个是空着的,右边的杯子,每个里 ...

  6. Java实现第八届蓝桥杯兴趣小组

    兴趣小组 为丰富同学们的业余文化生活,某高校学生会创办了3个兴趣小组 (以下称A组,B组,C组). 每个小组的学生名单分别在[A.txt],[B.txt]和[C.txt]中. 每个文件中存储的是学生的 ...

  7. Java实现第八届蓝桥杯外星日历

    外星日历 题目描述 某星系深处发现了文明遗迹. 他们的计数也是用十进制. 他们的文明也有日历.日历只有天数,没有年.月的概念. 有趣的是,他们也使用了类似"星期"的概念, 只不过他 ...

  8. Java实现第八届蓝桥杯正则问题

    正则问题 考虑一种简单的正则表达式: 只由 x ( ) | 组成的正则表达式. 小明想求出这个正则表达式能接受的最长字符串的长度. 例如 ((xx|xxx)x|(x|xx))xx 能接受的最长字符串是 ...

  9. Java实现第八届蓝桥杯9算数式

    9算数式 题目描述 观察如下的算式: 9213 x 85674 = 789314562 左边的乘数和被乘数正好用到了1~9的所有数字,每个1次. 而乘积恰好也是用到了1~9的所有数字,并且每个1次. ...

随机推荐

  1. js canvas压缩图片和jQuery ajax上传图片简单demo

    原来用的插件,里面东西太乱了,一会jq,一会原生js,本来原生js就不熟,看起来更难受,而且感觉好多东西都是没用的,而且后端php转存文件一直不是很熟悉,正好一起整理一下.就是很简单的一个demo,如 ...

  2. 深入理解Java虚拟机第三版,总结笔记【随时更新】

    最近一直在看<深入理解Java虚拟机>第三版,无意中发现了第三版是最近才发行的,听说讲解的JDK版本升级,新增了近50%的内容. 这种神书,看懂了,看进去了,真的看的很快,并没有想象中的晦 ...

  3. 谈谈Java常用类库中的设计模式 - Part Ⅰ

    背景 最近一口气看完了Joshua Bloch大神的Effective Java(下文简称EJ).书中以tips的形式罗列了Java开发中的最佳实践,每个tip都将其意图和要点压缩在了标题里,这种做法 ...

  4. Java POI 读取Excel数据转换为XML格式

    1.首先要下载poi相关的包:http://poi.apache.org/  ,以下是所需的jar包 2.贴上详细的代码 public class ExcelToXml { /** * 将excel的 ...

  5. WebApiClient性能参考

    1 文章目的 昨天写了走进WebApiClientCore的设计,介绍了WebApiClient的变化与设计之后,收到大家支持的.赞许的,还有好的建议和顾虑,比如WebApiClient性能怎么样,有 ...

  6. Django ORM性能优化之count和len方法的选择(非常详细推荐干货)

    接下来我将从源码层面分情况和应用分析我们在计算queryset数据集时是用orm的count函数计算长度还是用len函数计算数据集长度. 首先,我们知道ORM查询queryset数据集是惰性查询的,只 ...

  7. .NET 合并程序集(将 dll 合并到 exe 中)

    ------------恢复内容开始------------ ------------恢复内容开始------------ 背景:我们的应用程序通常都是由多个程序集组成,例如一个 exe 程序依赖于多 ...

  8. Layui 解决动态图标不动的问题

    <i class="layui-icon layui-icon-face-smile" style="color: red; font-size: 100px;&q ...

  9. Java学习大纲-0412更新

    非科班报培训班学习Java,从博客园,知乎,CNDS上搜了一圈,暂时按以下计划执行,有问题随时更新--0412 1.培训班的课程按时按点学习完成(毕竟掏钱在学的是不,不好好听亏不亏...) keys: ...

  10. linux高级应用第九章-正则表达式

    笔记部分 基础正则表达式: ^   第1个符号 ,以什么什么开头   ^m $  第2个符号,以什么什么结尾  m$    ,还表示空行,或空格,可以用cat  -An 试一下 ^$ 第3个符号,空行 ...