首页 > 其他分享 >The 2021 ICPC Asia Macau Regional Contest

The 2021 ICPC Asia Macau Regional Contest

时间:2023-10-26 21:00:25浏览次数:33  
标签:Macau min int sum Regional Contest pos ans 区间

\(C. Laser Trap\)

根据题意不难判断出需要极角排序,然后对于每个点寻找更小的一个 \(180\) 度的点数。即使听说是用双指针实现查找依旧没什么思路。后来看了别人的实现方法发现确实比较简单,甚至只需要维护极角就可以了。

const long double pi=acosl(-1);
void solve(){
    int n=read(),ans=n;
    vector<long double> p(n*2+2);
    for(int i=1;i<=n;i++){
        int x=read(),y=read();
        p[i]=atan2l(1.0*y,1.0*x);   //long double 的 atan2
    }
    sort(p.begin()+1,p.begin()+n+1);
    for(int i=1;i<=n;i++){
        p[i+n]=p[i]+2*pi;
    }
    for(int i=1,j=1;i<=n;i++){
        while(j<=2*n && p[j]-p[i]<pi){
            j++;
        }       
        ans=min(ans,j-i-1); 
    }
    cout<<ans<<'\n';
    //puts(ans>0?"YES":"NO");
    //puts(ans>0?"Yes":"No");
}

\(G. Cyclic Buffer\)

首先我们只需要把不在区间内的下一个查询点移到区间的端点上即可。这种做法必定可以转移到所谓的更优解上。那么我们只需要模拟这个过程, 对于不在区间上的点 \(dp\) 判断是把它移到 \(1/k\) 位置上。这个比较麻烦,因为我们需要知道这个区间上已经存在的点是哪个,下一个需要查询的不在目前区间上的点是哪个。这个用线段树去维护区间上第一个不存在于区间的点,然后记录下一个需要的不在区间内的点。

int a[N], poi[N]/*i元素位置*/, f[N][2]/*i在1/k位置的移动花费*/, p[N][2]/*i在1/k位置的下一个不在可取区间的数字*/;
int n, k ;
struct Node{
    int l, r, sum;
}tr[N * 4];
void pushup(Node &u, Node &l, Node &r){
    u.sum = l.sum + r.sum;
}                     
void pushup(int u){
    pushup(tr[u], tr[u << 1], tr[u << 1 | 1]);
}
void build(int u, int l, int r){                          
    if (l == r){
        tr[u] = (Node){l, r, 0};
        return ;
    }else{
        tr[u] = (Node){l, r, 0};
        int mid = (l + r) >> 1;
        build(u << 1, l, mid), build(u << 1 | 1, mid + 1, r);
    }
}
void modify(int u, int x, int v){   
    if (tr[u].l == x && tr[u].r == x){
        tr[u].sum+=v;
        return ;
    }else{
        int mid = tr[u<<1].r;
        if (x <= mid) modify(u << 1, x, v);
        else modify(u << 1 | 1, x, v);
        pushup(u);
    }
}
int query(int u, int l, int r){
    if(tr[u].sum==tr[u].r-tr[u].l+1)return INF; //如果区间满了 返回INF
    if (tr[u].l == tr[u].r) return tr[u].l;   //如果找到为0的单点 返回点的位置
    else{
        int mid=tr[u<<1].r,res=INF;     //mid 为左儿子的右界 
        if (l <= mid)res=query(u << 1, l, r);   //如果当前的查询左界小于mid 就去查询查询
        if (r > mid && res==INF)res=query(u<<1|1,l,r);  //如果上面没找到且 需要在右界查询
        return res; 
    }
}
void solve(){
    n=read(),k=read();
    for(int i=1;i<=n;i++){
        a[i]=read();
        poi[a[i]]=i;
        f[i][0]=f[i][1]=INF;        
    }
    if(k==n){
        cout<<0<<'\n';
        return ;
    }
    build(1,1,n);
    int s[2]={1,k},l=1,r=k;
    for(int i=1;i<=k;i++){
        modify(1,a[i],1);
    }
    for(int i=1;i<=n;i++){  //枚举n次 一个个判断
        for(int j=0;j<=1;j++){  
            if(a[s[j]]==n)continue; //如果在1/k的位置上的是n 说明结束了 continue;
            p[a[s[j]]][j]=query(1,a[s[j]]+1,n); //查询1/k下一个不在区间内的数
        }
        modify(1,a[s[1]],-1);   //删除k位置的数
        for(int j=0;j<2;j++){   
            s[j]=(s[j]-1+n)%n;  //修改1/k位置上的数字
            if(!s[j])s[j]=n;    
        }
        modify(1,a[s[0]],1);    //增加1位置的数字
    }
    int ans=INF,sta=1;
    while(poi[sta]<=k){
        sta++;  //先过掉所有已经可查询的数字
    }
    auto get_min=[](int x,int y){   //判断x到达y的最小移动距离
        if(y<x)swap(x,y);
        return min(y-x,x+n-y);
    };
    auto get_dis=[](int x,int y){   //判段x到达y的右移的距离
        return (y-x+n)%n;
    };
    f[sta][0]=get_min(poi[sta],1);
    f[sta][1]=get_min(poi[sta],k);
    for(int i=sta;i<n;i++){
        for(int j=0;j<=1;j++){
            if(f[i][j]==INF)continue;
            if(p[i][j]>n)ans=min(ans,f[i][j]);
            else {
                int pos=poi[p[i][j]]+get_dis(poi[i],j==0?1:k);
                if(pos>n)pos-=n;
                int d0=get_min(pos,1),d1=get_min(pos,k);
                f[p[i][j]][0]=min(f[p[i][j]][0],f[i][j]+d0);
                f[p[i][j]][1]=min(f[p[i][j]][1],f[i][j]+d1);
            }
        }
    }
    ans=min({ans,f[n][0],f[n][1]});
    cout<<ans<<'\n';
    //puts(ans>0?"YES":"NO");
    //puts(ans>0?"Yes":"No");
}

标签:Macau,min,int,sum,Regional,Contest,pos,ans,区间
From: https://www.cnblogs.com/edgrass/p/17790383.html

相关文章

  • AtCoder Beginner Contest(abc) 309
    B-Rotate题目大意给定一个n*n的矩阵,要求把矩阵的最外围按照顺时针转动一个数据,输出转动后的矩阵;解题思路数据不大,暴力即可;神秘代码#include<bits/stdc++.h>#defineintlonglong#defineIOSios::sync_with_stdio(false);cin.tie(0);cout.tie(0);#def......
  • [Leetcode Weekly Contest]368
    链接:LeetCode[Leetcode]2908.元素和最小的山形三元组I给你一个下标从0开始的整数数组nums。如果下标三元组(i,j,k)满足下述全部条件,则认为它是一个山形三元组:i<j<knums[i]<nums[j]且nums[k]<nums[j]请你找出nums中元素和最小的山形三元组,并返回......
  • AtCoder Regular Contest 167——B - Product of Divisors
    题目很明显,给定 所有因数的积不断除以最多能除几次。首先,很容易发现,对于每一对因子,都可以对答案得出B的贡献,设A的因子数目为n。将A进行质因数分解,PBa1,PBa2,PBa3……PBam,那么因数个数就是质因子加一的乘积。那么因子对数也就是前者一半。答案就是B乘因子对数除以二注意此处......
  • AtCoder Beginner Contest(abc) 308
    B-DefaultPrice题目大意小莫买了n个寿司,现在给出m个寿司的名称和m+1个价格,如果小莫买的其中一个寿司不在这m个寿司之中就用价格m0;请问小莫买的寿司花了多少钱解题思路数据不大,暴力哈希即可;神秘代码#include<bits/stdc++.h>#defineintlonglong#define......
  • Atcoder Beginner Contest 324 G Generate Arrays 题解-Treap
    为了更好的阅读体验,请点击这里题目链接套上平衡树板子就能做的很快的题,然后因为是指针存树,因此交换只需要把序列大小较小的挨个拿出来插到相应的地方即可。复杂度\(O(N\log^2N)\)。但是一定要记住不可以直接使用std::swap交换包含带有指针的类的实例(如代码中的Treap类)!......
  • AtCoder Regular Contest 167
    Preface补一下上周日的ARC,因为当天白天和队友一起VP了一场所以就没有精力再打一场了这场经典C计数不会D这种贪心乱搞反而是一眼秒了,后面的EF过的太少就没看A-ToastsforBreakfastParty用一个类似于蛇形的放法就好了,比如对于\(n=9,m=5\),放法为:567894321#includ......
  • Japan Registry Services (JPRS) Programming Contest 2023 (AtCoder Beginner Contes
    JapanRegistryServices(JPRS)ProgrammingContest2023(AtCoderBeginnerContest324)赛后总结可悲的是:我没来得及写题解。T1Same秒切。直接输入排一遍序再遍历即可。#include<bits/stdc++.h>usingnamespacestd;intn,a[101];intmain(){cin>>n;f......
  • 比赛总结:Japan Registry Services (JPRS) Programming Contest 2023 (AtCoder Beginn
    比赛:JapanRegistryServices(JPRS)ProgrammingContest2023(AtCoderBeginnerContest324)A-same1.常规方法intmain(){ intn; cin>>n; vector<int>s(n);//利用vector容器可以不需要确定内存大小 for(auto&n:s) { cin>>n; } for(inti=0;i......
  • AtCoder Regular Contest 066 F Contest with Drinks Hard
    洛谷传送门AtCoder传送门下文令\(a\)为原题中的\(T\)。考虑若没有饮料,可以设\(f_i\)表示,考虑了前\(i\)道题,第\(i\)道题没做的最大得分。转移就枚举上一道没做的题\(j\),那么\([j+1,i-1]\)形成一个连续段。设\(b_i\)为\(a_i\)的前缀和,可得:\[f_i=\max\li......
  • [Leetcode Weekly Contest]367
    链接:LeetCode[Leetcode]2903.找出满足差值条件的下标I给你一个下标从0开始、长度为n的整数数组nums,以及整数indexDifference和整数valueDifference。你的任务是从范围[0,n-1]内找出2个满足下述所有条件的下标i和j:abs(i-j)>=indexDifference且a......