A. Important Exam

水题

#include<iostream>
#include<string.h>
#include<algorithm>
#include<stdio.h>
using namespace std;
const int maxx = ;
char a[maxx][maxx];
int pre[maxx];
int b[maxx];
int main(){
int n,m;
while(~scanf("%d%d",&n,&m)){
for (int i=;i<n;i++){
scanf("%s",a[i]);
}
for (int i=;i<m;i++){
scanf("%d",&b[i]);
}
for (int i=;i<m;i++){
int sum[]={,,,,};
for (int j=;j<n;j++){
if (a[j][i]=='A'){
sum[]++;
}else if (a[j][i]=='B'){
sum[]++;
}else if (a[j][i]=='C'){
sum[]++;
}else if (a[j][i]=='D'){
sum[]++;
}else {
sum[]++;
}
}
int maxx=;
for (int i=;i<=;i++){
maxx=max(maxx,sum[i]);
}
pre[i]=maxx;
}
long long ans=;
for (int i=;i<m;i++){
ans+=(long long)pre[i]*b[i];
}
printf("%lld\n",ans);
}
return ;
}

B. Zero Array

这道题就非常难受了

题意就是说,你有一个序列,你每次可以选择两个数,使得这两个数的值减小1。

最开始不知道为啥,脑子短路了,我发现只要把最大的减去最小的,然后用剩下的去减次小的,用剩下的继续减去

而且找不到反例,后面同学讨论一波,同学提出了一个2,2,2的样例,瞬间把我的结论推翻了。

后面也发现了我的结论的不正确,因为我每次把最大的减去次大的,那么不可避免的产生了一个新的数,这个数的大小我们没法判断,如果这个值过小,但是后面还有一个比这个值大的数。其实就不正确的。

其实你可以这样想,既然要求有没有可能,我们的顺序最优秀的,我们发现,其实每次把最高的降到次高的即可。这样我们保证一定是最有效的。但是有两个特殊情况,一个是数的总和是奇数那么无论怎么样

都会剩下一个。还有就是最高的太高了,用其他的所有都不能降到0,特判这两种情况即可

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define LL long long
using namespace std;
const int maxx = 1e5+;
int a[maxx];
LL sum;
int main(){
int n;
while(~scanf("%d",&n)){
sum=;
for (int i=;i<=n;i++){
scanf("%d",&a[i]);
sum+=a[i];
}
sort(a+,a++n);
if (sum%== && sum-a[n]>=a[n]){
printf("YES\n");
}else {
printf("NO\n");
}
}
return ;
}

C. Maximum Median

每次对一个数+1,一共可以进行K次,问最大中位数是多少。

二分答案即可

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<vector>
#define LL long long
using namespace std;
const int maxx = ;
LL a[maxx];
LL n,k;
bool check(LL x){
LL ans=;
for (int i=(n+)/;i<=n;i++){
if (a[i]<x){
ans=(LL)ans+x-a[i];
}
}
if (ans<=k)return true;
else return false;
}
int main(){
while(~scanf("%lld%lld",&n,&k)){
for (int i=;i<=n;i++){
scanf("%lld",&a[i]);
}
sort(a+,a++n);
LL l=a[(n+)/],r=3e9;
LL ans=l;
while(l<=r){
LL mid=(l+r)/;
if (check(mid)){
ans=mid;
l=mid+;
}else {
r=mid-;
}
}
printf("%lld\n",ans);
}
return ;
}

D. Treasure Hunting

好题啊,这个题真的很秀,当时没看到这个题,后面看别人的代码+题解,懂了

题目意思是,给一些的坐标点,坐标点上会有宝藏,你可以在每一行中随意移动,并在指定的列中往下移动,问如何移动能收集所有的宝藏,同时使得移动的步数最少

你可以发现以下结论

1.我们在每一层搜集完了所有的宝藏后,应该尽快往下一层走。搜集完所有的时刻,一定是在某一个边界位置,可能是左边界,也有可能是右边界

2.我们每次下到新的一层,一定是从边界位置开始(因为那个时候刚好收集完成),向左走,找到最近的向下走的通道,或者向右走,找到最近向下走的通道

3.我们到达新的一层后,需要判断是否这一层有宝藏

4.如果有宝藏,我们需要考虑,是先往左走搜集完左边的,然后再向右走收集到最右边的,还是先向右走搜集完右边的,然后向左走搜集完最左边的

有了这个,我们的DP就非常好写了

转移方程为

dp[i][0]表示收集完第i层,最后在最左边

dp[i][1]表示收集完第i层,最后在最右边

用trans()计算从第j层边界到i层边界,水平移动的步数。

那么转移方程可以写为:

    dp[][]=abs(maxn[][]-)+abs(maxn[][]-maxn[][]);
//从起点开始,走到最右边,再走回来
dp[][]=abs(maxn[][]-);
//从起点开始,走到最右边
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);

前两个都是计算初始值

四个分别表示

从第j层左边界最后到第i层左边边界,搜集完第i层所需要的步数

从第j层左边界最后到第j层右边边界,搜集完第i层所需要的步数

从第j层右边界最后到第j层左边边界,搜集完第i层所需要的步数

从第j层右边界最后到第j层右边边界,搜集完第i层所需要的步数

最后在找边界的从左右两边找到最近的通道,用二分查找即可。

#include<iostream>
#include<algorithm>
#include<string.h>
#include<stdio.h>
#include<vector>
#define LL long long
#define rep(i,j,k) for(int i=j;i<=k;i++)
#define per(i,j,k) for(int i=j;i>=k;i--)
#define pb push_back
#define pii pair<int,int>
#define mp make_pair
using namespace std;
using namespace std;
const int maxx = 2e5+;
const int INF = 1e9;
LL maxn[maxx][];
LL dp[maxx][];
int n,m,k,q;
int b[maxx];
LL trans(int u,int du,int v,int dv){
int p=lower_bound(b+,b++q,maxn[u][du])-b;
LL rt=INF;
//找到第一个小于maxn[u][du]
if(p<=q){
rt=abs(maxn[u][du]-b[p])+abs(maxn[v][dv^]-b[p])+abs(maxn[v][dv]-maxn[v][dv^]);
//我们找到abs(maxn[u][du]-b[p])+abs(maxn[v][dv^1]-b[p])+abs(maxn[v][dv]-maxn[v][dv^1])了一条路后,水平消耗的路程,应该是出发点的通路的距离,以及到达目的层后,目标方向的反向的最远距离,以及目标防方向的最远距离
}
p=upper_bound(b+,b++q,maxn[u][du])-b-;
if(p){
rt=min(rt,abs(maxn[u][du]-b[p])+abs(maxn[v][dv^]-b[p])+abs(maxn[v][dv]-maxn[v][dv^]));
}
return rt;
}
int main(){
while(~scanf("%d%d%d%d",&n,&m,&k,&q)){
rep(i,,n){
maxn[i][]=INF;
maxn[i][]=-INF;
}
LL x,y;
rep(i,,k){
scanf("%lld%lld",&x,&y);
maxn[x][]=min(maxn[x][],y);
maxn[x][]=max(maxn[x][],y);
}
maxn[][]=;
maxn[][]=max(maxn[][],1LL);
rep(i,,q){
scanf("%d",&b[i]);
}
sort(b+,b++q);
memset(dp,0x3f,sizeof(dp));
dp[][]=abs(maxn[][]-)+abs(maxn[][]-maxn[][]);
//从起点开始,走到最右边,再走回来
dp[][]=abs(maxn[][]-);
//从起点开始,走到最右边
int j=;
rep(i,,n){
if (maxn[i][]==INF)continue;
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
dp[i][]=min(dp[i][],dp[j][]+trans(j,,i,)+i-j);
j=i;
}
printf("%lld\n",min(dp[j][],dp[j][]));
}
return ;
}

Codeforces Round #577 (Div 2)的更多相关文章

  1. Codeforces Round #577 (Div. 2) D. Treasure Hunting

    Codeforces Round #577 (Div. 2)  D. Treasure Hunting 这个一场div2 前面三题特别简单,这个D题的dp还是比较难的,不过题目告诉你了只能往上走,所以 ...

  2. 矩阵拿宝物--Codeforces 1201D - Treasure Hunting Codeforces Round #577 (Div. 2)

    网上题解比较少,自己比较弱研究了半天(已经过了),希望对找题解的人有帮助 题目链接:https://codeforc.es/contest/1201/problem/D 题意: 给你一个矩形,起始点在 ...

  3. Codeforces Round #577 (Div. 2) 题解

    比赛链接:https://codeforc.es/contest/1201 A. Important Exam 题意:有\(n\)个人,每个人给出\(m\)个答案,每个答案都有一个分值\(a_i\), ...

  4. Codeforces Round #577 (Div. 2) C. Maximum Median (模拟,中位数)

    题意:给你一个长度为奇数\(n\)的序列.你可以对任意元素加上\(k\)次\(1\),求操作后的中位数最大. 题解:先对序列进行排序,然后对中位数相加,如果中位数和后面的元素相等,就对后面所有和当前中 ...

  5. Codeforces Round #577 (Div. 2) C. Maximum Median

    题意:就是给一n(奇数)个元素数组,可以对它的元素执行k次+1操作,递增排序,求中位数最大是多少. 那我们在排完序之后,中位数前的元素可以不管它,只要对中位数后的操作就行,我们要判断和中位数相等的元素 ...

  6. Codeforces Round #366 (Div. 2) ABC

    Codeforces Round #366 (Div. 2) A I hate that I love that I hate it水题 #I hate that I love that I hate ...

  7. Codeforces Round #354 (Div. 2) ABCD

    Codeforces Round #354 (Div. 2) Problems     # Name     A Nicholas and Permutation standard input/out ...

  8. Codeforces Round #368 (Div. 2)

    直达–>Codeforces Round #368 (Div. 2) A Brain’s Photos 给你一个NxM的矩阵,一个字母代表一种颜色,如果有”C”,”M”,”Y”三种中任意一种就输 ...

  9. cf之路,1,Codeforces Round #345 (Div. 2)

     cf之路,1,Codeforces Round #345 (Div. 2) ps:昨天第一次参加cf比赛,比赛之前为了熟悉下cf比赛题目的难度.所以做了round#345连试试水的深浅.....   ...

随机推荐

  1. 洛谷P1006 NOIP提高组2008 传纸条

    P1006 传纸条 题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n 列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无 ...

  2. 【CodeVS】2822 爱在心中 [2017年6月计划 强连通分量03]

    2822 爱在心中 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond         题目描述 Description “每个人都拥有一个梦,即使彼此不相同,能够 ...

  3. 【模板】Tarjan缩点,强连通分量 洛谷P2341 [HAOI2006]受欢迎的牛 [2017年6月计划 强连通分量01]

    P2341 [HAOI2006]受欢迎的牛 题目描述 每头奶牛都梦想成为牛棚里的明星.被所有奶牛喜欢的奶牛就是一头明星奶牛.所有奶 牛都是自恋狂,每头奶牛总是喜欢自己的.奶牛之间的“喜欢”是可以传递的 ...

  4. SQLyog12.0.9下载、安装和破解

    原文转载连接:https://blog.csdn.net/lihua5419/article/details/73881837/ sqlyog百度云链接(永久有效):http://pan.baidu. ...

  5. selenium(4):初次尝试,通过百度进行搜索

    实现场景:打开chrome浏览器后,打开百度,再搜索栏里输入‘测试’,点击搜索按钮. 代码:定位方式,通过元素的ID. 定位技巧: ①鼠标定位需要定位的输入框,鼠标右键单击.选择检查. ②即可轻松的查 ...

  6. Hdu 1156

    题目链接 Brownie Points II Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  7. Codeforces 113C

    题目链接 C. Double Happiness time limit per test 5 seconds memory limit per test 128 megabytes input sta ...

  8. jquery 设置 html标签响应式布局

    function sWidth() {//计算当前设备宽度 var widthSize; if ($(window).width() <= 640) { widthSize = $(window ...

  9. POJ 1150 The Last Non-zero Digit 数论+容斥

    POJ 1150 The Last Non-zero Digit 数论+容斥 题目地址: id=1150" rel="nofollow" style="colo ...

  10. element-ui表格列金额显示两位小数

    对于金额的显示,大多情况下需要保留两位小数,比如下面的(表格采用 element-ui): 在vue.js中,对文本的处理通常是通过设置一系列的过滤器,过滤器可以用在两个地方:双花括号插值 和 v-b ...