Codeforces Round #371 (Div. 1)
A:
题目大意:
在一个multiset中要求支持3种操作:
1.增加一个数
2.删去一个数
3.给出一个01序列,问multiset中有多少这样的数,把它的十进制表示中的奇数改成1,偶数改成0后和给出的01序列相等(比较时如果长度不等各自用0补齐)
题解:
1.我的做法是用Trie数来存储,先将所有数用0补齐成长度为18位,然后就是Trie的操作了。
2.官方题解中更好的做法是,直接将每个数的十进制表示中的奇数改成1,偶数改成0,比如12345,然后把它看成二进制数10101,还原成十进制是21,然后cnt[21]++,求答案的时候只要把01序列看成二进制数,然后转换成十进制x,cnt[x]就是答案。
我的代码(Trie做法)
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
#include <cstdlib>
#include <set>
using namespace std; #define X first
#define Y second
#define N 1000010
typedef long long ll;
typedef pair<int,int> pii; int n,tot=;
int cnt[N],ch[N][]; void Add(int x,ll val,int k,int op)
{
cnt[x]+=op;
if (k==) return;
int v=(val%)&;
if (!ch[x][v]) ch[x][v]=++tot;
Add(ch[x][v],val/,k-,op);
} int Query(int x,ll val,int k)
{
if (k==) return cnt[x];
int v=(val%)&;
if (!ch[x][v]) return ;
return Query(ch[x][v],val/,k-);
} int main()
{
//freopen("in.in","r",stdin);
//freopen("out.out","w",stdout); scanf("%d",&n);
char op; ll x;
for (int i=;i<=n;i++)
{
scanf(" %c%I64d",&op,&x);
if (op=='+') Add(,x,,);
if (op=='-') Add(,x,,-);
if (op=='?') printf("%d\n",Query(,x,));
}
return ;
}
B:
题目大意:
在一个n*n的矩形中,有2个没有公共边也不重叠的矩形,然后最多可以询问200次,每次询问一个矩形中有多少个矩形(会给出答案0或1或2),要求找到这两个矩形的位置。
题解:
1.如果只有一个矩形,显然可以用二分确定它的坐标。所以基本思想是找到一条分界线 将这两个矩形分开,然后各自独立寻找位置。
2.对于这两个矩形,要么是左右关系,要么是上下关系。如果是左右关系,我们先确定询问的左边界,上边界和下边界,然后逐步增大右边界,那么矩形的个数肯定是从0到1到2递增的,所以可以二分出这个 分界线。 上下关系同理。 具体实现看代码。
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
#include <cstdlib>
#include <set>
using namespace std; #define X first
#define Y second
#define Mod 1000000007
#define N 100010
typedef long long ll;
typedef pair<int,int> pii; int xx1,yy1,xx2,yy2,X1,Y1,X2,Y2; int Check(int x1,int y1,int x2,int y2)
{
printf("? %d %d %d %d\n",x1,y1,x2,y2);
fflush(stdout);
int x; scanf("%d",&x);
return x;
} void Solve(int l,int d,int r,int u)
{
int L,R,Mid,x1,y1,x2,y2; L=d,R=u;
while (L<R)
{
Mid=(L+R)>>;
if (Check(l,d,r,Mid)==) L=Mid+;
else R=Mid;
}
y2=L; L=d,R=u;
while (L<R)
{
Mid=(L+R+)>>;
if (Check(l,Mid,r,u)==) R=Mid-;
else L=Mid;
}
y1=L; L=l,R=r;
while (L<R)
{
Mid=(L+R)>>;
if (Check(l,d,Mid,u)==) L=Mid+;
else R=Mid;
}
x2=L; L=l,R=r;
while (L<R)
{
Mid=(L+R+)>>;
if (Check(Mid,d,r,u)==) R=Mid-;
else L=Mid;
}
x1=L; if (!xx1) xx1=x1,yy1=y1,xx2=x2,yy2=y2;
else X1=x1,Y1=y1,X2=x2,Y2=y2;
} int main()
{
//freopen("in.in","r",stdin);
//freopen("out.out","w",stdout); int n,L,R,Mid;
scanf("%d",&n); L=,R=n;
while (L<R)
{
Mid=(L+R)>>;
if (Check(,,Mid,n)>=) R=Mid;
else L=Mid+;
} if (Check(,,L,n)== && Check(L+,,n,n)==)
{
Solve(,,L,n);
Solve(L+,,n,n);
}
else
{
L=,R=n;
while (L<R)
{
Mid=(L+R)>>;
if (Check(,,n,Mid)>=) R=Mid;
else L=Mid+;
}
Solve(,,n,L);
Solve(,L+,n,n);
}
printf("! %d %d %d %d %d %d %d %d\n",xx1,yy1,xx2,yy2,X1,Y1,X2,Y2);
fflush(stdout); return ;
}
C:
题目大意:
给出一个长度为n的数列(n<=3000),每次操作可以把其中一个数增大1或者减小1,求最少操作次数使得数列单调增。
题解:
1.首先一个转化: $a_i<a_{i+1} <--> a_i-i<=a_{i+1}-(i+1)$ 所以把所有$a_i$减去$i$就转化为单调不减,就和POJ3666一样了。
具体可以看 http://www.cnblogs.com/vb4896/p/5877962.html 时间复杂度O(n2)
代码:
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
#include <cstdlib>
#include <set>
using namespace std; #define X first
#define Y second
#define Mod 1000000007
#define N 3010
typedef long long ll;
typedef pair<int,int> pii; inline int Mul(int x,int y){return 1ll*x*y%Mod;}
inline int Add(int x,int y){return ((x+y)%Mod+Mod)%Mod;} int n,m;
int a[N],b[N];
ll dp[N][N],Ans=1ll<<; void Solve()
{
for (int j=;j<=m;j++) dp[][j]=abs(b[j]-a[]);
for (int i=;i<=n;i++)
{
ll tmp=1ll<<;
for (int j=;j<=m;j++)
{
tmp=min(tmp,dp[i-][j]);
dp[i][j]=abs(b[j]-a[i]);
dp[i][j]+=tmp; }
}
for (int j=;j<=m;j++) Ans=min(Ans,dp[n][j]);
} int main()
{
//freopen("in.in","r",stdin);
//freopen("out.out","w",stdout); scanf("%d",&n);
for (int i=;i<=n;i++) scanf("%d",&a[i]),b[i]=a[i]=a[i]-i;
sort(b+,b+n+);
for (int i=;i<=n;i++) if (i== || b[i]!=b[i-]) b[++m]=b[i]; Solve();
printf("%I64d\n",Ans); return ;
}
2.其实这题还有个O(nlogn做法)
以下是在 评论区看到的:为了便于理解 我修改了下,加了点注释。
There is a very short (10 lines!) and fast O(nlgn) solution for Div1C, proposed by woqja125 :
Start by subtracting i from all A[i]. We'll define fi(X) = Minimum operation to make A[1 .. i] monotonically increasing, while keeping A[i] <= X. it's easy to see that the recurrence is —
- Min{|Ai — Y| } (Y <= X) if i == 1
- Min{fi-1(Y) + |Ai — Y|} (Y<=X) otherwise.
Observation : fi is an unimodal function, which have a nonpositive integer slopes. Denote Opt(i) as a point X which makes fi(X) minimum.
We will maintain those functions and calculate the minimum. we examine two cases :
(Fi(X)的图像可以看成是Fi-1(X)和 |Ai — Y|这两个图像的叠加。 Fi(X)的图像总是斜率小于0且斜率绝对值递减的曲线,并且从Opt(i)开始变成水平)
Case 1 : Opt(i-1) <= A[i]. In this case, Opt(i) = A[i], and every point under A[i] have +1 slope.
Case 2 : Opt(i-1) > A[i]. This case is tricky. every point over A[i] have +1 slope, and every point under A[i] have -1 slope. Opt(i) is the point where slope of fi is changed into -1 -> 0. When observing the function, Fi(Opt(i)) = F(i-1)(Opt(i-1)) + Opt(i-1) — A[i].
为了求答案,只要维护好这个Fi(X)曲线就好。具体实现的时候,由于从Fi-1到Fi只和Opt(i-1)有关,我们用个大根堆保存好每一个斜率开始改变的点,然后每次取出最大的点就是
Opt(i-1). 具体实现看代码:
///By woqja125, contest: Codeforces Round #371 (Div. 1), problem: (C) Sonya and Problem Wihtout a Legend, Accepted, # #include<stdio.h>
#include<queue> int main()
{
int n, t;
long long ans = ;
std::priority_queue<int> Q;
scanf("%d%d", &n, &t);
Q.push(t);
for(int i=; i<n; i++)
{
scanf("%d", &t); t-=i;
Q.push(t); //Ai>=Opt(i-1)的时候仅仅增加了Ai这个转折点
if(Q.top() > t) //Ai<Opt(i-1)的时候
{
ans += Q.top() - t;
Q.pop();
Q.push(t); //t push了2次,为的是使得任意相邻的两点之间斜率相差1,即使两点可能重合
}
}
printf("%lld", ans);
return ;
}
D:
题目大意:
给出n*m的01矩阵,最多1e6个询问,每次询问一个子矩形中最大的全1正方形的边长。
题解
1.首先可以dp出以(i,j)为左下角的最大正方形边长,dp[i][j]=max{dp[i+1][j+1],dp[i+1][j],dp[i][j+1]}+1.当然如果(i,j)格是0 那dp[i][j]=0.
2.考虑二分答案。二分时检查矩形(x1,y1)-(x2-Mid+1,y2-Mid+1)里的dp值是否有大于等于Mid的。 区间的最大值可以用二维RMQ解决...第一次写,竟然写对了。然后想起前几天学了个用树状数组求区间最大值的方法,就想把它也扩展到二维,写到一半发现剧难写,再加上这几天心不是很静,弃疗了。
还是贴个RMQ的代码吧。
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
#include <cstdlib>
#include <set>
using namespace std; #define X first
#define Y second
#define Mod 1000000007
#define N 1010
#define Inf 1000000007
typedef long long ll;
typedef pair<int,int> pii; int n,m,Q;
int v[N][N],dp[N][N],sum[N][N];
int f[N][N][][]; void Build()
{
for (int p=,l1=;l1<=n;l1<<=,p++)
{
for (int q=,l2=;l2<=m;l2<<=,q++)
{
if (p+q==) continue;
for (int i=;i+l1-<=n;i++)
{
for (int j=;j+l2-<=m;j++)
{
if (p!=) f[i][j][p][q]=max(f[i][j][p-][q],f[i+(l1>>)][j][p-][q]);
else f[i][j][p][q]=max(f[i][j][p][q-],f[i][j+(l2>>)][p][q-]);
}
}
}
}
} int Getmax(int x1,int y1,int x2,int y2)
{
int p=,l1=,q=,l2=,tmp1,tmp2;
while (l1<=x2-x1+) l1<<=,p++; p--,l1>>=;
while (l2<=y2-y1+) l2<<=,q++; q--,l2>>=; tmp1=max(f[x1][y1][p][q],f[x1][y2-l2+][p][q]);
tmp2=max(f[x2-l1+][y1][p][q],f[x2-l1+][y2-l2+][p][q]);
return max(tmp1,tmp2);
} int main()
{
//freopen("in.in","r",stdin);
//freopen("out.out","w",stdout); scanf("%d%d",&n,&m);
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
{
scanf("%d",&v[i][j]);
sum[i][j]=sum[i][j-]+sum[i-][j]-sum[i-][j-]+v[i][j];
}
for (int i=n;i>=;i--)
{
for (int j=m;j>=;j--)
{
if (v[i][j]==) dp[i][j]=;
else dp[i][j]=min(min(dp[i+][j+],dp[i][j+]),dp[i+][j])+;
f[i][j][][]=dp[i][j];
}
} Build();
int L,R,Mid,x1,y1,x2,y2;
scanf("%d",&Q);
while (Q--)
{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
if (sum[x2][y2]-sum[x1-][y2]-sum[x2][y1-]+sum[x1-][y1-]==) printf("0\n");
else
{
L=,R=min(x2-x1+,y2-y1+);
while (L<R)
{
Mid=(L+R+)>>;
if (Getmax(x1,y1,x2-Mid+,y2-Mid+)>=Mid) L=Mid;
else R=Mid-;
}
printf("%d\n",L);
}
} return ;
}
Codeforces Round #371 (Div. 1)的更多相关文章
- Codeforces Round #371 (Div. 2)B. Filya and Homework
题目链接:http://codeforces.com/problemset/problem/714/B 题目大意: 第一行输入一个n,第二行输入n个数,求是否能找出一个数x,使得n个数中的部分数加上x ...
- Codeforces Round #371 (Div. 2) - B
题目链接:http://codeforces.com/contest/714/problem/B 题意:给定一个长度为N的初始序列,然后问是否能找到一个值x,然后使得序列的每个元素+x/-x/不变,最 ...
- Codeforces Round #371 (Div. 2) - A
题目链接:http://codeforces.com/contest/714/problem/A 题意:有两个人A,B 给定A的时间区间[L1,R1], B的时间区间[L2,R2],然后在正好K分钟的 ...
- 严格递增类的dp Codeforces Round #371 (Div. 1) C dp
http://codeforces.com/contest/713 题目大意:给你一个长度为n的数组,每次有+1和-1操作,在该操作下把该数组变成严格递增所需要的最小修改值是多少 思路:遇到这类题型, ...
- Codeforces Round #371 (Div. 2) C 大模拟
http://codeforces.com/contest/714/problem/C 题目大意:有t个询问,每个询问有三种操作 ①加入一个数值为a[i]的数字 ②消除一个数值为a[i]的数字 ③给一 ...
- Codeforces Round #371 (Div. 1) C. Sonya and Problem Wihtout a Legend 贪心
C. Sonya and Problem Wihtout a Legend 题目连接: http://codeforces.com/contest/713/problem/C Description ...
- Codeforces Round #371 (Div. 1) D. Animals and Puzzle 二维倍增
D. Animals and Puzzle 题目连接: http://codeforces.com/contest/713/problem/D Description Owl Sonya gave a ...
- Codeforces Round #371 (Div. 2) C. Sonya and Queries 水题
C. Sonya and Queries 题目连接: http://codeforces.com/contest/714/problem/C Description Today Sonya learn ...
- Codeforces Round #371 (Div. 2) D. Searching Rectangles 交互题 二分
D. Searching Rectangles 题目连接: http://codeforces.com/contest/714/problem/D Description Filya just lea ...
随机推荐
- C#知识点总结系列:5、CLR的组成和运转
clr基本 CLR(Common Language Runtime)是一个可由多种编程语言使用的“运行时”.(例如:c#,c++/cli,vb,f#,ironpython,ironruby,il... ...
- 使用 CSS3 绘制 Hello Kitty
偶然间看到了 SegmentFault 上的一篇文章,感觉这个 Hello Kitty 画的还不错,心血来潮也用 CSS3 画了个 Hello Kitty,现在在这里记录一下详细的绘制过程.想要源码. ...
- linux下添加环境变量
我安装完 RedHat Linux 5 之后,在终端使用一些命令,如: ifcinfig 查看本机的IP,发现不能使用此命令,提示说“command not found”,这该怎么办呢 想想肯定是环境 ...
- C++ int与string的转化
int本身也要用一串字符表示,前后没有双引号,告诉编译器把它当作一个数解释.缺省情况下,是当成10进制(dec)来解释,如果想用8进制,16进制,怎么办?加上前缀,告诉编译器按照不同进制去解释.8进制 ...
- WeX5的简单介绍及UI的简单讲解
WeX5的简单介绍及UI的简单讲解 (2016-01-13 14:49:05) 标签: it 分类: WeX5的初步自学 一.WeX5的简单讲解 1.WeX5是前端快速开发框架,可开发跨端运行应用.是 ...
- Spring的注解方式实现AOP
Spring对AOP的实现提供了很好的支持.下面我们就使用Spring的注解来完成AOP做一个例子. 首先,为了使用Spring的AOP注解功能,必须导入如下几个包.aspectjrt.jar,asp ...
- 关于前端CSS预处理器Sass的小知识!
前面的话 "CSS预处理器"(css preprocessor)的基本思想是,用一种专门的编程语言,进行网页样式设计,然后再编译成正常的CSS文件.SASS是一种CSS的开发工 ...
- 常见的js 里对数字进行处理的函数方法集合
常见的对小数值舍入为整数的几个方法:Math.ceil().Math.floor()和Math.round(). 这三个方法分别遵循下列舍入规则: Math.ceil()执行向上舍入,即它总是将数值向 ...
- np2016课程总结
林牧 SA16222166 课程目标 课程安排 A1a A2 A3 项目集成 环境搭建 其他方面的收获 本课心得 课程目标 通过实现一个医学辅助诊断的专家系统原型,具体为实现对血常规检测报告OCR识别 ...
- MySQL DML 整理
DML(Data Manipulation Language)数据操纵语言statements are used for managing data within schema objects. 由D ...