题目0x1

主要记录解题中一些stl的函数

[toc]

字符串处理

使用string标准库

  • 当使用getline(cin,string)时,如果前面有cin>>,注意吸收换行符。
substr(a,b); 返回a开始的b个字符
erase(a); earse(a,b) 移除a之后的字符; 移除a开始的b个字符
find(“a”,b); 从前向后查找a第一次出现的位置, b(可选)查找开始的下标 ,找不到时返回值-1
rfind(“a”) 从后向前查找a第一次出现的位置。
find.first_of(“abcd”) 从前向后查找abcd任一字符第一次出现的位置
find.last_of(“abcd”) 从前向后查找abcd任一字符最后一次出现的位置
find_first_not_of(“abcd”) 不在”abcd”中的字母第一次出现的地方。
find_last_not_of(“abcd”) 不在”abcd”中的字母最后一次出现的地方。
剪切粘贴

照着题目写代码,注意下字符串匹配返回的是字串的首地址就OK了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#include <iostream>
#include <string>
using namespace std;
int main()
{
std::string S;
int n;
cin >> S >> n;
while (n--)
{
std::string s1, s2, tmp, tmp1, ss;
int a, b;
int pos;


cin >> a >> b >> s1 >> s2;
a--, b--;
tmp = S.substr(a, b - a + 1); //剪切中的复制
S.erase(a, b - a + 1); //剪切中的删除
ss = s1 + s2;
if ((pos = S.find(ss)) == string::npos)
{
S += tmp; //找不到添加到字符串末尾
}
else
{
tmp1 = S.substr(0, pos);
tmp1 += s1; //注意find返回找到的字符串的首下标,因此我们要添加匹配字符串的前半部分
tmp1 += tmp; //剪切中的粘贴
tmp1 += S.substr(pos+s1.length(), S.length() - pos - s1.length()); //剩下的内容,包含匹配字符串的后半部分
S = tmp1;
}

}
cout << S << endl;

return 0;
}
估值一亿的AI核心代码
1
2
3
4
5





使用正则表达式

栈,启动!

empty() 判断是否为空,空返回true,否则返回false
size() 返回元素个数,类型为size_t
push(x) 将x压栈
top() 返回栈顶元素
pop() 从栈顶删除元素,注意没有返回值,
emplace(x) 将x压栈
  • pushemplace的区别:emplace只调用一次构造函数,而push会调用一次构造函数+一次移动构造函数。即emplace使用了移动构造函数,直接在容器内构造对象。而push需要先构造对象,再push到容器中。由于省去了拷贝构造过程,emplace一般比push快。
堆宝塔
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include <iostream>
#include <stack>

using namespace std;
int main()
{
int N, in, Tower = 0, hight = 0;
stack <int> a, b;
cin >> N;
while(N--)
{
cin >> in;
if (a.empty()) a.push(in);
else if (in < a.top()) a.push(in);
else if (b.empty() || b.top() < in) b.push(in);
else
{
Tower++;
hight = max(hight, (int)a.size()); //注意显示类型转换
while (!a.empty())
{
a.pop();
}
while (!b.empty() && b.top() > in)
{
a.push((int)b.top());
b.pop();
}
a.push(in);
}

}

if (!a.empty())
{
hight = max(hight, (int)a.size());
Tower++;
}
if (!b.empty())
{
hight = max(hight, (int)b.size());
Tower++;
}
cout << Tower << " "<< hight << endl;
return 0;
}
汉诺塔

//TODO

set

此标头是容器库的一部分,是一个内部自动有序不含重复元素的容器,因此我们要通过迭代器来访问set内的元素。

insert() 插入容器
find(value) 查找元素,如果找到了该元素,则返回指向该元素的迭代器;如果没有找到,则返回一个指向集合末尾的迭代器(即 end())。
earse(value) 删除值为value的元素
erase(first, last) 删除一个区间内的所有元素。
size() set内元素个数
clear() 清空set
集合相似度

参考:L2-005 集合相似度_题目描述 给定两个整数集合,它们的相似度定义为:nc/nt×100%。其中nc是两个集合都-CSDN博客

要注意一下这题相似度的计算公式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
#include <iostream>
#include<set>
using namespace std;
const int N = 55;
int main()
{
int n;
set<int> jihe[55];

cin >> n;
int pos = 1; //方便计算
while (n--)
{
int tmp;
cin >> tmp;
while (tmp--)
{
int tmp2;
cin >> tmp2;
jihe[pos].insert(tmp2);
}
pos++;
}
int m ;
cin >> m;
while (m--)
{
int a, b, count = 0;
cin >> a >> b;
int tmp = jihe[a].size() + jihe[b].size();
for (auto it : jihe[a])
{
if (jihe[b].find(it) != jihe[b].end()) {
count++;
}


}

double p = count * 100.0 / (tmp - count);
printf("%.2f%\n", p);
count = 0;
}
return 0;
}

队列

普通队列

top() 访问队首元素
empty() 队列是否为空
size() 返回队列内元素个数
push(a) 队尾插入元素a
emplace() 构造一个元素并插入队列
pop() 弹出队头元素
swap(other) 与other交换内容

优先队列

不同于普通队列,优先队列元素的出队顺序不是按照它们进入队列的顺序,而是根据它们的优先级来确定的。

priority_queue q; 默认情况下是最大值优先队列
priority_queue ,greater > q; 最大值优先队列
priority_queue ,less >q; 最小值优先队列
天梯赛的赛场安排

这题在比赛的时候没做出来,好吧复盘我么没做出来,分析下来是需要我们认真读题。我们看n<C的情况,需要取n最小,即没有座位的考生人数最小的组优先安排赛场,然后再插入其他组,若其他组加入后容量大于所有组的赛场容量,则新开一个赛场。(PS:这就是为什么不能直接直接将考生整除C,因为这样分配有些考场会有空座位)。

那么根据题意我们用优先队列解决问题。

参考:L2-2 天梯赛的赛场安排(C++注释版)-CSDN博客

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
#include <iostream>
#include <cmath>
#include <queue>
const int MAX_N = 5005;
int a[MAX_N];
using namespace std;
int main()
{
int N, C; //参赛学校数量和规定容量
int count = 0, cnt = 0; //坐满的考场数和多出来的学生用的考场的数量

priority_queue<int> q;
cin >> N >> C;
while (N--)
{
std::string name;
int num,kaochang;
cin >> name >> num;
kaochang = ceil(1.0*num / C); //转成浮点型,否则/会直接四舍五入

count += num / C; //坐满的考场数
cout << name << " " << kaochang << endl;
if (num % C )
{

q.emplace(num%C); //没有座位的学生人数
}
}
while (!q.empty())
{
int tmp = q.top();
q.pop();
bool flag = true;
for (int i = 0; i < cnt; i++) //查找新开的每一个考场
{
if (a[i] + tmp <= C)
{
a[i] += tmp, flag = false;
break;
}
}
if (flag) a[cnt++] = tmp; //新开考场
}

cout << cnt+count << endl;
return 0;
}

others

取整函数

注意这些函数的返回值都为double,通过隐式转换为intfloat

floor(x) 把一个小数向下取整
ceil(x) 把一个小数向上取整
round(x) 把一个小数四舍五入

数字字母判断,大小转换

#include<cctype>

isdigit() 是否为数字
isalpha() 是否未字母
isalnum() 是否为数字或者字母
.islower() 是否为小写字母
isupper() 是否为大写字母
toupper(a) 转换为大写字母
tolower(a) 转换为小写字母

一些错题

分寝室

比赛没有做出来这题,赛后总结一下,思路是对的,但是具体实现还有问题,总结就是——还要多练!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include <iostream>
#include <string>
using namespace std;
int main()
{
int n0, n1, n;
cin >> n0 >> n1 >> n;
int flag = 0;
int min = 100005;
int min_i;
for (int i = 2; i <= n && i < n0 && (n - i) < n1; i++) //限定i的范围,且不能一人一间寝室,
{
if (n0 % i == 0 && n1 % (n - i) == 0) ///能整除i时进入,即学生们能被均分
{
flag = 1;
int tmp = abs(n0 / i - n1 / (n - i)); //取绝对值,防止负数
if (tmp < min) //取相差最小的
{
min = tmp;
min_i = i;
}
}
}
if (flag)
{
cout << min_i << " " << n - min_i << endl;
}
else
{
cout << "No Solution" << endl;
}

return 0;
}
单身狗

PTA 1065 单身狗 (25 分) C++实现_找出单身狗c++-CSDN博客