Bestcoder Round 47 && 48
1.Senior's Array(hdu 5280)
题目大意:给出大小为N的数组和P,求将数组中的某个元素替换为P后的最大连续子段和。N<=1000
题解:
1.送分题,比赛的时候只想到枚举替换的元素然后贪心找最大连续子段和.时间复杂度O(N2)
2.实际上有更好的做法。类似线段树的合并,枚举替换元素的时候 可以把左右两边的答案合并.F[i]表示以i结尾,G[i]表示以i开头的最优解.合并的时候只要考虑把中间的P用起来,即
A[x]+max(0,F[x-1])+max(0,G[x+1]) . 时间复杂度O(N).
2.Senior's Gun(hdu 5281)
题目大意:给出两个大小分别为N,M的数组A,B,将这两个数组的元素配对起来,配对的得分为A[i]-B[j](只有A[i]>B[j]才可以把A[i],B[j]配对).要求得分和最大。
题解:
1.比赛的时候想到贪心地把A中大的和B中小的配对。 但是不会证明,感觉不大靠谱。然后就只A了T1 滚粗了。
2.官方题解很简洁:二分出最大可能的配对数,然后枚举配对数,取最大的答案。 这么简洁的题解像我这么弱的可能会一下反应不过来,所以来解释一下: 假设有2N个数能两两配对,那么配对的得分总和是一定的,即A的总和减去B的总和。 所以枚举了所有的配对数,也就枚举了所有的配对情况。 再提一下二分的时候怎么判断可行性:从小到大排序,然后小的和小的对,大的和大的对。 如果这样不行,那么一定不存在可行方案。 脑残的我又想来证明一下了(果然我不适合OI...OIer不是只要记住结论就好了么。。):假设存在某种方案不是这样配对的,如果A,B数组已经从小到大排序,如果A[i]和B[j]配对就在他们之间连一条线。那么至少有两条线是交叉的.显然把交叉的线 弄成不交叉 即交换配对对象 也是满足条件的。 那么重复这样的操作 最终得到我们一开始的配对方式。
3.看了下别人的代码,发现1中的方法也是可行的。如果n<m 把B中最小的n个拿出来就好。 如果n>m把A中最大的m个拿出来就好。 所以只考虑n=m的情况。
考虑A中的两个元素P,Q,P<=Q. B中的两个元素S,T,S<=T.
如果Q<S,那么不能匹配任意一个.
如果P>=T,那么可以随便怎么匹配,得分为P+Q-S-T。 可以认为是PT,QS匹配。
除去上面的两种情况,也就是Q>=S,P<T的时候。这时候一定是QS匹配最优。我们可以从赚差值的角度理解。假设如果A[i]-A[j]<0也可以匹配,但实际得分却是0,这样我们就赚了
这个负的值。所以要想尽可能多赚这个负的值,要让A中小的P和B中大的T匹配。
。。憋出个证明蛋疼死了。而且感觉还是不大完美。 其实还是感觉官方题解好理解,但大多数人貌似都是3中的方法。
3.Senior's String(hdu 5282)
题目大意:给出2个字符串AB,求它们的LCS,假设长度是len,再求A中长度为len的子序列中有多少个是B的子序列。(一开始以为是LCS的个数,仔细想想其实不是。。)
题解:
先搞出求LCS时的dp数组。然后再做一次动态规划。F[i][j]表示A[1..i]的长度为dp[i][j]的子序列中有多少个是B[1...j]的子序列。
1.考虑不选A[i],那么必须有dp[i-1][j]==dp[i][j]. F[i][j]+=F[i-1][j].
2.考虑选 A[i],设p是满足p<=j && B[p]=A[i]的最大的p. 那么必须有dp[i-1][p-1]+1=dp[i][j] . F[i][j]+=F[i-1][p-1].
感觉这个dp略吊啊。
4.Senior's Fish(hdu 5283)
题目大意:
给出坐标系上的一个矩形,和一些点(编号1-N),要求支持以下操作:
1.把编号属于区间[L,R]的点横坐标+x(x>0).
2.把编号属于区间[L,R]的点纵坐标+x(x>0).
3.查询在矩形中点的个数。
题解:
1.这题要充分利用点的坐标是单调增的这个性质。 也就是说 一个点出了矩形 就不可能再进来了。 然后可以把询问拆成4个,有多少个点在某个点左下方的询问。
2.对xy坐标分别建4棵线段树,然后如果某个坐标超过了上界,就把它从线段树里删去。 这里有个技巧,就是可能一次操作之后有多个点超过了上界,但是这些点又不是连续的,我们可以一个一个删,因为每个点只会被删去一次,复杂度有保证. 只要每次找坐标值最大的点,如果越界了就删去,并把他的值改为负无穷。 具体实现起来好像有点麻烦,要好多线段树...我就没有写..懒。
5.wyh and pupil(hdu 5285)
题目大意:给出无向图的一些边,要求把点分为2个非空集合,集合内不能有边,且第一个集合点最多。
题解:
1.首先判断一下是不是二分图,如果不是无解。
2.可以发现连通块之间是独立的,只要每个连通块里选尽量多的点放到第一个集合就好。所以对每个连通块二分图染色,选择点多的那种颜色。
3.另外考虑特殊情况,只有一个点和没有边的情况。 比赛的时候没考虑 FST了。
6.wyh and sequence(hdu 5286)
题目大意:Q次询问区间[Li,Ri]中,把所有数去重之后,设第i个数pi出现了ki次, 求sigma(pi^ki).
题解:
1.这题写了4K左右的代码...BC难得有代码量这么大的题。感觉我就算会做也不可能在1个小时之内写出来A掉。跪nodgd分块大神!!
2.太难表述。。直接copy官方题解:
3.另外这题卡常数。。极限数据我的程序过不了,但是OJ上还是AC了。由于要离散化,我一开始写的时候每次都从map里取出那个元素...结果复杂度多了个log,死都查不出来。。
贴个代码:
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <vector>
#include <map>
#include <set>
#include <cstdlib>
#include <ctime>
#include <algorithm>
using namespace std; typedef long long ll;
typedef pair<int,int> pii;
const int N=,Mod=1e9+; int n,Q,unit,tot;
int A[N],v[N],cnt[N],cur[N],ans[][],rk[N];
int S[N][],L[N],R[N],id[N],lb[],rb[];
vector<int> g[N];
map<int,int> mp; inline int Add(int x,int y){return (x+y)%Mod;}
inline int Mul(int x,int y){return 1ll*x*y%Mod;} void Init()
{
mp.erase(mp.begin(),mp.end()); tot=;
for (int i=;i<=n;i++)
{
if (!mp.count(A[i]))
{
mp[A[i]]=;
v[++tot]=A[i];
}
}
sort(v+,v+tot+);
for (int i=;i<=tot;i++) mp[v[i]]=i;
for (int i=;i<=n;i++) rk[i]=mp[A[i]]; for (int i=;i<=tot;i++) g[i].clear(),g[i].push_back(),cur[i]=;
for (int i=;i<=n;i++)
{
cur[rk[i]]=Mul(cur[rk[i]],A[i]);
g[rk[i]].push_back(cur[rk[i]]);
} unit=sqrt(n)+;
L[]=,id[]=;
for (int i=;i<=n;i++)
{
id[i]=(i-)/unit+;
L[i]=(id[i]==id[i-])? L[i-]:i;
}
R[n]=n;
for (int i=n-;i>=;i--) R[i]=(id[i]==id[i+])? R[i+]:i; for (int i=;i<=id[n];i++) lb[i]=(i-)*unit+,rb[i]=min(i*unit,n); for (int i=;i<=tot;i++) cnt[i]=;
for (int i=;i<=n;i++)
{
cnt[rk[i]]++;
if (i%unit== || i==n)
{
for (int j=;j<=tot;j++)
S[j][id[i]]=cnt[j];
}
} for (int i=;i<=id[n];i++)
{
for (int j=;j<=tot;j++) cnt[j]=;
for (int j=i;j<=id[n];j++)
{
ans[i][j]=(i==j)? :ans[i][j-];
for (int k=lb[j];k<=rb[j];k++)
{
int x=rk[k]; cnt[x]++;
ans[i][j]=Add(ans[i][j],-g[x][cnt[x]-]);
ans[i][j]=Add(ans[i][j],g[x][cnt[x]]);
ans[i][j]=Add(ans[i][j],Mod);
}
}
}
memset(cnt,,sizeof(cnt));
} int Query(int l,int r)
{
int Ans=;
if (id[r]-id[l]<=)
{
for (int i=l;i<=r;i++)
{
int x=rk[i];
Ans=Add(Ans,-g[x][cnt[x]]);
Ans=Add(Ans,g[x][++cnt[x]]);
Ans=Add(Ans,Mod);
}
for (int i=l;i<=r;i++) cnt[rk[i]]=;
}
else
{
Ans=ans[id[l]+][id[r]-];
for (int i=l;i<=R[l];i++)
{
int x=rk[i];
Ans=Add(Ans,-g[x][cnt[x]+S[x][id[r]-]-S[x][id[l]]]);
Ans=Add(Ans,g[x][++cnt[x]+S[x][id[r]-]-S[x][id[l]]]);
Ans=Add(Ans,Mod);
}
for (int i=L[r];i<=r;i++)
{
int x=rk[i];
Ans=Add(Ans,-g[x][cnt[x]+S[x][id[r]-]-S[x][id[l]]]);
Ans=Add(Ans,g[x][++cnt[x]+S[x][id[r]-]-S[x][id[l]]]);
Ans=Add(Ans,Mod); }
for (int i=l;i<=R[l];i++) cnt[rk[i]]=;
for (int i=L[r];i<=r;i++) cnt[rk[i]]=;
}
return Ans;
} void Read(int &x)
{
x=; char ch=getchar();
while (ch<'' || ch>'') ch=getchar();
while (ch>='' && ch<='') x=x*+ch-'',ch=getchar();
} int main()
{
int T,lastans,x,y,l,r;scanf("%d",&T);
while (T--)
{
scanf("%d%d",&n,&Q);
for (int i=;i<=n;i++) Read(A[i]);
Init(); lastans=;
for (int i=;i<=Q;i++)
{
Read(x),Read(y);
x^=lastans,x%=n,x++;
y^=lastans,y%=n,y++;
l=min(x,y),r=max(x,y);
lastans=Query(l,r);
printf("%d\n",lastans);
}
} return ;
}
Bestcoder Round 47 && 48的更多相关文章
- BestCoder Round #47
1001 Senior's Array 题目链接:1001 题意:给你一个长度为n的序列,你必须修改序列中的某个数为P,求修改后的最大连续子序列和. 思路:数据量比较小,可以直接暴力做, 枚举序列的每 ...
- BestCoder Round #47 1003
solution : 就按题解敲了一遍,好久没写这种dp ; ; LL f[MAX][MAX]; ]; scanf( scanf(,b+); ...
- HDU 5281 BestCoder Round #47 1002:Senior's Gun
Senior's Gun Accepts: 235 Submissions: 977 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: ...
- HDU 5280 BestCoder Round #47 1001:Senior's Array
Senior's Array Accepts: 199 Submissions: 944 Time Limit: 2000/1000 MS (Java/Others) Memory Limit ...
- 二分图判定+点染色/并查集 BestCoder Round #48 ($) 1002 wyh2000 and pupil
题目传送门 /* 二分图判定+点染色:因为有很多联通块,要对所有点二分图匹配,若不能,存在点是无法分配的,no 每一次二分图匹配时,将点多的集合加大最后第一个集合去 注意:n <= 1,no,两 ...
- BestCoder Round #14
Harry And Physical Teacher Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Ja ...
- DP BestCoder Round #50 (div.2) 1003 The mook jong
题目传送门 /* DP:这题赤裸裸的dp,dp[i][1/0]表示第i块板放木桩和不放木桩的方案数.状态转移方程: dp[i][1] = dp[i-3][1] + dp[i-3][0] + 1; dp ...
- BestCoder Round #89 02单调队列优化dp
1.BestCoder Round #89 2.总结:4个题,只能做A.B,全都靠hack上分.. 01 HDU 5944 水 1.题意:一个字符串,求有多少组字符y,r,x的下标能组成等比数列 ...
- BestCoder Round #90 //div all 大混战 一题滚粗 阶梯博弈,树状数组,高斯消元
BestCoder Round #90 本次至少暴露出三个知识点爆炸.... A. zz题 按题意copy Init函数 然后统计就ok B. 博弈 题 不懂 推了半天的SG..... 结果这 ...
随机推荐
- tomcat部署web项目的3中方法
1.直接把项目复制到Tomcat安装目录的webapps目录中,这是最简单的一种Tomcat项目部署的方法,也是初学者最常用的方法. 2.在tomcat安装目录中有一个conf文件夹,打开此文件夹,其 ...
- 最小化安装的CentOS7挂载ntfs格式的U盘
准备从系统中拷贝一些文件到U盘,插上U盘. 一.获得U盘的设备识别符 fdisk -l 啊哈,我看到了,是/dev/sdb1 二.熟练的挂载 mount /dev/sdb1 /mnt/usb Duan ...
- [CF752D]Santa Claus and a Palindrome(优先队列,贪心乱搞)
题目链接:http://codeforces.com/contest/752/problem/D 题意:给长度为k的n个字符串,每一个字符串有权值,求构造一个大回文串.使得权值最大. 因为字符串长度都 ...
- ahk之路:利用ahk在window7下实现窗口置顶
操作系统:win7 64位 ahk版本:autohotkey_L1.1.24.03 今天安装了AutoHotkey_1.1.24.03.SciTE.PuloversMacroCreator,重新开始我 ...
- codeforces 85D D. Sum of Medians 线段树
D. Sum of Medians time limit per test 3 seconds memory limit per test 256 megabytes input standard i ...
- css3箭头效果
css3 record1 尝试用css写了个箭头效果 思路就是通过span和span子元素i分别通过设置他们的伪元素构造两个箭头,但是i构造的箭头两条线height都是0,hover的时候渐近的动画效 ...
- 物理Data Guard的日常维护
关闭顺序 1. 首先关闭primary数据库 [oracle@Master admin]$ sqlplus / as sysdba SQL> shutdown immediate 2. 关闭st ...
- 【java基础】成员变量和局部变量02
成员变量和局部变量(2) 成员变量的初始化和内存中的运行机制 当系统加载类或者创建类的实例的时候,系统会自动为成员变量分配内存空间,并自动指定初始值. package object; ...
- centos7 升级内核到最新版本
centos7 从问世以来,官网提供的镜像始终是3.10 版本,该版本最大的一个问题是对硬件驱动(尤其是无线网卡)的支持不是很好,本人亲测>5种机型,无线网卡均无法正常使用,如果是非主流机型,手 ...
- java怎么定义一个二维数组?
java中使用 [][] 来定义二维数组 定义数组时也可同时初始化下面是一些例子float[][] numthree; //定义一个float类型的2维数组numthree=new float[5][ ...