NOIP2019年普及组复赛上机试题

A. 数字游戏

小 K 同学向小 P 同学发送了一个长度为 8 的 01 字符串 来玩数字游戏,小 P 同学想要知道字符串中究竟有多少个 1。

注意:01 字符串为每一个字符是 0 或者 1 的字符串,如“101”(不含双引号)为一个长度为 3 的 01 字符串。

输入格式
共一行,包含一个长度为 8 的 01 字符串 s。

输出格式
共一行,包含一个整数,即 01 字符串中 字符 1 的个数。

数据范围
对于 20% 的数据,保证输入的字符全部为 0。

对于 100% 的数据,输入只可能包含字符 0 和字符 1,字符串长度固定为 8。

输入样例1

00010100

输出样例1

2

输入样例2

11111111

输出样例2

8

输入样例3

01010101

输出样例3

4

AC代码

#include <iostream>
#include <cstdio>
using namespace std;
int main()
{
	string s;
	cin >> s;
	int len = s.length();
	int ans = 0;
	for (int i = 0; i < len; i++)
	{
		if (s[i] == '1')
		{
			ans++;
		}
	}
	printf("%d\n", ans);
	return 0;
}

B. 公交换乘

著名旅游城市 B 市为了鼓励大家采用公共交通方式出行,推出了一种地铁换乘公交车的优惠方案:

在搭乘一次地铁后可以获得一张优惠票,有效期为 45 分钟,在有效期内可以消耗这张优惠票,免费搭乘一次票价不超过地铁票价的公交车。在有效期内指开始乘公交车的时间与开始乘地铁的时间之差小于等于 45 分钟,即:tbus−tsubway≤45
搭乘地铁获得的优惠票可以累积,即可以连续搭乘若干次地铁后再连续使用优惠票搭乘公交车,但每次搭乘公交车只能使用一张优惠券。
搭乘公交车时,如果可以使用优惠票一定会使用优惠票;如果有多张优惠票满足条件,则优先消耗获得最早的优惠票。
现在你得到了小轩最近的公共交通出行记录,你能帮他算算他的花费吗?

输入格式
第一行包含一个正整数 n,代表乘车记录的数量。

接下来的 n 行,每行包含 3 个整数,相邻两数之间以一个空格分隔。第 i 行的第 1 个整数代表第 i 条记录乘坐的交通工具,0 代表地铁,1 代表公交车;第 2 个整数代表第 i 条记录乘车的票价 pricei ;第三个整数代表第 i 条记录开始乘车的时间 ti(距 0 时刻的分钟数)。

我们保证出行记录是按照开始乘车的时间顺序给出的,且 不会有两次乘车记录出现在同一分钟。

输出格式
有一行,包含一个正整数,代表小轩出行的总花费。

数据范围
对于 30% 的数据,n≤1000,ti≤106。

另有 15% 的数据,ti≤107,pricei 都相等。

另有 15% 的数据,ti≤109,pricei 都相等。

对于 100% 的数据,n≤105,ti≤109,1≤pricei≤1000。

输入样例1

6
0 10 3
1 5 46
0 12 50
1 3 96
0 5 110
1 6 135

输出样例1

36

输入样例2

6
0 5 1
0 20 16
0 7 23
1 18 31
1 4 38
1 7 68

输出样例2

32

样例解释
样例#1:
第一条记录,在第 3 分钟花费 10 元乘坐地铁。

第二条记录,在第 46 分钟乘坐公交车,可以使用第一条记录中乘坐地铁获得的优惠票,因此没有花费。

第三条记录,在第 50 分种花费 12 元乘坐地铁。

第四条记录,在第 96 分钟乘坐公交车,由于距离第三条记录中乘坐地铁已超过 45 分钟,所以优惠票已失效,花费 3 元乘坐公交车。

第五条记录,在第 110 分钟花费 5 元乘坐地铁。

第六条记录,在第 135 分钟乘坐公交车,由于此时手中只有第五条记录中乘坐地铁获得的优惠票有效,而本次公交车的票价为 6 元,高于第五条记录中地铁的票价 5 元,所以不能使用优惠票,花费 6 元乘坐公交车。

总共花费 36 元。

样例#2:
第一条记录,在第 1 分钟花费 5 元乘坐地铁。

第二条记录,在第 16 分钟花费 20 元乘坐地铁。

第三条记录,在第 23 分钟花费 7 元乘坐地铁。

第四条记录,在第 31 分钟乘坐公交车,此时只有第二条记录中乘坐的地铁票价高于本次公交车票价,所以使用第二条记录中乘坐地铁获得的优惠票。

第五条记录,在第 38 分钟乘坐公交车,此时第一条和第三条记录中乘坐地铁获得的优惠票都可以使用,使用获得最早的优惠票,即第一条记录中乘坐地铁获得的优惠票。

第六条记录,在第 68 分钟乘坐公交车,使用第三条记录中乘坐地铁获得的优惠票。

总共花费 32 元。
AC代码

#include <iostream>
#include <cstdio>
using namespace std;
const int N = 100005;
struct Ticket
{
	int time, price;
} q[N];
int vis[N];
int main()
{
	int n;
	scanf("%d", &n);
	
	int ans = 0, l = 0, r = 0;
	for (int i = 0; i < n; i++)
	{
		int type, price, time;
		scanf("%d%d%d", &type, &price, &time);
		if (type == 0)
		{
			q[r++] = {time, price};
			ans += price;
		}
		else
		{
			while (l < r && time - q[l].time > 45) l++;
			
			int success = 0;
			for (int j = l; j < r; j++)
			{
				if (!vis[j] && q[j].price >= price)
				{
					vis[j] = 1;
					success = 1;
					break;
				}
			}
			if (!success)
			{
				ans += price;
			}
		}
	}
	printf("%d\n", ans);
	return 0;
}

C. 纪念品

小伟突然获得一种超能力,他知道未来 T 天 N 种纪念品每天的价格。
某个纪念品的价格是指购买一个该纪念品所需的金币数量,以及卖出一个该纪念品换回的金币数量。
每天,小伟可以进行以下两种交易无限次:

  1. 任选一个纪念品,若手上有足够金币,以当日价格购买该纪念品,注意同一个 纪念品可以在同一天重复买;
  2. 卖出持有的任意一个纪念品,以当日价格换回金币。
    每天卖出纪念品换回的金币可以立即用于购买纪念品,当日购买的纪念品也可以当日卖出换回金币。

当然,一直持有纪念品也是可以的。
T 天之后,小伟的超能力消失。
因此他一定会在第 T 天卖出所有纪念品换回金币。
小伟现在有 M 枚金币,他想要在超能力消失后拥有尽可能多的金币。

输入格式
第一行包含三个正整数 T,N,M,相邻两数之间以一个空格分开,分别代表未来天数 T,纪念品数量 N,小伟现在拥有的金币数量 M。

接下来 T 行,每行包含 N 个正整数,相邻两数之间以一个空格分隔。第 i 行的 N 个正整数分别为 Pi,1,Pi,2,……,Pi,N,其中 Pi,j 表示第 i 天第 j 种纪念品的价格。

输出格式
输出仅一行,包含一个正整数,表示小伟在超能力消失后最多能拥有的金币数量。

数据范围
对于 10% 的数据,T=1。

对于 30% 的数据,T≤4,N≤4,M≤100,所有价格 10≤Pi,j≤100。

对于 15% 的数据,T≤100,N=1。

对于 15% 的数据,T=2,N≤100。

对于 100% 的数据,T≤100,N≤100,M≤103,所有价格 1≤Pi,j≤104,数据保证任意时刻,小明手上的金币数不可能超过 104。

输入样例1

6 1 100
50
20
25
20
25
50

输出样例1

305

输入样例2

3 3 100
10 20 15
15 17 13
15 25 16

输出样例2

217

样例解释
样例#1:
最佳策略是:

第二天花光所有 100 枚金币买入 5 个纪念品 1;

第三天卖出 5 个纪念品 1,获得金币 125 枚;

第四天买入 6 个纪念品 1,剩余 5 枚金币;

第六天必须卖出所有纪念品换回 300 枚金币,第四天剩余 5 枚金币,共 305 枚金币。

超能力消失后,小伟最多拥有 305 枚金币。

样例#2:
最佳策略是:

第一天花光所有金币买入 10 个纪念品 1;

第二天卖出全部纪念品 1 得到 150 枚金币并买入 8 个纪念品 2 和 1 个纪念品 3,剩余 1 枚金币;

第三天必须卖出所有纪念品换回 216 枚金币,第二天剩余 1 枚金币,共 217 枚金币。

超能力消失后,小伟最多拥有 217 枚金币。
AC代码

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
const int N = 1005, M = 10005;
int t, n, m;
int dp[M], w[N][N];
int main()
{
	scanf("%d%d%d", &t, &n, &m);
	for (int i = 1; i <= t; i++)
	{
		for (int j = 1; j <= n; j++)
		{
			scanf("%d", &w[i][j]);
		}
	}
	
	for (int i = 1; i < t; i++)
	{
		memset(dp, 0, sizeof(dp));
		for (int j = 1; j <= n; j++)
		if (w[i + 1][j] > w[i][j])
		{
			for (int k = w[i][j]; k <= m; k++)
			{
				dp[k] = max(dp[k], dp[k - w[i][j]] + w[i + 1][j] - w[i][j]);
			}
		}
		m += dp[m];
	}
	printf("%d\n", m);
	return 0;
}

D. 加工零件

凯凯的工厂正在有条不紊地生产一种神奇的零件,神奇的零件的生产过程自然也很神奇。
工厂里有 n 位工人,工人们从 1∼n 编号。

某些工人之间存在双向的零件传送带。

保证每两名工人之间最多只存在一条传送带。

如果 x 号工人想生产一个被加工到第 L(L>1) 阶段的零件,则所有与 x 号工人有传送带直接相连的工人,都需要生产一个被加工到第 L−1 阶段的零件(但 x 号工人自己无需生产第 L−1 阶段的零件)。

如果 x 号工人想生产一个被加工到第 1 阶段的零件,则所有与 x 号工人有传送带直接相连的工人,都需要为 x 号工人提供一个原材料。

轩轩是 1 号工人。

现在给出 q 张工单,第 i 张工单表示编号为 ai 的工人想生产一个第 Li 阶段的零件。

轩轩想知道对于每张工单,他是否需要给别人提供原材料。

他知道聪明的你一定可以帮他计算出来!

输入格式
第一行三个正整数 n,m 和 q,分别表示工人的数目、传送带的数目和工单的数目。

接下来 m 行,每行两个正整数 u 和 v,表示编号为 u 和 v 的工人之间存在一条零件传输带。保证 u≠v。

接下来 q 行,每行两个正整数 a 和 L,表示编号为 a 的工人想生产一个第 L 阶段的零件。

输出格式
共 q 行,每行一个字符串 “Yes” 或者 “No”。如果按照第 i 张工单生产,需要编号为 1 的轩轩提供原材料,则在第 i 行输出 “Yes”;否则在第 i 行输出 “No”。注意输出不含引号。

数据范围
1≤u,v,a≤n。
测试点 1∼4,1≤n,m≤1000,q=3,L=1。
测试点 5∼8,1≤n,m≤1000,q=3,1≤L≤10。
测试点 9∼12,1≤n,m,L≤1000,1≤q≤100。
测试点 13∼16,1≤n,m,L≤1000,1≤q≤105。
测试点 17∼20,1≤n,m,q≤105,1≤L≤109。

输入样例1

3 2 6
1 2
2 3
1 1
2 1
3 1
1 2
2 2
3 2

输出样例1

No
Yes
No
Yes
No
Yes

输入样例2:

5 5 5
1 2
2 3
3 4
4 5
1 5
1 1
1 2
1 3
1 4
1 5

输出样例2

No
Yes
No
Yes
Yes

样例解释
样例#1:
在这里插入图片描述

编号为 1 的工人想生产第 1 阶段的零件,需要编号为 2 的工人提供原材料。

编号为 2 的工人想生产第 1 阶段的零件,需要编号为 1 和 3 的工人提供原材料。

编号为 3 的工人想生产第 1 阶段的零件,需要编号为 2 的工人提供原材料。

编号为 1 的工人想生产第 2 阶段的零件,需要编号为 2 的工人生产第 1 阶段的零件,需要为编号 1 和 3 的工人提供原材料。

编号为 2 的工人想生产第 2 阶段的零件,需要编号为 1 和 3 的工人生产第 1 阶段的零件,他/她们都需要编号为 2 的工人提供原材料。

编号为 3 的工人想生产第 2 阶段的零件,需要编号为 2 的工人生产第 1 阶段的零件,需要编号为 1 和 3 的工人提供原材料。

样例#2:

2.png

编号为 1 的工人想生产第 1 阶段的零件,需要编号为 2 和 5 的工人提供原材料。

编号为 1 的工人想生产第 2 阶段的零件,需要编号为 2 和 5 的工人生产第 1 阶段的零件,需要编号为 1,3,4 的工人提供原材料。

编号为 1 的工人想生产第 3 阶段的零件,需要编号为 2 和 5 的工人生产第 2 阶段的零件,需要编号为 1,3,4 的工人生产第 1 阶段的零件,需要编号为 2,3,4,5 的工人提供原材料。

编号为 1 的工人想生产第 4 阶段的零件,需要编号为 2 和 5 的工人生产第 3 阶段的零件,需要编号为 1,3,4 的工人生产第 2 阶段的零件,需要编号为 2,3,4,5 的工人生产第 1 阶段的零件,需要全部工人提供原材料。

编号为 1 的工人想生产第 5 阶段的零件,需要编号为 2 和 5 的工人生产第 4 阶段的零件,需要编号为 1,3,4 的工人生产第 3 阶段的零件,需要编号为 2,3,4,5 的工人生产 第 2 阶段的零件,需要全部工人生产第 1 阶段的零件,需要全部工人提供原材料。
AC代码

#include <iostream>
#include <cstdio>
#include <queue> 
#include <cstring>
using namespace std;

typedef pair<int, int> PII;

const int N = 100005, M = 200005;
int n, m, query, cnt;
int head[N], dis[N][2];

struct Edge{
	int to;
	int next;
} edge[M];

queue<PII> q; 

void add(int u, int v)
{
	edge[cnt].to = v;
	edge[cnt].next = head[u];
	head[u] = cnt++; 
}

void bfs()
{
	memset(dis, 0x3f, sizeof(dis));
	
	q.push({1, 0});
	dis[1][0] = 0;
	
	while (!q.empty())
	{
		PII t = q.front();
		q.pop();
		int now = t.first, type = t.second;
		
		for (int i = head[now]; ~i; i = edge[i].next)
		{
			int j = edge[i].to;
			if (dis[j][type ^ 1] > dis[now][type] + 1)
			{
				dis[j][type ^ 1] = dis[now][type] + 1;
				q.push({j, type ^ 1});
			} 
		}
	}
}

int main()
{
	memset(head, -1, sizeof head);
	scanf("%d%d%d", &n, &m, &query);
	for (int i = 0; i < m; i++)
	{
		int u, v;
		scanf("%d%d", &u, &v);
		add(u, v);
		add(v, u);
	}
	
	bfs();
	
	while (query--)
	{
		int a, l;
		scanf("%d%d", &a, &l);
		if (head[1] == -1 && a == 1) puts("No");
		else if (dis[a][l & 1] <= 1) puts("Yes");
		else puts("No");
	}
	
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值