比赛链接:http://codeforces.com/contest/879
A. Borya's Diagnosis
题意:一个人去看医生,这n个医生分别在s[i],s[i]+2*d[i]...这些天上班,这个人必须按照顺序拜访医生,问这个人拜访最后一个医生的最早时间。
解法:按顺序模拟。
#include <bits/stdc++.h>
using namespace std;
const int maxn = 1010;
struct node{
int s,d;
}a[maxn];
bool vis[2000010];
int n;
int main(){
scanf("%d",&n);
for(int i=1; i<=n; i++){
scanf("%d %d", &a[i].s, &a[i].d);
}
int curpos = 0;
for(int i=1; i<n; i++){
int t = a[i].s;
while(1){
if(!vis[t]&&t>curpos){
curpos = t;
vis[t] = 1;
break;
}
t = t + a[i].d;
}
}
int t = a[n].s;
while(1){
if(!vis[t]&&t>curpos){
vis[t] = 1;
break;
}
t = t + a[n].d;
}
printf("%d\n", t);
return 0;
}
B. Table Tennis
题意:有1-n组成的一个排列,每个数代表一个人的power,两个人比赛时power大的人会获胜比赛从左到右比,输了的人移到末尾,赢的人和下一个人比,当有一个人赢了k盘时输出该人的power
解法:只有power为n的人才能赢>=n盘 ,所以k>=n时输出n。k<n时从左到右进行遍历,遇到可以赢k盘的人就结束。我用vector直接模拟这个过程。
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int maxn = 550;
int a[maxn];
vector <int> b;
int main(){
int n;
LL k;
scanf("%d %lld", &n,&k);
int mx = 0;
for(int i=1; i<=n; i++) scanf("%d", &a[i]), mx = max(mx, a[i]);
if(k >= n){
return 0*printf("%d\n", mx);
}
for(int i=1; i<=n; i++) b.push_back(a[i]);
for(int i=0; i<b.size(); i++){
int now = 0;
int mx = 0;
for(int j=0; j<i; j++) mx = max(mx, b[j]);
if(mx<b[i]&&i>0) now=1;
for(int j=i+1; j<b.size(); j++){
if(b[j]>b[i]) break;
else now++;
}
if(now==0) b.push_back(b[i]);
else if(now>=k){
return 0*printf("%d\n", b[i]);
}else b.push_back(b[i]);
}
return 0;
}
C. Short Program
题意:就是给你n个操作,然后化简,在5步之内输出跟n步操作同样效果的操作。
解法:很容易想到,最后一定用三种操作 &|^能表示完所有的操作。然而x不知道是0还是1,那么最后就看假如x是0,x变成什么值,y是1,y变成什么值,由什么操作可以得来就好了。
#include <bits/stdc++.h>
using namespace std;
const int maxn = 5e5+10;
int main(){
int n;
scanf("%d", &n);
int x=1023, y=0;
for(int i=0; i<n; i++){
char s[2];
int d;
scanf("%s%d",s,&d);
if(s[0]=='|') x|=d,y|=d;
else if(s[0]=='&') x&=d,y&=d;
else if(s[0]=='^') x^=d,y^=d;
}
int a=-1,b=-1;
for(int i=0; i<10; i++){
int xx = (x&(1<<i))>0?1:0;
int yy = (y&(1<<i))>0?1:0;
if(xx==1&&yy==1){
if(a==-1) a=1<<i;
else a+=(1<<i);
}
if(xx==0&&yy==0){
if(a==-1) a=1<<i;
else a=a|(1<<i);
if(b==-1) b=1<<i;
else b=b|(1<<i);
}
if(xx==0&&yy==1){
if(b==-1) b=1<<i;
else b=b|(1<<i);
}
}
int ans=0;
if(a!=-1) ans++;
if(b!=-1) ans++;
printf("%d\n", ans);
if(a!=-1) printf("| %d\n", a);
if(b!=-1) printf("^ %d\n", b);
return 0;
}
D. Teams Formation
题意:有一辆车每次运送n个人,这n个人分别来自于不同的城市。一共运送m趟,现在题目要求你将每次排队的n个人进行分队,每一队的人数要求是k个,也就是说如果相邻的人来自于同一个城市,同时满足加在一起的人数是大于或等于K的,那么就可以减少k个人,问你最后能能够剩下多少人?类似于祖玛。。
解法:对于一个长度为n的序列,先将这个序列压缩一下,就是如果相同的数连在一起那我们就记一个并记下他连续的个数,之后我们先考虑他的内部是否已经存在连续k个一样的数字,如果已经存在,那么先将这些消去,否则我们用两个指针,一个指向序列开头,另一个指向尾部,表示当相邻两个串合并起来的时候是否有连续k个相同的数字,如果有那么删掉,最后就直接暴力做下去,直到不存在满足要求的数。
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int maxn = 100010;
LL n, m, k, x, s[maxn][2], cnt;
int main(){
scanf("%d %d %d", &n,&k,&m);
for(int i=1; i<=n; i++){
scanf("%d", &x);
if(!cnt||s[cnt][0]!=x){
s[++cnt][0]=x;
s[cnt][1]=1;
}else s[cnt][1]++;
if(s[cnt][1]==k){
s[cnt][1]=0;
cnt--;
}
}
LL tot=0;
for(int i=1; i<=cnt; i++) tot+=s[i][1];
int head=1,tail=cnt;
while(head<tail&&s[head][0]==s[tail][0]){
if((s[head][1]+s[tail][1])%k==0){
head++;
tail--;
}
else{
s[head][1]=(s[head][1]+s[tail][1])%k;
s[tail][1]=0;
break;
}
}
LL ans = 0;
if(head<tail){
for(int i=head;i<=tail;i++) ans+=s[i][1];
ans = ans*(m-1);
ans+=tot;
}else if(head==tail){
if((s[head][1]*m)%k==0) ans=0;
else{
ans = tot+s[head][1]*(m-1);
ans -= s[head][1]*m-s[head][1]*m%k;
}
}
printf("%lld\n", ans);
return 0;
}
E. Tournament
题意:现在有n个人参加比赛,每个人都有k项能力值,每个能力值表示这个人在这个技能上面的能力。现在这些人会两两进行比赛,他们会挑选一个技能进行比较,分值高的获胜,胜利者留下来,失败者离开。你需要回答,最后究竟会有多少个人可能成为冠军。
解法:http://www.cnblogs.com/qscqesze/p/7762484.html
假设现在有一群人都可能获得冠军,那么必须满足任何一个人,都能找到一个人的某项技能小于等于他,也能找到一个人的某项技能强于他。那么对于这个集合,对于每个人都能存在一个比赛方案,使得他成为冠军。然后我们动态维护这个集群就好了,用set去维护。你可以用图论的方式去理解。如果A能够打败B,那么连一条A->B的边,显然胜利者的含义就是如果A能够达到其他的所有点,那么A就是胜利者。最后的胜利者集合,里面的任何两个点都能互相到达,如果已经成为了团,我们就进行缩点就好了,我们用set去缩点。然后维护每一个团就行。
#include <bits/stdc++.h>
using namespace std;
int n, x, k;
struct people{
int mx[15], mi[15], sz;
bool operator<(const people &rhs) const{
for(int j=1; j<=k; j++) if(mx[j]>rhs.mi[j]) return 0; return 1;
}
}tmp;
set <people> s;
int main(){
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++){
tmp.sz=1;
for(int j=1;j<=k;j++){
scanf("%d", &x);
tmp.mi[j]=tmp.mx[j]=x;
}
auto x=s.lower_bound(tmp), y=s.upper_bound(tmp);
while(x!=y){
tmp.sz+=x->sz;
for(int j=1;j<=k;j++) tmp.mi[j]=min(tmp.mi[j],x->mi[j]), tmp.mx[j]=max(tmp.mx[j],x->mx[j]);
s.erase(x++);
}
s.insert(tmp);
printf("%d\n", s.rbegin()->sz);
}
return 0;
}