数组模拟小总结 一


本文只是总结了本人(小废物)碰到的数组模拟的题,及个人的小感受。很是粗浅,哪里有不合适的地方,欢迎各位大佬指点@^@!

一、逃跑的zhb

在这里插入图片描述

分析

  1. 我们要存储m条通路及其费用,且无向
    • 邻接矩阵存储通路
    • g[a][b] = g[b][a] = 钱数
  2. 对于给出的 k 条通路:
    • (1)存储,这些为无序整数,但我们要存储其的顺序(按顺序访问),我们可以设置:i 代表的顺序,s[i] =数值
    • (2)如何判断是否走了重复的边:再设置一个辅助数组,这样:st[s[i]] 在不断输入s[i]的同时,不断更新st[s[i]]的值,使其记录一个点出现的次数。
    • (3)如何判断是否走重边,是否把每个点走过? 对于给出的k条路径,在输入的同时,设置一个flag,遇到重边即为false,判断路径的个数是否为n,即可
    • (4)在给出的路径合法时,我们直接用一个函数 min 求出最小值即可。
    • (5)int ans = 0x3f3f3f3f; 为初始化为无限大的意思。
    • (6)边输入k条路,边更新最小值,如何把这条路上的每个点都走完?
    • 我们已知起点是0, g[0][s[i]] 即为走的第一步花的钱数, s[i]即为下一步的起点, i ++ 即为下个点, 即 t = s[i], i ++, g[t][s[i]]

代码

#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstring>
using namespace std;
const int N = 310;
int n, m, k;
int g[N][N];//邻接矩阵存储通路
int s[N], st[N];

int main(void){
    cin >> n >> m;
    //输入m条通路
    while(m --){
        int a, b, c;
        cin >> a >> b >> c;
        g[a][b] = g[b][a] = c;
    }
    cin >> k;
    int ans = 0x3f3f3f3f;
    while(k --){
        memset(st, 0, sizeof(st));
        int q;
        bool flag = true;
        cin >> q;
        st[0] = 1; //初始化起点,算是出现一次
        for(int i = 0; i < q; i ++){
            cin >> s[i];
            if(st[s[i]]) flag = false;//记录有重复的点
            st[s[i]] ++;
        }
        //判断这条路是否合法(不走重复点,不漏走点)
        if(!flag || q != n){
            continue;
        }
        else{
            bool flag1 = true; //判断给出的两点间是否有路可走
            int sum = 0, t = 0;//t存储当前是哪个点,sum存储的总钱数
            for(int i = 0; i < n; i ++){
                if(g[t][s[i]] == 0){
                    flag1 = false;
                    break;
                }
                else sum += g[t][s[i]], t = s[i];
            }
            if(g[t][0] == 0){ //判断最后一个点能否到家
                flag1 = false;
                continue;
            }
            else sum += g[t][0], t = 0;
            if(flag1) ans = min(ans, sum);
        }
    }
    if(ans == 0x3f3f3f3f) cout << -1 << endl;
    else cout << ans << endl;
    return 0;
}

二、数组映射

1. 简述

利用数组下标表示某种含义,不单单是元素的位置

例题1:输入一个数,判断每个数字出现了几回

分析
arr[a] = k, 表示这个数出现了几次

取余

#include <algorithm>
#include <iostream>
#include <cstdio>
using namespace std;
int arr[10]; //a[i] = k, 表示 数字i出现了k次
int main(void){
    int n;
    cin >> n;
    while(n > 0){
        int a = n % 10;
        arr[a] ++;
        n /= 10;
    }
    for(int i = 0; i < 10; i ++){
        printf("%d 出现了 %d 次\n", i, arr[i]);
    }
    return 0;
}

以字符串形式读数

优点
不用为数的范围操心

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int arr[10]; //a[i] = k, 表示 数字i出现了k次
char str[100];
int main(void){
    cin >> str;
    int len = strlen(str);
    for(int i = 0; i < len; i ++){
        int a;
        a = str[i] - '0';
        arr[a] ++;
    }
    for(int i = 0; i < 10; i ++){
        printf("%d 出现了 %d 次\n", i, arr[i]);
    }
    return 0;
}

例题2:买奶茶

问题描述:
(来自td新生赛,找不到题目链接了,凑活着看看感受一下吧)
学长早早便将学妹喜欢喝的奶茶价格和奶茶店的价格表弄到了手!现在你要根据奶茶价格,判断学妹到底喜欢喝哪一种奶茶,如果有多种奶茶价格都符合要求,输入种类靠前的一种

 x 输入描述:第一行包含一个整数n,表示有n种奶茶第二行有n个数字a[i],a[i]表示第i种奶茶价格第三行包含一个整数Q,表示Q次询问第四行包含Q个整数x[i],表示学妹喜欢喝的第i杯奶茶的价格,如果没有找到,则输出-1

分析

我们还可以这样模拟:
arr[a] = i + 1 b表示 a 数出现的顺序

#include <stdio.h>
long long int n, q, i, j, a;
long long int  arr1[10000000];
int main(void){
    scanf("%lld",&n);
    for(i=0; i<n; i++){
        scanf("%lld",&a);
        if(arr1[a]==0){//这里保证了只记录第一次出现的位置
            arr1[a]=i+1;
        }
    }
    scanf("%lld",&q);
    for(i=0; i<q; i++){
        scanf("%lld",&a);
        if(arr1[a]==0){
            printf("%d\n",-1);
        }
        else{
            printf("%d\n",arr1[a]);
        }
    }
    return 0;
}

例题三:花瓣数

问题描述:
(来自td新生赛,找不到链接了)
3.他现在有一把N朵漂亮的花,每多花都有a[i]个花瓣,如果有1朵花的花瓣数与其他任意一朵花的花瓣数之差等于2,那么他可以选择把这朵花扔掉,也可以选择保留。 请问经过若干次操作之后蒲煜凡学学长能找到唯一的一支特别的花送给小学妹吗?

分析

数组在这相当于 标记 作用

2 4 6 
p[0]= p[4]= p[2]=p[6]= p[8]=1,   
对于p[4]=1, 说明有比他小2的,也有比他大二的,它可以是最后保留的一个,也可以是去掉的那一个
我可以留2 4 6中任何一朵,最后判断时是从i=1开始,已经留了一朵(如果其他均被标记则其他均可被去掉)
想法:除第一朵外,如果其他均可被去掉,则成立
#include<stdio.h>
#include<string.h>

int arr[100010],n,t;
int p[1000010];
int main(void){
    scanf("%d",&t);
    while(t--){
        scanf("%d",&n);
        memset(p,0,sizeof(p));
        for(int i=0;i<n;i++){
            scanf("%d",&arr[i]);
            p[arr[i]+2]=1;       // 比arr[i] 大2的数记录一下
            p[arr[i]-2]=1;       // 比arr[i] 小2的数记录一下
        }
        int flag=0;
        for(int i=1;i<n;i++){
            if(!p[arr[i]]) {  //遍历一下arr[i]中是否存在 没有在p中出现过的数
                  flag=1;
                  break;
            }
        }
        if(flag==0)
            puts("YES");
        else
            puts("NO");
    }
    return 0;
}

一道不应景的前缀和小题

在这里插入图片描述

分析

维护着前n个数的前缀和, 比如当前求第i项和后边的项数相乘的结果
sum += (s[n] - s[i] ) * a[i] 即可。

代码

#include <algorithm>
#include <iostream>
using namespace std;
typedef long long LL;
const LL m = 1e9 + 7;
const int N = 2e5 + 10;
int n;
int a[N];
LL s[N];
LL ans;
int main(void){
    cin >> n;
    for(int i = 1; i <= n; i ++){
        cin >> a[i];
        s[i] = s[i - 1] + a[i];
    }
    for(int i = 1; i < n; i ++){
        ans = (ans % m)  + a[i] * (s[n] - s[i]) % m;
    }
    cout << ans << endl;
    return 0;
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Xuhx&

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值