IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> AtCoder Beginner Contest 253 A~E 题解 -> 正文阅读

[C++知识库]AtCoder Beginner Contest 253 A~E 题解

A - Median?

题目大意

给定正整数 a , b , c a,b,c a,b,c,判断 b b b是否为三个数中的中位数(即从小到大排序后是第二个,不是平均数)。
1 ≤ a , b , c ≤ 100 1\le a,b,c\le 100 1a,b,c100

输入格式

a ? b ? c a~b~c a?b?c

输出格式

如果 b b b是三个数中的中位数,输出Yes;否则,输出No

样例

a a a b b b c c c输出
5 5 5 3 3 3 2 2 2Yes
2 2 2 5 5 5 3 3 3No
100 100 100 100 100 100 100 100 100No

分析

本来就是A题,其实没什么难的,比赛的时候就是看成平均数WA了…(上面应该讲的够清楚了)

当然可以直接将三个数排序(简单粗暴),也可以判断 a ≤ b ≤ c a\le b\le c abc c ≤ b ≤ a c\le b\le a cba中是否有至少一个成立。

代码

#include <cstdio>
using namespace std;

int main()
{
	int a, b, c;
	scanf("%d%d%d", &a, &b, &c);
	puts((a <= b && b <= c) || (c <= b && b <= a)? "Yes": "No");
	return 0;
}

B - Distance Between Tokens

题目大意

H × W H\times W H×W的网格上,有恰好两个位置上各有一颗棋子,别的都是空位。
你可以从任意一个棋子开始,通过上下左右移动,前往另一个棋子的位置。
求至少要移动多少次?

输入格式

先是一行 H , W H,W H,W,用空格隔开,然后有 H H H行,每行是一个长度为 W W W的字符串,-表示这个位置是空位,o表示这里有一颗棋子(详见样例)。

输出格式

输出一行,即至少要移动的次数。

样例

样例输入1

2 3
--o
o--

样例输出1

3

样例输入2

5 4
-o--
----
----
----
-o--

样例输出2

4

分析

本题不需要 BFS \text{BFS} BFS,由于没有障碍物,直接找到两颗棋子,并输出 x diff + y diff x_\text{diff}+y_\text{diff} xdiff?+ydiff?(即 x , y x,y x,y的坐标差之和)即可。

代码

#include <cstdio>
#include <vector>
using namespace std;

int main()
{
	int h = 0, w = 0, x1 = -1, y1 = -1, x2 = -1, y2 = -1;
	char c;
	while((c = getchar()) != ' ')
		h = (h << 3) + (h << 1) + (c ^ 48);
	while((c = getchar()) != '\n')
		w = (w << 3) + (w << 1) + (c ^ 48);
	for(; h--; getchar())
		for(int i=w; i--; )
			if(getchar() == 'o')
				if(x1 == -1) x1 = h, y1 = i;
				else { x2 = h, y2 = i; break; }
	printf("%d\n", x1 - x2 + (y1 > y2? y1 - y2: y2 - y1));
	return 0;
}

C - Max - Min Query

题目大意

我们有一个序列 S S S,初始为空。
请处理如下 Q Q Q个操作:

  • 1 x:将 x x x插入至 S S S的末尾。
  • 2 x c:从 S S S中删除 c c c x x x,如果不够删就直接删完。
  • 3:求 S S S中最大值与最小值的差。

1 ≤ Q ≤ 2 × 1 0 5 1\le Q\le 2\times 10^5 1Q2×105
0 ≤ x ≤ 1 0 9 0\le x\le 10^9 0x109
1 ≤ c ≤ Q 1\le c\le Q 1cQ

输入格式

Q Q Q
query 1 \text{query}_1 query1?
query 2 \text{query}_2 query2?
? \vdots ?
query Q \text{query}_Q queryQ?

输出格式

对于每个操作 3 3 3,输出 S S S中最大值与最小值的差。

分析

典型STL练习题

本题可以用multisetmap解决,这里介绍使用map的方法(仅限C++使用)。
C++中,我们需要用到std::map<int, int>的如下方法:

  • mp[x]int& operator[](int&& key)
    返回key对应的value的引用,如果之前没有用到过则创建并返回 0 0 0
    时间复杂度: O ( log ? n ) \mathcal O(\log n) O(logn),其中 n n nmap中元素总数。
  • iterator begin()
    返回最小的元素对应的指针,mp.begin()->first可以获得mp的最小元素
    时间复杂度: O ( 1 ) \mathcal O(1) O(1)
  • iterator rbegin()
    返回最大的元素对应的指针,mp.rbegin()->first可以获得mp的最大元素
    时间复杂度: O ( 1 ) \mathcal O(1) O(1)
  • size_type erase(const int& key)
    key以及对应的valuemap中删除,返回删除的元素个数( 0 0 0 1 1 1),返回值一般可以忽略。
    时间复杂度: O ( log ? n ) \mathcal O(\log n) O(logn),其中 n n nmap中元素总数。

这时,每个查询都可转换为上述操作,详见代码。

代码

#include <cstdio>
#include <map>
using namespace std;

int main()
{
	int q;
	scanf("%d", &q);
	map<int, int> cnt;
	while(q--)
	{
		int op;
		scanf("%d", &op);
		if(op == 3) printf("%d\n", cnt.rbegin()->first - cnt.begin()->first);
		else if(op == 1)
		{
			int x;
			scanf("%d", &x);
			cnt[x] ++;
		}
		else if(op == 2)
		{
			int x, m;
			scanf("%d%d", &x, &m);
			if(cnt[x] > m) cnt[x] -= m;
			else cnt.erase(x);
		}
	}
	return 0;
}

D - FizzBuzz Sum Hard

题目大意

输出 1 1 1 N N N之间不是 A A A B B B的倍数的数之和。

1 ≤ N , A , B ≤ 1 0 9 1\le N,A,B\le 10^9 1N,A,B109

输入格式

N ? A ? B N~A~B N?A?B

输出格式

输出答案。

分析

根据容斥原理, 1 1 1 N N N之间 A A A B B B的倍数的数之和为:
( A (A (A的倍数之和 ) + ( B )+(B )+(B的倍数之和 ) ? ( )-( )?(同时为 A , B A,B A,B的倍数之和 ) ) )
又因为同时为 A , B A,B A,B的倍数的数是 [ A , B ] [A,B] [A,B](最小公倍数)的倍数,所以可转化为 ( A (A (A的倍数之和 ) ? ( B )-(B )?(B的倍数之和 ) + ( [ A , B ] )+([A,B] )+([A,B]的倍数之和 ) ) )

再设 f ( N ) = 1 + 2 + ? + N , g ( x , N ) = x f ( ? N X ? ) = ( N f(N)=1+2+\dots+N,g(x,N)=xf(\lfloor\frac N X\rfloor)=(N f(N)=1+2+?+N,g(x,N)=xf(?XN??)=(N以内所有 x x x的倍数之和 ) ) )
则答案为
Ans = f ( N ) ? g ( A ) ? g ( B ) + g ( [ A , B ] ) \text{Ans}=f(N)-g(A)-g(B)+g([A,B]) Ans=f(N)?g(A)?g(B)+g([A,B])
总时间复杂度为求解 [ A , B ] [A,B] [A,B]的复杂度,即 O ( log ? max ? { A , B } ) \mathcal O(\log \max\{A,B\}) O(logmax{A,B})

代码

这里使用了另一种 g ( x , N ) g(x,N) g(x,N)的求法,思路类似。

#include <cstdio>
using namespace std;

using LL = long long;
inline LL sum(const LL& x, const LL& n)
{
	LL cnt = n / x;
	return x * cnt * (cnt + 1LL) >> 1LL;
}

int main()
{
	int n, a, b;
	scanf("%d%d%d", &n, &a, &b);
	LL x = b, y = a;
	while(b ^= a ^= b ^= a %= b);
	LL t = x / a * y; // t = lcm(a, b)
	printf("%lld\n", sum(1, n) - sum(x, n) - sum(y, n) + sum(t, n));
	return 0;
}

E - Distance Sequence

题目大意

求符合如下条件的 A = ( A 1 , … , A N ) A=(A_1,\dots,A_N) A=(A1?,,AN?)的个数,对 998244353 998244353 998244353取模:

  • 1 ≤ A i ≤ M 1\le A_i\le M 1Ai?M 1 ≤ i ≤ N 1\le i\le N 1iN
  • ∣ A i ? A i + 1 ∣ ≥ K |A_i-A_{i+1}|\ge K Ai??Ai+1?K 1 ≤ i < N 1\le i<N 1i<N

2 ≤ N ≤ 1000 2\le N\le 1000 2N1000
1 ≤ M ≤ 5000 1\le M\le 5000 1M5000
0 ≤ K < M 0\le K<M 0K<M

输入格式

N ? M ? K N~M~K N?M?K

输出格式

输出符合条件的序列的个数,对 998244353 998244353 998244353取模。

分析

很明显是 DP \text{DP} DP(动态规划)的思路,仿照01背包的方式,我们设计如下状态:
dp ( i , j ) = ( A i = j 的可能数 ) \text{dp}(i,j)=(A_i=j\text{的可能数}) dp(i,j)=(Ai?=j的可能数)
状态转移方程也很简单,即:
dp ( i , j ) = ∑ p = 1 j ? k dp ( i ? 1 , p ) + ∑ p = j + k m dp ( i ? 1 , p ) \text{dp}(i,j)=\sum_{p=1}^{j-k}\text{dp}(i-1,p)+\sum_{p=j+k}^m\text{dp}(i-1,p) dp(i,j)=p=1j?k?dp(i?1,p)+p=j+km?dp(i?1,p)
那么,如果直接暴力循环计算,整个算法的时间复杂度是 O ( N M 2 ) \mathcal O(NM^2) O(NM2),显然不能通过。

但是注意到这里有个求和的操作,显然可以用前缀/后缀和优化,用 O ( 1 ) \mathcal O(1) O(1)的时间复杂度求出两个和,因此时间复杂度降到 O ( N M ) \mathcal O(NM) O(NM),可以通过。

最后一个坑,需要注意特判 K = 0 K=0 K=0的情况,答案为 M N ? m o d ? 998244353 M^N\bmod 998244353 MNmod998244353
本题到此结束。

代码

特判使用快速幂, DP \text{DP} DP建议使用滚动表(又称数组重复利用)技巧,优化后实测:

  • 时间: 49 m s → 39 m s 49\mathrm{ms}\to39\mathrm{ms} 49ms39ms
  • 空间: 21220 k b → 1664 k b 21220\mathrm{kb}\to1664\mathrm{kb} 21220kb1664kb
#include <cstdio>
#define maxn 1002
#define maxm 5005
#define MOD 998244353
using namespace std;

using LL = long long;
int qpow(LL a, LL b)
{
	LL ans = 1LL;
	while(b > 0)
	{
		if(b & 1LL) ans = ans * a % MOD;
		a = a * a % MOD, b >>= 1LL;
	}
	return ans;
}

inline void mod(int& x) { if(x >= MOD) x -= MOD; }
int dp[2][maxm];

int main()
{
	int n, m, k;
	scanf("%d%d%d", &n, &m, &k);
	if(k == 0)
	{
		printf("%d\n", qpow(m, n));
		return 0;
	}
	for(int i=1; i<=m; i++)
		dp[0][i] = 1;
	for(int i=1; i<n; i++)
	{
		int c = i & 1, p = c ^ 1, s = 0;
		for(int j=k+1; j<=m; j++)
			mod(s += dp[p][j]);
		for(int j=1; j<=m; j++)
		{
			if(j > k) mod(s += dp[p][j - k]);
			mod(dp[c][j] = s);
			if(j + k <= m)
			{
				mod(s -= dp[p][j + k]);
				if(s < 0) s += MOD;
			}
		}
	}
	int ans = 0, t = n & 1 ^ 1;
	for(int i=1; i<=m; i++)
		mod(ans += dp[t][i]);
	printf("%d\n", ans);
	return 0;
}
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-06-01 15:00:47  更:2022-06-01 15:01:29 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/12 20:44:20-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码