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 解题报告的更多相关文章

  1. USACO Section 2.1 Sorting a Three-Valued Sequence 解题报告

    题目 题目描述 给N个整数,每个整数只能是1,2,或3.现在需要对这个整数序列进行从小到大排序,问最少需要进行几次交换.N(1 <= N <= 1000) 样例输入 9 2 2 1 3 3 ...

  2. timus 1175. Strange Sequence 解题报告

    1.题目描述: 1175. Strange Sequence Time limit: 1.0 secondMemory limit: 2 MB You have been asked to disco ...

  3. USACO Section 1.1 Your Ride Is Here 解题报告

    题目 问题描述 将字符串转变为数字,字母A对应的值为1,依次对应,字母Z对应的值为26.现在有一个字符串,将其中的每个字符转变为数字之后进行累乘,最终的结果对47求余数. 题目给你两个字符串,其中的字 ...

  4. LeetCode: Permutation Sequence 解题报告

    Permutation Sequence https://oj.leetcode.com/problems/permutation-sequence/ The set [1,2,3,…,n] cont ...

  5. LeetCode: Longest Consecutive Sequence 解题报告

    Longest Consecutive Sequence Given an unsorted array of integers, find the length of the longest con ...

  6. BZOJ 1367 [Baltic2004]sequence 解题报告

    BZOJ 1367 [Baltic2004]sequence Description 给定一个序列\(t_1,t_2,\dots,t_N\),求一个递增序列\(z_1<z_2<\dots& ...

  7. Winter-1-F Number Sequence 解题报告及测试数据

    Time Limit:1000MS     Memory Limit:32768KB Description ​A number sequence is defined as follows:f(1) ...

  8. 洛谷 P4597 序列sequence 解题报告

    P4597 序列sequence 题目背景 原题\(\tt{cf13c}\)数据加强版 题目描述 给定一个序列,每次操作可以把某个数\(+1\)或\(-1\).要求把序列变成非降数列.而且要求修改后的 ...

  9. gzhu 2013 Good Sequence 解题报告

    题目链接:(这个是内网的网址)  http://172.22.27.1/problem?pid=1013 Good Sequence Time Limit: 4000/2000 MS (Java/Ot ...

随机推荐

  1. 0 - python简介

    Python介绍 python的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏姆为了在阿姆斯特丹打发时间,决心开发一个新的脚本解释程序,作为ABC语言 ...

  2. 快速提取邮箱地址(利用word或网站)

    在word中,CTRL+F,输入:[A-z,0-9]{1,}\@[A-z,0-9,\.]{1,} 点击“高级”,勾选“使用通配符”,点击“查找全部”: 复制.粘贴. 还可通过以下页面在线提取. htt ...

  3. IOS 多线程-NSThread 和线程状态

    @interface HMViewController () - (IBAction)btnClick; @end @implementation HMViewController - (void)v ...

  4. IOS 多线程-pthread

    #import <pthread.h> @interface HMViewController () - (IBAction)btnClick; @end @implementation ...

  5. hdu-2844&&POJ-1742 Coins---多重背包

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=2844 题目大意: Tony想要买一个东西,他只有n中硬币每种硬币的面值为a[i]每种硬币的数量为c[ ...

  6. 使用node.js + socket.io + redis实现基本的聊天室场景

    在这篇文章Redis数据库及其基本操作中介绍了Redis及redis-cli的基本操作. 其中的publish-subscribe机制应用比较广泛, 那么接下来使用nodejs来实现该机制. 本文是对 ...

  7. mask r-cnn

    mask R-cnn, kaiming he的新作.可以同时完成object detection和segmentation,还可以做pose estimation,简直就是功能多多啊.在coco上测试 ...

  8. 前端jQuery基本语法

    1.概念 1.1基础知识 jQuery是一个兼容多浏览器的JavaScript库,封装了开发过程中常用的一些功能,类似Python模块 jQuery就是用JS写的,JS是基础 jQuery写起来简单, ...

  9. SpringBoot非官方教程 | 第二十五篇:2小时学会springboot

    转载请标明出处: http://blog.csdn.net/forezp/article/details/61472783 本文出自方志朋的博客 一.什么是spring boot Takes an o ...

  10. VCTransitionsLibrary –自定义iOS交互式转场动画的库

    简介 VCTransitionsLibrary 提供了许多适用于入栈,出栈,模态等场景下控制器切换时的转场动画.它本身提供了一个定义好的转场动画库,你可以拖到自己工程中直接使用;也提供了许多拥有不同转 ...