紫书第三章例题--UVa340

题目:UVa340

 

地址:https://vjudge.net/problem/UVA-340

 

题解:

先给你一个标准答案序列,然后再给你一个等长的序列,让你判断两个序列有多少位对应相等,又有多少位相同只是位置不同。

 

思路:

题目要求的第一个比较好求,只要把两个序列对应位置的数字比较,就可以直接得到答案,至于第二个所要求的,可以先抛下位置的束缚,找出两个序列有多少位数字相同(不管位置是否对应),即对于0~9这10个数字,统计每个数字在两个序列出现的次数c1、c2(对第二个答案的贡献),求出min(c1,c2),再把10个数字这个值累加起来,再减去第一个所求的答案,所得的就是第二个所要求的答案。

 

实现:

第一个所要求的,可以直接遍历一遍序列并比较就可以,而第二个所要求的,则可以用长度为10的数组,分别统计两个序列中各个数字出现的次数,然后对于每个数字,求min后再累加,最后减去第一个求得答案就可以了。

 

编程技巧:

① 如果需要n长度的数组,可以先向向量填充任意n个数字,这样这个向量的长度将一直保持为n,当然,也可以继续增长,但是没法减小长度了,用clear也没用,clear函数只是把当前向量n个位置清零,如果实在要缩小长度,重新填充,只能重新声明定义了。

 

注意点:

① 注意的输入停止的条件,可以参见代码。

 

 

AC代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstdlib>
#include<string>
#include<cstring>
#include<vector>
using namespace std;

int n;
int cnt1[10];           ///统计答案序列中各个数字出现的次数
int cnt2[10];           ///统计问题序列中各个数字出现的次数

int main(){
    int Time = 0;
    while(cin >> n){
        if(!n) break;
        vector<int> ans;        ///答案序列
        vector<int> now;        ///问题序列
        printf("Game %d:\n",++Time);
        memset(cnt1,0,sizeof(cnt1));
        for(int i = 0;i < n;++i){
            int temp;
            cin >> temp;
            cnt1[temp]++;               ///统计答案序列中各个数字出现的次数
            ans.push_back(temp);
            now.push_back(temp);        ///这里只是为了把now向量的容量扩展到n
        }
        while(1){
            int sum = 0;                ///用于判断本次游戏是否已经结束是,sum=0时,游戏结束
            now.clear();
            memset(cnt2,0,sizeof(cnt2));
            for(int i = 0;i < n;++i){
                cin >> now[i];
                cnt2[now[i]]++;             ///统计问题序列中各个数字出现的次数
                sum += now[i];
            }
            if(!sum) break;
            int ans1 = 0;
            for(int i = 0;i < n;++i)            ///求得第一个所要求的答案
                if(ans[i] == now[i])
                    ans1++;
            int ans2 = 0;
            for(int i = 0;i < 10;++i){
                if(cnt2[i]&&cnt1[i])
                    ans2 += min(cnt1[i],cnt2[i]);   ///第二个所要求的答案=ans2-ans1
            }
            printf("    (%d,%d)\n",ans1,ans2-ans1);
        }
    }
    return 0;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值