USACO Section2.1 Sorting a Three-Valued Sequence 解题报告
sort3解题报告 —— icedream61 博客园(转载请注明出处)
------------------------------------------------------------------------------------------------------------------------------------------------
【题目】
给你N,而后给出N个数,每个数都是1~3中的一个。请问,要把这个数列升序排好,最少需要进行几次两两交换?
【数据范围】
1<=N<=1000
【输入样例】
9
2
2
1
3
3
3
2
3
1
【输出样例】
4
------------------------------------------------------------------------------------------------------------------------------------------------
【分析】
模拟一遍最优策略,便可得出答案,具体做法如下:
1、读入N,并读入N个数d[1]~d[N],定义最少所需交换次数s=0
2、统计下1~3分别的个数,得到1的个数num[1]和1、2的总个数num[2]。此时,便可确定最终1~3的位置分别是:
1: 1~num[1]
2: num[1]+1~num[2]
3: num[2]+1~N
3、第一轮交换,使所有的1都到位:
i=1~num[1],找!=1的位置,不存在则不妨令i==num[1]+1
当i==num[1]+1则说明所有1均到位,本轮结束。否则,
j=num[1]+1~num[2],找==1的位置,不存在则不妨令j==num[2]+1
k=num[2]+1~N,找==1的位置,不存在则不妨令j==N+1
能到这里说明需要交换一次,故而++S,
对当前i,首先争取一步换到位(即d[i]==2与d[j]换,或者d[i]==3与d[k]换),如不可能则找一个合法的换便可,不可能出现没有合法的换的情况。
4、第二轮交换,使所有的2都到位(3自然也就都到位了):
i=num[1]+1~num[2],找!=2的位置,不存在则不妨令i==num[2]+1
当i==num[2]+1则说明所有2均到位,本轮结束。否则,
j=num[2]+1~N,找==2的位置,不存在则不妨令j==N+1
能到这里说明需要交换一次,故而++S,
交换d[i]与d[j]。
5、至此,两轮交换完成,所得s即为最少所需交换次数。
------------------------------------------------------------------------------------------------------------------------------------------------
【总结】
开始没有注意到随便交换并非最优,没考虑到的就是分析中第3步的情况:在第一轮中应当先争取一步到位,这样可以减少一次交换次数。
本题还有个需要注意的地方(快排、二分等方法中也常出现这个问题),为了让下标不超出范围,要随时判断是否越界。
------------------------------------------------------------------------------------------------------------------------------------------------
【代码】
/*
ID: icedrea1
PROB: sort3
LANG: C++
*/ #include <iostream>
#include <fstream>
using namespace std; int N,d[];
int num[]; int main()
{
ifstream in("sort3.in");
ofstream out("sort3.out"); in>>N;
for(int i=;i<=N;++i) { in>>d[i]; ++num[d[i]]; }
num[]+=num[]; // The situation of vector d:
// 1: d[1]~d[num[1]]
// 2: d[num[1]+1]~d[num[2]]
// 3: d[num[2]+1]~d[N] int s=;
for(int i=,j=num[]+,k=num[]+;;)
{
while(i<=num[] && d[i]==) ++i;
if(i==num[]+) break;
while(j<=num[] && d[j]!=) ++j;
while(k<=N && d[k]!=) ++k;
++s;
if(d[i]== && j<=num[]) swap(d[i],d[j]); else if(d[i]== && k<=N) swap(d[i],d[k]); // 一步换到位更优
else if(j<=num[]) swap(d[i],d[j]); else swap(d[i],d[k]); // 无法一步到位,那就换个合法的就行
}
for(int i=num[]+,j=num[]+;;)
{
while(i<=num[] && d[i]==) ++i;
if(i==num[]+) break;
while(j<=N && d[j]!=) ++j;
swap(d[i],d[j]); ++s;
} out<<s<<endl; in.close();
out.close();
return ;
}
USACO Section2.1 Sorting a Three-Valued Sequence 解题报告的更多相关文章
- USACO Section 2.1 Sorting a Three-Valued Sequence 解题报告
题目 题目描述 给N个整数,每个整数只能是1,2,或3.现在需要对这个整数序列进行从小到大排序,问最少需要进行几次交换.N(1 <= N <= 1000) 样例输入 9 2 2 1 3 3 ...
- timus 1175. Strange Sequence 解题报告
1.题目描述: 1175. Strange Sequence Time limit: 1.0 secondMemory limit: 2 MB You have been asked to disco ...
- USACO Section 1.1 Your Ride Is Here 解题报告
题目 问题描述 将字符串转变为数字,字母A对应的值为1,依次对应,字母Z对应的值为26.现在有一个字符串,将其中的每个字符转变为数字之后进行累乘,最终的结果对47求余数. 题目给你两个字符串,其中的字 ...
- LeetCode: Permutation Sequence 解题报告
Permutation Sequence https://oj.leetcode.com/problems/permutation-sequence/ The set [1,2,3,…,n] cont ...
- LeetCode: Longest Consecutive Sequence 解题报告
Longest Consecutive Sequence Given an unsorted array of integers, find the length of the longest con ...
- BZOJ 1367 [Baltic2004]sequence 解题报告
BZOJ 1367 [Baltic2004]sequence Description 给定一个序列\(t_1,t_2,\dots,t_N\),求一个递增序列\(z_1<z_2<\dots& ...
- Winter-1-F Number Sequence 解题报告及测试数据
Time Limit:1000MS Memory Limit:32768KB Description A number sequence is defined as follows:f(1) ...
- 洛谷 P4597 序列sequence 解题报告
P4597 序列sequence 题目背景 原题\(\tt{cf13c}\)数据加强版 题目描述 给定一个序列,每次操作可以把某个数\(+1\)或\(-1\).要求把序列变成非降数列.而且要求修改后的 ...
- gzhu 2013 Good Sequence 解题报告
题目链接:(这个是内网的网址) http://172.22.27.1/problem?pid=1013 Good Sequence Time Limit: 4000/2000 MS (Java/Ot ...
随机推荐
- C++ POD类型
POD( Plain Old Data)概念: Arithmetic types (3.9.1), enumeration types, pointer types, and pointer to m ...
- 22个必须知道的css技巧
1.改变选中文字的背景和颜色 ::selection{ /* Safari and Opera */ background:#c3effd; color:#000; } ::-moz-selectio ...
- Radix Sort
为了完成二维数据快速分类,最先使用的是hash分类. 前几天我突然想,既然基数排序的时间复杂度也不高,而且可能比hash分类更稳定,所以不妨试一下. 在实现上我依次实现: 1.一维数组基数排序 基本解 ...
- POJ-1195 Mobile phones---裸的二维树状数组(注意下标从1,1开始)
题目链接: https://vjudge.net/problem/POJ-1195 题目大意: 直接维护二维树状数组 注意横纵坐标全部需要加1,因为树状数组从(1,1)开始 #include<c ...
- Jmeter文件目录,功能简介
1.Jmeter文件目录:1)bin文件: Jmeter启动:bin/jmeter.bat Jmeter日志文件:jmeter.log Linux的启动文件:Jmeter.sh Jmeter系统配置文 ...
- 2017.11.17 C++系列---用malloc动态给c++二维数组的申请与释放操作
方法一:利用二级指针申请一个二维数组. #include<stdio.h> #include<stdlib.h> int main() { int **a; //用二级指针动态 ...
- 关于requireJS的同步加载和异步加载
这篇随笔主要记录require('name')和require(['name1','name2'])在同步和异步加载使用的区别 1.require('name')同步加载模块的形式 define(fu ...
- IDEA的使用方法(一)(IDEA基本快捷键)
一个软件的快捷键显得尤为重要,接下来来讲讲快捷键 CTR+N 搜索类 CTR+SHIT+N 搜索文件 CTR+ALT+空格 代码提示(类似于 ALT+/) ALT+F7 查询在某处使用 CTR+Q 查 ...
- JS之执行上下文
执行上下文(execution context),是JS中的一个很重要的概念.它对于我们理解函数定义,执行时都做了什么有着很大的意义.理解它我们才能明白我们常说的函数声明提升,作用域链,闭包等原理. ...
- Python系列6之面向对象
目录 生成器和迭代器 字符串格式化 内置函数vars 反射 面向对象编程 一. 生成器和迭代器 1. 生成器 生成器具有一种生成的能力,它仅仅代表着一种生成的能力,当我们需要使用的时候,才会通过迭代 ...