2.黛玉泡茶
我的思路代码:(但我不知道哪有错误)
cpp
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
int main(){
int n,m,k,res=1;
cin>>n>>m>>k;
vector<int>num(n+1,0);
for(int i=1;i<=n;i++) cin>>num[i];
sort(num.begin()+1,num.end());
int sum=m;
for(int i=1;i<=k;i++){
sum-=num[i];
if(sum<0){sum+=m;res++;}
}
cout<<res<<endl;
return 0;
}
正确的思路代码:
cpp
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
int main(){
int n,m,k,res=0;
cin>>n>>m>>k;
vector<int>num(n+1,0);
for(int i=1;i<=n;i++) cin>>num[i];
sort(num.begin(),num.end());
for(int i=1;i<=k;i++) res+=num[i];
if(res%m==0)cout<<res/m<<endl;
else cout<<res/m+1<<endl;
return 0;
}
反思:我知道了。查看了测试用例后,我才明白,可能茶壶容量远小于一杯茶的容量,也就是说,你要倒满一杯茶,需要接好几次水壶,而我的代码中循环一次res最多也加一。后果就是,对于这类测试用例,我的循环结束了,我的sum还远远小于0,还没加成正值。
3.宝玉请安
我的错误思路代码:
cpp
#include <iostream>
#include<cmath>
using namespace std;
int main()
{
int t,x1,x2,x3;
int res=0;//需走的步数
cin>>t;
while(t--){
cin>>x1>>x2>>x3;
if((x2<x1&&x3<x1)||(x2>x1&&x3>x1)){
if(x2>x3) res=abs(x2-x1);
else res=abs(x3-x1);
}
else if((x2<x1&&x3>x1)||(x2>x1&&x3<x1)){
if(abs(x1-x3)<abs(x1-x2)){
res=abs(x1-x3)+abs(x2-x3);
}
else res=abs(x1-x2)+abs(x3-x2);
}
cout<<res<<endl;
}
// 请在此输入您的代码
return 0;
}
正确思路代码:
cpp
#include <iostream>
#include<cmath>
using namespace std;
int main()
{
int t,x1,x2,x3;
int res=0;//需走的步数
cin>>t;
while(t--){
cin>>x1>>x2>>x3;
res=min(abs(x1-x2),abs(x1-x3))+abs(x2-x3);
cout<<res<<endl;
}
// 请在此输入您的代码
return 0;
}
反思:思考了好长时间,不知道自己为什么错,就是觉得自己的思路很正确,难受死了。最后看了看别人的和我相似的思路,才发现原来自己思路有漏洞,我把可能情况分两部分,一是两个目标地点在同一侧,二是不在同一侧。在同一侧的步数统计有问题,我直接判断两点谁更大,实际上应该判断两点谁离x1更远。
还学到了别人的直触本质的思路。本质上是x1到x2或x3其中一个,让后再在x2,x3之间行走。
4.贾母祝寿
我的思路代码:
cpp
#include <iostream>
#include<cmath>
#include<vector>
using namespace std;
int main()
{
int n,q;
long long res=0;
cin>>n>>q;
vector<long long>stone(n,0);
while(q--){
int t,x,y;
cin>>t>>x>>y;
if(t==1){
for(int i=0;i<x;i++){
stone[i]+=y;
}
}
else {
for(int i=n-1;i>=n-x;i--){
stone[i]-=y;
}
}
}
for(int i=0;i<n;i++){
stone[i]=abs(stone[i]);
res=max(res,stone[i]);
}
cout<<res<<endl;
// 请在此输入您的代码
return 0;
}
正确思路代码:
cpp
#include <iostream>
#include<cmath>
#include<vector>
using namespace std;
int main()
{
int n,q;
long long res=0;
cin>>n>>q;
long long num1=0,num2=0;
while(q--){
int t,x,y;
cin>>t>>x>>y;
if(t==1){
num1+=y;
if(x==n)num2+=y;
}
else {
num2-=y;
if(x==n) num1-=y;
}
}
res=max(abs(num1),abs(num2));
cout<<res<<endl;
// 请在此输入您的代码
return 0;
}
反思:我的思路时间复杂度太高,O(n*m),n的数据范围是10^5,m的数据范围是10^9。
5.清洁客房
我根本没想到用dp数组,我想用排列组合做一下,没做出来。
#include <iostream>
#include<vector>
using namespace std;
int N = 1e5+1;
int mod = 1e9 + 7;
int main()
{
vector<vector<long>> dp(N, vector<long>(4,0));
//前i个房间选择j种的方案类型
for(int i=1;i<N;i++)dp[i][1]=9;
for (int i = 2; i <N; i++) {
for (int j = 2; j <= 3; j++) {
dp[i][j] = (dp[i - 1][j - 1] * (10 - (j - 1))%mod + dp[i - 1][j] * j%mod ) % mod;
}
}
int t, n;
cin >> t;
while (t--) {
cin >> n;
cout << dp[n][3] << endl;
}
// 请在此输入您的代码
return 0;
}
dp数组含义:dp[i][j]前i个房间选择j种等级的方案数 (i,j均从0开始)
6.宝玉与黛玉的考验
正确代码
cpp
#include<iostream>
#include<string>
using namespace std;
int n, m, k;
string a, b;
long get(long a, long b) {
return k * a + k * b - 2 * a * b;
}
int main() {
cin >> n >> m >> k >> a >> b;
int mxa = 0, mxb = 0, mna = k, mnb = k, sum = 0;
//滑动窗口,找出4个值、
for (int i = 0; i < n; i++) {
sum += a[i] - '0';
if (i >= k) {
sum -= a[i - k] - '0';
mxa = max(sum, mxa);
mna = min(sum, mna);
}
}
sum = 0;
for (int i = 0; i < m; i++) {
sum += b[i] - '0';
if (i >= k) {
sum -= b[i - k] - '0';
mxb = max(sum, mxb);
mnb = min(sum, mnb);
}
}
cout << max(max(get(mxa, mxb), get(mxa, mnb)), max(get(mna, mxb), get(mna, mnb))) << endl;
return 0;
}
思路:这题看来是和数学思维相关。开始我想了遍历整个二维数组,但是很明显超时。
用数学思维来分析:设k*k区间中分别有a行,b列被选中,即长度为n的字符串中有a个1,长度为m的字符串中有b个1。那么这个区间的价值为ka-ab+kb-ab=-2(a-k/2)(b-k/2)+k^2/2。(k是常量,a,b是变量)a,b单独变化,这个表达式 的最值在a,b取到端点的时候取到最值。那么就求a,b的最大小值。求的过程用到滑动窗口。