UOJ 152 汉诺塔 分治
题意:
有三根编号为\((1, \, 2, \, 3)\)的柱子,然后第一根柱子上有编号为\(1 \sim n(n \leq 10000)\)的盘子,从上到下第\(i\)个盘子的编号是\(A_i\),其他两根柱子是空的。
你可以进行一种操作x y
,表示将第\(x\)根柱子最上面的盘子放到第\(y\)根柱子的最上面去。
输出不超过\(10^6\)次操作,使得最终所有的盘子都在同一根柱子(柱子的编号不限)上而且从上到下编号是递增的。
分析:
首先很容易想到一个\(O(n^2)\)的做法:
就是每次遇到不是\(n\)的盘子的时候就扔到柱子\(3\)上去,否则就扔到柱子\(2\)上去。
然后重复这一过程,在\(n-1\)号盘子所在的柱子上,遇到不是\(n-1\)的就扔到其他柱子上,遇到\(n-1\)就扔到柱子\(2\),也就是\(n\)号盘子的上面。
然后有一个\(O(n^{\frac{3}{2}})\)的做法。是基于平方分割的思想,每次取出\(\sqrt{n}\)个编号最大的盘子,然后暴力用\(O((\sqrt{n})^2)=O(n)\)次操作将其排序。
最后是\(O(nlogn)\)的做法:
这种做法是基于分治的思想,和快排的思路一模一样:先分解问题然后再合并问题。
我们规定从上到下递增的是顺序,从上到下递减的是逆序。
分:
对于某个柱子上乱序的区间\([l, \, r]\),如果我们要将它顺序排序,我们可以将\([l, \, mid]\)中的盘子和\([mid+1, \, r]\)中的盘子分别扔到其他两根柱子上。
然后对这两个根柱子上的盘子逆序排序。
合:
左右区间逆序排好序后然后按照从大到小的顺序再放回原来的柱子上,使得整个\([l, \, r]\)是顺序的。
对\([l, \, mid]\)和\([mid+1, \, r]\)这两个区间逆序排序,这样问题的规模就减小了一半。
相反地,如果要对一个区间逆序排序,就要先对它的左右区间顺序排序,然后合并。
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
const int maxn = 10000 + 10;
int n;
int cnt[3], t[3][maxn];
int lft[] = { 1, 0, 0 };
int rgh[] = { 2, 2, 1 };
void op(int i, int j) {
printf("%d %d\n", i + 1, j + 1);
t[j][cnt[j]] = t[i][cnt[i]-1];
cnt[j]++; cnt[i]--;
}
void solve(int cur, int l, int r, bool inv) {
if(l == r) return;
int mid = (l + r) / 2;
for(int i = 1; i <= r-l+1; i++) {
if(t[cur][cnt[cur]-1] <= mid) op(cur, lft[cur]);
else op(cur, rgh[cur]);
}
solve(lft[cur], l, mid, !inv);
solve(rgh[cur], mid+1, r, !inv);
if(!inv) {
for(int i = mid + 1; i <= r; i++) op(rgh[cur], cur);
for(int i = l; i <= mid; i++) op(lft[cur], cur);
} else {
for(int i = l; i <= mid; i++) op(lft[cur], cur);
for(int i = mid + 1; i <= r; i++) op(rgh[cur], cur);
}
}
int ans;
void dfs(int t) {
if(t == 1) return;
ans += t * 2;
dfs(t / 2);
dfs(t - t / 2);
}
int main()
{
scanf("%d", &n);
cnt[0] = n;
for(int i = n-1; i >= 0; i--) scanf("%d", &t[0][i]);
ans = 0;
dfs(n);
printf("%d\n", ans);
solve(0, 1, n, false);
return 0;
}
UOJ 152 汉诺塔 分治的更多相关文章
- [UOJ #167]【UR #11】元旦老人与汉诺塔
题目大意:给你一个有$n$个盘子的汉诺塔状态$S$,问有多少种不同的操作方法,使得可以在$m$步以内到达状态$T$.$n,m\leqslant100$ 题解:首先可以知道的是,一个状态最多可以转移到其 ...
- 算法笔记_013:汉诺塔问题(Java递归法和非递归法)
目录 1 问题描述 2 解决方案 2.1 递归法 2.2 非递归法 1 问题描述 Simulate the movement of the Towers of Hanoi Puzzle; Bonus ...
- C#递归解决汉诺塔问题(Hanoi)
using System;using System.Collections.Generic;using System.Linq;using System.Text; namespace MyExamp ...
- 数据结构0103汉诺塔&八皇后
主要是从汉诺塔及八皇后问题体会递归算法. 汉诺塔: #include <stdio.h> void move(int n, char x,char y, char z){ if(1==n) ...
- Conquer and Divide经典例子之汉诺塔问题
递归是许多经典算法的backbone, 是一种常用的高效的编程策略.简单的几行代码就能把一团遭的问题迎刃而解.这篇博客主要通过解决汉诺塔问题来理解递归的精髓. 汉诺塔问题简介: 在印度,有这么一个古老 ...
- 几年前做家教写的C教程(之四专讲了指针与汉诺塔问题)
C语言学习宝典(4) 指针:可以有效的表示复杂的数据结构,能动态的分配动态空间,方便的使用字符串,有效的使用数组,能直接处理内存单元 不掌握指针就没有掌握C语言的精华 地址:系统为每一个变量分配一个内 ...
- python实现汉诺塔
经典递归算法汉诺塔分析: 当A柱子只有1个盘子,直接A --> C 当A柱子上有3个盘子,A上第一个盘子 --> B, A上最后一个盘子 --> C, B上所有盘子(1个) --&g ...
- fzu1036四塔问题(汉诺塔问题拓展)
#include<iostream> #include<cstdio> #include<cmath> using namespace std; ]; int ru ...
- 1019: [SHOI2008]汉诺塔
1019: [SHOI2008]汉诺塔 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 1495 Solved: 916[Submit][Status] ...
随机推荐
- webpack.config.js====CSS相关:css和scss配置loader
1. 安装: //loader加载器加载css和sass模块 cnpm install style-loader css-loader node-sass sass-loader --save-dev ...
- ABAP常用事务码
开发----------------------------------------------- SE51 屏幕制作 SE91 MESSAGE OBJECT SE80 对象浏览器(可以建开发类 ...
- ArrayList集合--关于System.out.print(Object obj);的理解
1.ArrayList集合中常用的方法 ArrayList<Student> stuArrayList = new ArrayList<>(); //定义一个集合对象 stuA ...
- elasticsearch 2.4 windows版jvm内存设置
本文编写目的是因为网上有很多es修改内存配置的文章,方法也各有不同,但在我的情况下(es 2.4 windows版)发现很多方法都是无效的,有效只有以下方法 第一个是xms,第二个是xmx
- Android 4.4及以后将内容布局延伸到状态栏
首先说明:该文章不是大家说的沉浸式状态栏,网上沉浸式状态栏的博客很多,搜索就有了! 该篇博客的主要目的就是为了将图片显示在状态栏上,让APP看起来更有型!如下图所示: 界面 这个界面的布局就是co ...
- ios 设置导航栏背景色
//设置导航栏背景色 如果上面的不好用 就用下面的 [self.navigationController.navigationBar setBackgroundImage:[UIImage image ...
- Leet-code144. Binary Tree Preorder Traversal
这是一道将二叉树先序遍历,题目不难. 首先采用深搜递归 /** * Definition for a binary tree node. * public class TreeNode { * int ...
- python中的构造函数和构造函数和析构函数的作用
构造函数和构造函数和析构函数都属于python中的特殊方法 其中的“__del__”就是一个析构函数了,当使用del 删除对象时,会调用他本身的析构函数,另外当对象在某个作用域中调用完毕,在跳出其作用 ...
- HTML 中的特殊字符
空格符 <小于号 < >大于号 > &和好 & ¥人民币 ¥ © 版权 © ® 注册商标 ® ℃ 摄氏度 ° ...
- 洛谷 P3353 在你窗外闪耀的星星
题目描述 飞逝的的时光不会模糊我对你的记忆.难以相信从我第一次见到你以来已经过去了3年.我仍然还生动地记得,3年前,在美丽的集美中学,从我看到你微笑着走出教室,你将头向后仰,柔和的晚霞照耀着你玫瑰色的 ...