USACO:2.1.3 Sorting a Three-Valued Sequence 三值的排序

2.1.3 Sorting a Three-Valued Sequence 三值的排序


一、题目描述

       排序是一种很频繁的计算任务.现在考虑最多只有三值的排序问题.一个实际的例子是,当我们给某项竞赛的优胜者按金银铜牌序的时候.
在这个任务中可能的值只有三种1,2 和3.我们用交换的方法把他排成升序的.
写一个程序计算出,给定的一个1,2,3 组成的数字序列,排成升序所需的最少交换次数.
PROGRAM NAME: sort3
INPUT FORMAT
Line 1: N (1 <= N <= 1000)
Lines 2-N+1: 每行一个数字,共N 行.(1.2.3)
SAMPLE INPUT (file sort3.in)
9
2
2
1
3
3
3
2
3
1
OUTPUT FORMAT
共一行,一个数字.表示排成升序所需的最少交换次数.
SAMPLE OUTPUT (file sort3.out)
4

二、解题思路

       题意求排序所需的最少移动次数,可以先将输入的数字排序,然后得到不同的地方,就是需要我们进行交换的位置。

       建模,注意建模的思想很重要。这是我们对题目的高度数学抽象描述,必须是我们对题目进深入分析和理解之后的高度提炼。这对我们理清思路,解决问题具有重大意义,是我们编程解决问题的前提。强调,我们做题时要时刻牢记通过数学建模对题目进行描述,概况,提取,抽象表示。

      用一个组合(a,b)表示应该排序后某个位置应该是b,但之前的是a。则原题样例我们得到(2,1), (2,1), (1,2), (3,2), (3,2), (2,3), (1,3)然后求这些组合能组成的环,结果就是所有环的长度和减去环的个数。总的原则是找环以减少交换次数。

 (2,1), (1,2) ---环长度为2
 (2,1), (1,3), (3,2) ---环长度为3
 (3,2), (2,3) ---环长度为2
 结果2+3+2-3=4
长度为3的环拆分(没有顺序,一般按存储索引顺序):
(2,1), (1,3), (3,2)----拆分为(2,1),(1.3)和(3,2),(2,3). 分两步交换,第二步正好形成一个长度为2的环。

        

       进一步分析由于题目只有三个数排序,对所有排列交换形成的环长度只可能为2或3(特别之处)。所以我就开始找环,我首先找长度为2的环,每个环交换次数为1;然后找长度为3的环,我把长度为3的环拆分成两步交换(巧妙之处),(可以任意两个组合先交换)第一步交换完,会形成一个长度为2的环,因此长度为3的环交换次数就是2。如上,


源代码(仅供参考)

#include<iostream>
#include<cstdio>
#include<algorithm>

using namespace std;

int N;
int seq1[50+10],seq2[50+10];//,vis[50+10]
int cnt=0;

int main()
{
	freopen("sort3.in" , "r",stdin);
	freopen("sort3.out", "w",stdout);

	cin>>N;
	int i,j;
	for (i=0;i<N;i++)
	{
		cin>>seq1[i];seq2[i]=seq1[i];
	}
	sort(seq1,seq1+N); //5\7


	int temp;
	for (i=0;i<N;i++)
	{
		if(seq2[i]!=seq1[i]){//&&seq2[i]!=-1    
			for(j=i+1;j<N;j++){ //计算所有长度为2的环,交换次数加1         
				if(seq2[j]==seq1[i]&&seq2[j]!=seq1[j]&&seq2[i]==seq1[j]){
					cnt=cnt+1;temp=seq2[i];seq2[i]=seq2[j];seq2[j]=temp;break;
				}
			}
		}
	}

	for (i=0;i<N;i++)
	{
	    if(seq2[i]!=seq1[i]){
    	    for(j=i+1;j<N;j++){ //计算有长度为3的环,拆开两步交换(长度为3的环拆分成两个长度为2的环)           
	    	if(seq2[j]==seq1[i]&&seq2[j]!=seq1[j]){
				cnt=cnt+1;temp=seq2[i];seq2[i]=seq2[j];seq2[j]=temp;break;
			}
	   }
	   }
	}

	cout<<cnt<<endl;
	return 0;   
}</span></span></span>

由于自身是初学者,编程能力有限,未达到专业程序员的水平,可能误导大家,请大家甄读;文字编辑也一般,文中可能会有措辞不当。博文中的错误和不足敬请读者批评指正。


  • 5
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蓝亦

感谢博主辛勤的付出

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值