牛客练习赛123(A,B,C,D)

2024-03-31 10:52
文章标签 牛客 123 练习赛

本文主要是介绍牛客练习赛123(A,B,C,D),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

牛客挑战赛,练习赛和小白月赛周赛不是一种东西。这玩意跟CF的div12差不多难度。而且找不到题解。所以决定不等题解补题了,直接写题解了。

比赛链接

光速签到下班,rk++。感觉E可能能补掉,看情况补吧。

B题感觉之前考了两次,结论和证明构造过程需要理解,赛时记得结论直接秒了。C题是不太裸的裸完全背包,思路不裸但是写法裸。D题需要加个优化,思路看的别人的,很妙。


A 炸鸡块哥哥的粉丝题

思路:

签到,截取前一半的子串即可。

code:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;int n;
string s;int main(){cin>>n>>s;cout<<s.substr(0,(n+1)/2);return 0;
}

B 智乃想考一道鸽巢原理

思路:

看了评论区题解才发现,自己这么长时间理解的方法原来就是鸽巢原理。指路

我们想让某一种小球留下来,肯定先让其他小球先配对抵消掉,其他小球抵消剩下的再用这种小球来消。那么问题就变成了求其他小球抵消剩下的小球有几个,如果剩下的球比这种小球少,那就可以剩下这种小球,否则就剩不下。

我们把最多的那个小球看作是鸽巢,然后剩余的小球看作是鸽子,也就是鸽巢有 m x mx mx 个,鸽子有 r m = t o t − a i rm=tot-a_i rm=totai 个。

  1. m x > ⌊ r m 2 ⌋ mx\gt \left\lfloor\dfrac{rm}2\right\rfloor mx>2rm 时:可以每个鸽巢放入一只鸽子,此时剩余的小球个数就是剩余的鸽巢个数,即 m x − ( r m − m x ) = 2 ∗ m x − r m mx-(rm-mx)=2*mx-rm mx(rmmx)=2mxrm
  2. m x ≤ ⌊ r m 2 ⌋ mx\le \left\lfloor\dfrac{rm}2\right\rfloor mx2rm 时:可以用其他小球来充当鸽巢,使得鸽巢拓展到 ⌈ r m 2 ⌉ \left\lceil\dfrac{rm}2\right\rceil 2rm 个。然后其他小球依次放入鸽巢即可,可以保证存在一种方法使得小球不会放在同种颜色小球的鸽巢里(因为最少可以只有一种颜色又当鸽巢又放小球,其他颜色就不可能重复了。而每种颜色小球个数不会超过 ⌈ r m 2 ⌉ \left\lceil\dfrac{rm}2\right\rceil 2rm 个,所以这种颜色可以不重复)。此时会剩余 r m % 2 rm\%2 rm%2 个鸽巢。

所以我们枚举 i i i 时,对每个 i i i 只要知道剩余的小球的总个数和剩余最大个数的小球,就可以计算其他小球抵消最后会剩多少小球,也就知道能否剩下这种小球。我们可以设置一个变量记录所有小球个总个数,并处理出前缀最大值和后缀最大值,这样就可以快速查询了。

code:

#include <iostream>
#include <cstdio>
using namespace std;
typedef long long ll;
const int maxn=1e6+5;int T,n;
int a[maxn],prem[maxn],sufm[maxn];
ll tot;int main(){cin>>T;while(T--){cin>>n;tot=0;for(int i=1;i<=n+1;i++)prem[i]=sufm[i]=0;for(int i=1;i<=n;i++){cin>>a[i];prem[i]=max(prem[i-1],a[i]);tot+=a[i];}for(int i=n;i>=1;i--)sufm[i]=max(sufm[i+1],a[i]);for(int i=1;i<=n;i++){ll mx=max(prem[i-1],sufm[i+1]),rm=tot-a[i],lst;if(mx>rm/2)lst=2*mx-rm;else lst=rm&1;cout<<"01"[a[i]>lst]<<" \n"[i==n];}}return 0;
}

C 智乃想考一道完全背包(Easy version)

思路:

如果位置 k k k 上的物品本来就很好了,那么肯定选这个物品就行了。但是如果有个很好的物品远离位置 k k k,那么我们就需要选上从这个位置到 k k k 上的所有物品,才能选上这一个物品,它就相当于和中间的物品进行捆绑销售了。

反正一定会捆绑销售,所以我们不如一开始就将 l ∼ k l\sim k lk k ∼ r k\sim r kr 位置上的物品进行绑定,把它们变成一个物品,然后跑完全背包。

不过由于我们选了 l ∼ k l\sim k lk 的物品的话,我们还能去拓展另一边到 r r r,并省下一个 k k k。所以另一边我们也需要绑定过来,也就是说需要把 l ∼ r l\sim r lr 上的物品进行绑定。

如果我们以位置为横坐标,每个位置上物品选择的个数为纵坐标画直方图的话,它的形状就是一个金字塔型。我们绑定物品同时选取,其实就相当于绑定一行,每行都是整块选取。

时间复杂度看似是 O ( n 2 ∗ m ) O(n^2*m) O(n2m) 的。但是由于背包容量最多 m m m,重量超过 m m m 的物品我们根本更新不了答案,所以最多有 m 2 m^2 m2 级别个物品会更新 d p dp dp 值,所以时间复杂度最多是 O ( n 2 + m 3 ) O(n^2+m^3) O(n2+m3) 的。

code:

没必要开longlong

#include <iostream>
#include <cstdio>
using namespace std;
const int maxn=2005;
const int maxm=505;
typedef long long ll;
const ll linf=1e18;int n,m,k;
ll dp[maxm];ll tw[maxn],tv[maxn];int main(){cin>>n>>m>>k;for(int i=1;i<=n;i++){cin>>tw[i]>>tv[i];tw[i]+=tw[i-1];tv[i]+=tv[i-1];}for(int l=1;l<=k;l++)for(int r=k;r<=n;r++){ll w=tw[r]-tw[l-1],v=tv[r]-tv[l-1];for(int j=w;j<=m;j++)dp[j]=max(dp[j],dp[j-w]+v);}for(int i=1;i<=m;i++){dp[i]=max(dp[i-1],dp[i]);cout<<dp[i]<<" ";}return 0;
}

D 智乃想考一道完全背包(Hard version)

思路:

朴素的想法是设 d p [ k ] [ j ] dp[k][j] dp[k][j] 表示位置 k k k 为中心,容量为 j j j 的最大价值。直接枚举 k k k 的位置,对每个位置做法和上面相同,这样时间复杂度就会变成 O ( n ∗ m 3 ) O(n*m^3) O(nm3),会 T T T 1 3 \frac13 31 的点。

考虑优化,发现对于选取了区间 l ∼ r l\sim r lr 的整块物品,它可以去更新 k ∈ [ l , r ] k\in[l,r] k[l,r] 的所有位置,而没有必要在选取不同 k k k 的时候分别去更新每个 d p [ k ] [ − ] dp[k][-] dp[k][]

原本我们先从小到大枚举 l l l,固定好 l l l 后,然后枚举 r r r k k k,对一个位置 k k k,它应该用 [ l , r = k ∼ n ] [l,r=k\sim n] [l,r=kn] 的物品来更新。这里为了实现上面说的优化,我们从大到小枚举 r r r,并且令 k = r k=r k=r,这样算出来的 d p [ r ] [ − ] dp[r][-] dp[r][] 就考虑到了 [ l , r = k ∼ n ] [l,r=k\sim n] [l,r=kn] 所有物品。 d p [ r ] [ − ] dp[r][-] dp[r][] 直接从 d p [ r + 1 ] [ − ] dp[r+1][-] dp[r+1][] 转移过来或者从 [ l , r ] [l,r] [l,r] 这件物品转移过来即可

这样优化后,由于不需要枚举 k k k 的位置,时间复杂度被优化成了 O ( m 3 ) O(m^3) O(m3)。可以通过。

code:

#include <iostream>
#include <cstdio>
using namespace std;
const int maxn=2005;
const int maxm=505;
typedef long long ll;
const ll linf=1e18;int n,m;
ll dp[maxn][maxm];
//位置为k,背包容量为j ll tw[maxn],tv[maxn];
int ans[maxn],id[maxn];int main(){cin>>n>>m;for(int i=1;i<=n;i++){cin>>tw[i]>>tv[i];tw[i]+=tw[i-1];tv[i]+=tv[i-1];}for(int l=1;l<=n;l++){ll w,v;for(int r=n;r>=l;r--){w=tw[r]-tw[l-1];v=tv[r]-tv[l-1];for(int j=w;j<=m;j++)dp[r][j]=max(dp[r][j],max(dp[r+1][j],dp[r][j-w]+v));}}for(int i=1;i<=m;i++){ll maxx=-linf,idx;for(int k=1;k<=n;k++){dp[k][i]=max(dp[k][i-1],dp[k][i]);if(dp[k][i]>maxx){maxx=dp[k][i];idx=k;}}ans[i]=maxx;id[i]=idx;}for(int i=1;i<=m;i++)cout<<ans[i]<<" \n"[i==m];for(int i=1;i<=m;i++)cout<<id[i]<<" \n"[i==m];return 0;
}

这篇关于牛客练习赛123(A,B,C,D)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/864156

相关文章

你的华为手机升级了吗? 鸿蒙NEXT多连推5.0.123版本变化颇多

《你的华为手机升级了吗?鸿蒙NEXT多连推5.0.123版本变化颇多》现在的手机系统更新可不仅仅是修修补补那么简单了,华为手机的鸿蒙系统最近可是动作频频,给用户们带来了不少惊喜... 为了让用户的使用体验变得很好,华为手机不仅发布了一系列给力的新机,还在操作系统方面进行了疯狂的发力。尤其是近期,不仅鸿蒙O

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数

牛客小白月赛100部分题解

比赛地址:牛客小白月赛100_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ A.ACM中的A题 #include<bits/stdc++.h>using namespace std;#define ll long long#define ull = unsigned long longvoid solve() {ll a,b,c;cin>>a>>b>

牛客小白月赛100(A,B,C,D,E,F三元环计数)

比赛链接 官方讲解 这场比较简单,ABC都很签到,D是个不太裸需要预处理的 B F S BFS BFS 搜索,E是调和级数暴力枚举,F是三元环计数。三元环考的比较少,没见过可能会偏难。 A ACM中的A题 思路: 就是枚举每个边变成原来的两倍,然后看看两短边之和是否大于第三边即可。 不能只给最短边乘 2 2 2,比如 1 4 8 这组数据,也不能只给第二短边乘 2 2 2,比

笔试强训,[NOIP2002普及组]过河卒牛客.游游的水果大礼包牛客.买卖股票的最好时机(二)二叉树非递归前序遍历

目录 [NOIP2002普及组]过河卒 牛客.游游的水果大礼包 牛客.买卖股票的最好时机(二) 二叉树非递归前序遍历 [NOIP2002普及组]过河卒 题里面给的提示很有用,那个马的关系,后面就注意,dp需要作为long的类型。 import java.util.Scanner;// 注意类名必须为 Main, 不要有任何 package xxx 信息publ

每日OJ_牛客_求和(递归深搜)

目录 牛客_求和(递归深搜) 解析代码 牛客_求和(递归深搜) 求和_好未来笔试题_牛客网 解析代码         递归中每次累加一个新的数,如果累加和大于等于目标,结束递归。此时如果累加和正好等于目标,则打印组合。向上回退搜索其它组合。此题本身就是一个搜索的过程,找到所有的组合。 #include <iostream>#include <cmath>#in

RC4加密解密算法123

RC4是一种对称密码算法,它属于对称密码算法中的序列密码(streamcipher,也称为流密码),它是可变密钥长度,面向字节操作的流密码。 RC4是流密码streamcipher中的一种,为序列密码。RC4加密算法是Ron Rivest在1987年设计出的密钥长度可变的加密算法簇。起初该算法是商业机密,直到1994年,它才公诸于众。由于RC4具有算法简单,运算速度快,软硬件实现都

牛客《剑指Offer》 -- 数值的整数次方

题目描述 给定一个double类型的浮点数base和int类型的整数exponent。求base的exponent次方。 思路 特别注意负数的情况,出现负数,将其转化为正数然后求倒数。 class Solution {public:double Power(double base, int exponent) {double total = 1;bool flag = false

牛客网《剑指Offer》 二进制中1的个数

题目描述 输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。 思路 负数用补码,其实就是求一个数据在计算机中是存储是怎么样子的。用位运算,就能很好实现。 class Solution {public:int NumberOf1(int n) {int count = 0;int flag = 1;while (flag != 0) {if ((n & f

牛客网《剑指Offer》 矩形覆盖

题目描述 我们可以用2*1的小矩形横着或者竖着去覆盖更大的矩形。请问用n个2*1的小矩形无重叠地覆盖一个2*n的大矩形,总共有多少种方法? class Solution {public:int rectCover(int number) {if(number==0) return 0;if(number==1) return 1;if(number==2) return 2;retu