链接:https://www.nowcoder.com/acm/contest/131/B
来源:牛客网

矩阵 M 包含 R 行 C 列,第 i 行第 j 列的值为 Mi,j
请寻找一个子矩阵,使得这个子矩阵的和最大,且满足以下三个条件:
子矩阵的行数不能超过 X 行。
子矩阵的列数不能超过 Y 列。
子矩阵中 0 的个数不能超过 Z 个。
请输出满足以上条件的最大子矩阵和。

输入描述:

第一行输入五个整数 R,C,X,Y,Z。
接下来 N 行,每行输入 M 个整数,第 i 行第 j 列的整数表示 Mi,j
1 ≤ R,C ≤ 500.
1 ≤ X ≤ R.
1 ≤ Y ≤ C.
1 ≤ Z ≤ R x C. |Mij|<=1e9

输出描述:

输出满足以上条件的最大子矩阵和。

考虑枚举行数。
枚举行数后枚举从哪行开始。
预处理出来纵向前缀和,然后枚举的时候就把每一个元素转化成柱形图那样的格式,然后求最大值就好了。 求最大值时的思想是枚举最后一个元素,然后看前面的元素,前面元素大于0直接放,小于0的就把队尾取出来,加到这个元素,直到不小于0或者队列为空位置。
以数结尾的最大值是目前队列元素里面的和。 用单调栈也可以做。
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std;
const int N=;
int q[N],p[N],zt[N][N],L[N];
long long mp[N][N],up[N][N],a[N];
int R,C,X,Y,Z;
long long work(){
for(int i=;i<=C;++i) p[i]=i;
int h=,r=;
long long ans=,sum=;
q[]=;
for(int i=;i<=C;++i)
{
while(h<=r&&p[q[h]]<=i-Y) sum-=a[q[h++]];
while(h<=r&&a[i]<) {
sum-=a[q[r]];
a[i]+=a[q[r]];
p[i]=p[q[r--]];
}
if(a[i]>) sum+=a[i],q[++r]=i;
while(h<=r&&(L[i]-L[p[q[h]]]>Z)) sum-=a[q[h++]];
ans=max(ans,sum);
}
return ans;
}
int main(){
long long ans=;
scanf("%d%d%d%d%d",&R,&C,&X,&Y,&Z);
for(int i=;i<=R;++i) for(int j=;j<=C;++j) scanf("%lld",&mp[i][j]);
for(int i=;i<=R;++i) for(int j=;j<=C;++j) up[i][j]=up[i-][j]+mp[i][j],zt[i][j]=zt[i-][j]+(mp[i][j]==);
for(int len=;len<=X;++len){
for(int i=;i<=R-len+;++i){
for(int j=;j<=C;++j) a[j]=up[i+len-][j]-up[i-][j],L[j]=zt[i+len-][j]-zt[i-][j];
for(int j=;j<=C;++j) L[j]+=L[j-];
ans=max(ans,work());
}
}
printf("%lld\n",ans);
}

链接:https://www.nowcoder.com/acm/contest/131/C
来源:牛客网

题目描述

有一棵树包含 N 个节点,节点编号从 1 到 N。节点总共有 K 种颜色,颜色编号从 1 到 K。第 i 个节点的颜色为 Ai
Fi 表示恰好包含 i 种颜色的路径数量。请计算:

输入描述:

第一行输入两个正整数 N 和 K,N 表示节点个数,K 表示颜色种类数量。
第二行输入 N 个正整数,A

i

表示第 i 个节点的颜色。
接下来 N - 1 行,第 i 行输入两个正整数 Ui 和 Vi,表示节点 Ui 和节点 Vi 之间存在一条无向边,数据保证这 N-1 条边连通了 N 个节点。
1 ≤ N ≤ 50000.
1 ≤ K ≤ 10.
1 ≤ Ai ≤ K.

输出描述:

输出一个整数表示答案。

题解:T[i]表示集合意义上小于等于i的路径的总数,状态总数(i的范围)为2^k。
求T[i]只要把i集合中的点拿出来做联通块,每个联通块的价值就是联通块中点的数目+C (点的数目,2)
然后求每个状态的点的数量,只需要让他减去他的所有真子集即可。
然后转态再转为数量就可以了。
#include<vector>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=5e4+;
const int P=1e9+;
vector<pair<int,int> >G;
int f[N],sz[N],n,k,a[N];
int T[],tran[],F[];
int findx(int x){
return (x==f[x])?x:(f[x]=findx(f[x]));
}
int main(){
scanf("%d%d",&n,&k);
for(int i=;i<=n;++i) scanf("%d",a+i),--a[i];
for(int i=;i<n;++i) {
int x,y;
scanf("%d%d",&x,&y);
G.push_back(make_pair(x,y));
}
for(int i=;i<;++i) {
int x,y,fx,fy;
for(int j=;j<=n;++j) f[j]=j,sz[j]=;
for(int j=;j<n-;++j) {
x=G[j].first,y=G[j].second;
if(((<<a[x])&i)&&((<<a[y])&i)) {
fx=findx(x),fy=findx(y);
sz[fx]+=sz[fy];
f[fy]=fx;
}
}
for(int j=;j<=n;++j) if(((<<a[j])&i)&&(f[j]==j)){
T[i]=(T[i]+sz[j]+(1LL*sz[j]*(sz[j]-)/)%P)%P;
}
}
for(int i=;i<;++i) for(int j=;j<i;++j)
if(!((i^j)&j))
{
T[i]=(T[i]-T[j]+P)%P;
}
for(int i=;i<;++i) for(int j=;j<;++j)
if(i&(<<j)) ++tran[i];
for(int i=;i<;++i) F[tran[i]]=(F[tran[i]]+T[i])%P;
int x=,ans=;
for(int i=;i<=k;++i){
ans=(ans+1LL*F[i]*x%P)%P;
x=131LL*x%P;
}
printf("%d\n",ans);
}

链接:https://www.nowcoder.com/acm/contest/131/F
来源:牛客网

ZZT 得到了一个字符串 S 以及一个整数 K。
WZH 在 1995 年提出了“优雅 K 串”的定义:这个字符串每一种字符的个数都是 K 的倍数。
现在 ZZT 想要对字符串进行 Q 次询问,第 i 次询问给出一个区间 [Li, Ri],他想计算 [Li, Ri] 中有多少个子串是“优雅 K 串”。
由于 ZZT 忙于工作,所以他把这个问题交给了你,请你帮忙解决。

输入描述:

第一行输入一个正整数 K。
第二行输入一个字符串 S。
第三行输入一个正整数 Q,表示有 Q 次询问。
接下来 Q 行,每行输入两个正整数 Li 和 Ri,表示第 i 次询问。
1 ≤ K ≤ 50.
1≤ | S | ≤ 3 x 104 且 S 仅包含小写英文字母.
1≤ Q ≤ 3 x 104.
1 ≤ Xi ≤ Yi ≤ N.

输出描述:

每次询问,输出一个正整数,表示满足条件的“优雅 K 串”的数量。

输入

复制

1
abc
3
1 3
1 2
2 3

输出

复制

6
3
3 题解:先求前缀和,然后可知前缀和相同的左开右闭区间就是所求区间(在模k意义下),所以每个询问的l要先减掉1。
然后前缀和转哈希,对每一个点赋予哈希后的权值。
然后就莫队就好了。
也可以用字典树写。
#include<map>
#include<iostream>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=3e4+;
int now,Be[N],ans[N];
int pp[N][];
unsigned long long a[N]; struct Query{
int l,r,id;
bool operator < (const Query &A)const{
return Be[l]==Be[A.l]?r<A.r:l<A.l;
}
}q[N];
char s[N];
int k,l,r;
map<unsigned long long,int> M;
void work(int x,int y){
if(y==) now+=M[a[x]]++;
else now-=--M[a[x]];
}
int main(){
int m,len,unit;
now=;
scanf("%d %s",&k,s+);
len=strlen(s+);
//M[0]=1;
for(int i=;s[i];++i) {
for(int j=;j<;++j) pp[i][j]=pp[i-][j];
pp[i][s[i]-'a']=(pp[i][s[i]-'a']+)%k;
}
for(int i=;s[i];++i) {
unsigned long long pt=;
for(int j=;j<;++j) pt=pt*+pp[i][j];
a[i]=pt;
}
unit=sqrt(len);
for(int i=;i<=len;++i) Be[i]=i/unit+;
scanf("%d",&m);
for(int i=;i<=m;++i) scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i,--q[i].l;
sort(q+,q+m+);
l=q[].l,r=q[].l-;
for(int i=;i<=m;++i) {
while(l<q[i].l) work(l++,-);
while(l>q[i].l) work(--l,);
while(r<q[i].r) work(++r,);
while(r>q[i].r) work(r--,-);
ans[q[i].id]=now;
}
for(int i=;i<=m;++i) {
printf("%d\n",ans[i]);
}
}


牛客网挑战赛19 B,C,F的更多相关文章

  1. 牛客~~wannafly挑战赛19~A 队列

    链接:https://www.nowcoder.com/acm/contest/131/A来源:牛客网 题目描述 ZZT 创造了一个队列 Q.这个队列包含了 N 个元素,队列中的第 i 个元素用 Qi ...

  2. 牛客网挑战赛24 青蛙(BFS)

    链接:https://www.nowcoder.com/acm/contest/157/E来源:牛客网 有一只可爱的老青蛙,在路的另一端发现了一个黑的东西,想过去一探究竟.于是便开始踏上了旅途 一直这 ...

  3. 题解——牛客网Wannafly挑战赛23 B-游戏 (SG函数)

    前言 比赛的时候没学过SG函数的蒟蒻以为是道结论题,但是不是QwQ 和dummyummy巨佬一起推了快三个小时的规律 最后去问了真正的巨佬__stdcall __stdcall面带微笑的告诉我们,这是 ...

  4. 牛客网 牛客小白月赛1 F.三视图

    F.三视图   链接:https://www.nowcoder.com/acm/contest/85/F来源:牛客网     这个题自己想一下三维的,正视图和左视图中y轴为行数,x轴和z轴是列数,因为 ...

  5. 牛客网练习赛23 F 托米的游戏

    链接:https://www.nowcoder.com/acm/contest/156/F 来源:牛客网 题目描述 题目背景编不下去了 托米有一棵有根树 T, 树根为1,每轮他会在剩下的子树中等概率一 ...

  6. 牛客网——F小牛再战(博弈,不懂)

    链接:https://www.nowcoder.net/acm/contest/75/F来源:牛客网 题目描述 共有N堆石子,已知每堆中石子的数量,两个人轮流取石子,每次只能选择N堆石子中的一堆取一定 ...

  7. 牛客网——F求最大值

    链接:https://www.nowcoder.net/acm/contest/29/F来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32768K,其他语言65536K ...

  8. 牛客网 牛客练习赛43 F.Tachibana Kanade Loves Game-容斥(二进制枚举)+读入挂

    链接:https://ac.nowcoder.com/acm/contest/548/F来源:牛客网 Tachibana Kanade Loves Game 时间限制:C/C++ 1秒,其他语言2秒 ...

  9. 牛客网暑期ACM多校训练营(第五场):F - take

    链接:牛客网暑期ACM多校训练营(第五场):F - take 题意: Kanade有n个盒子,第i个盒子有p [i]概率有一个d [i]大小的钻石. 起初,Kanade有一颗0号钻石.她将从第1到第n ...

随机推荐

  1. 如何在Spring boot中修改默认端口

    文章目录 介绍 使用Property文件 在程序中指定 使用命令行参数 值生效的顺序 如何在Spring boot中修改默认端口 介绍 Spring boot为应用程序提供了很多属性的默认值.但是有时 ...

  2. Java初学者最近三次作业的心得体会

    作为一个初学者,简单的谈一下自己的作业心得体会.如果你是完全没有接触过Java的学习,本篇博文可能会有些收获,如果你已经学习Java有一段时间了,那么可以放弃这篇文章了,因为这篇文章讲解的是基本的东西 ...

  3. 什么是.pyc文件

    1. Python是一门解释型语言? Python是一门解释性语言,我就这样一直相信下去,直到发现了*.pyc文件的存在. 如果是解释型语言,那么生成的*.pyc文件是什么呢?c应该是compiled ...

  4. element-ui 通用表单封装及VUE JSX应用

    一.存在及需要解决的问题 一般在做后台OA的时候会发现表单重复代码比较多,且逻辑基本一样,每次新加一个表单都需要拷贝基本一致的代码结构,然后只是简单地修改对应的字段进行开发 二.预期结果 提取重复的表 ...

  5. Vue学习—— Vuex学习笔记

    组件是Vue最强大的功能之一,而组件实例的作用域是相互独立的,意味着不同组件之间的数据是无法相互使用.组件间如何传递数据就显得至关重要,这篇文章主要是介绍Vuex.尽量以通俗易懂的实例讲述这其中的差别 ...

  6. C++编程入门题目--No.3

    题目:一个整数,它加上100后是一个完全平方数,再加上168又是一个完全平方数,请问该数是多少? 程序分析: 在10万以内判断,先将该数加上100后再开方,再将该数加上268后再开方,如果开方后 的结 ...

  7. 数学--数论--Hdu 1452 Happy 2004(积性函数性质+和函数公式+快速模幂+乘法逆元)

    Consider a positive integer X,and let S be the sum of all positive integer divisors of 2004^X. Your ...

  8. Jenkins 构建 Jmeter 项目之源代码管理(SVN)

    1.查看项目创建中是否又 svn 插件,没有的话下载插件 subversion 2.配置 svn 源代码管理,如下图(testcases 目录下包含 build.xml 和脚本文件) 3.查看 Jen ...

  9. undef用法

    #undef的语法 定义:#undef 标识符,用来将前面定义的宏标识符取消定义. 整理了如下几种#undef的常见用法. 1. 防止宏定义冲突在一个程序块中用完宏定义后,为防止后面标识符冲突需要取消 ...

  10. 跟哥一起学python(4)- 数据类型之Number

    本节我们开始学习python的数据类型. 什么是数据类型呢?前面我们提过,所谓的编程,就是控制一系列的数据去完成我们预设的逻辑或者功能.所以,编程语言首先要定义一系列对“数据”的处理规则.这些处理规则 ...