20260422紫题训练

A - 订货

简单题,直接简图跑费用流。

/*
Luogu P2517 [HAOI2010] 订货
2026-04-22
*/
#include<bits/stdc++.h>
using namespace std;
namespace IO{
    template<typename T>
    inline void read(T&x){
        x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=x*10+c-'0',c=getchar();
        f?x=-x:0;
    }
    template<typename T>
    inline void write(T x){
        if(x==0){putchar('0');return ;}
        x<0?x=-x,putchar('-'):0;short st[50],top=0;
        while(x) st[++top]=x%10,x/=10;
        while(top) putchar(st[top--]+'0');
    }
    inline void read(char&c){c=getchar();while(isspace(c)) c=getchar();}
    inline void write(char c){putchar(c);}
    inline void read(string&s){s.clear();char c;read(c);while(!isspace(c)&&~c) s+=c,c=getchar();}
    inline void write(string s){for(int i=0,len=s.size();i<len;i++) putchar(s[i]);}
    template<typename T>inline void write(T*x){while(*x) putchar(*(x++));}
    template<typename T,typename...T2> inline void read(T&x,T2&...y){read(x),read(y...);}
    template<typename T,typename...T2> inline void write(const T x,const T2...y){write(x),putchar(' '),write(y...),sizeof...(y)==1?putchar('\n'):0;}
}using namespace IO;
const int maxn=100,inf=1000000000;
int n,m,S;
template<int maxn,int maxm>struct LSQXX{
    int head[maxn],nxt[maxm*2],to[maxm*2],val[maxm*2],cost[maxm*2],cnt=1;
    void add(int u,int v,int z,int c){nxt[++cnt]=head[u],to[cnt]=v,val[cnt]=z,cost[cnt]=c,head[u]=cnt;}
    void clear(){memset(head,0,sizeof(head)),cnt=1;}
};
class Network_Flow{
private:
    LSQXX<maxn,maxn*3>e;
    int cur[maxn],s,t,ans2,dis[maxn];
    bool inque[maxn],vis[maxn];
    void edge_add(int u,int v,int w,int z){e.add(u,v,w,z),e.add(v,u,0,-z);}
    bool spfa(){
        for(int i=s;i<=t;i++) inque[i]=0,dis[i]=inf;
        queue<int>q;q.push(s);dis[s]=0;
        while(!q.empty()){
            int u=q.front();q.pop();inque[u]=0;
            for(int i=e.head[u];i;i=e.nxt[i]){
                int v=e.to[i];
                if(e.val[i]==0) continue;
                if(dis[v]>dis[u]+e.cost[i]){
                    dis[v]=dis[u]+e.cost[i];
                    if(!inque[v]) q.push(v),inque[v]=1;
                }
            }
        }
        return dis[t]!=inf;
    }
    int dfs(int u,int flow=inf){
        if(flow==0||u==t) return flow;
        int ans=0;
        if(vis[u]) return 0;vis[u]=1;
        for(int&i=cur[u];i;i=e.nxt[i]){
            int v=e.to[i];
            if(dis[v]!=dis[u]+e.cost[i]) continue;
            int new_flow=dfs(v,min(flow,e.val[i]));
            flow-=new_flow,ans+=new_flow,ans2+=new_flow*e.cost[i];
            e.val[i]-=new_flow,e.val[i^1]+=new_flow;
            if(flow==0) return vis[u]=0,ans;
        }
        vis[u]=0;
        return ans;
    }
public:
    void build(){
        read(n,m,S);
        s=0,t=n+1;
        for(int i=1;i<n;i++) edge_add(i,i+1,S,m);
        for(int i=1;i<=n;i++){
            int U;read(U);
            edge_add(i,t,U,0);
        }
        for(int i=1;i<=n;i++){
            int d;read(d);
            edge_add(s,i,inf,d);
        }
    }
    int work(){
        ans2=0;
        while(spfa()){
            for(int i=s;i<=t;i++) cur[i]=e.head[i];
            dfs(s);
        }
        return ans2;
    }
}wll;
signed main(){
    wll.build();
    write(wll.work());
    return 0;
}

B - 礼物

简单题,做过。把式子拆开,其中一项用 \(NTT\) 算贡献,其它直接计算。

/*
Luogu P3723 [AHOI2017/HNOI2017] 礼物
2026-04-22
*/
#include<bits/stdc++.h>
using namespace std;
namespace IO{
    template<typename T>
    inline void read(T&x){
        x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=x*10+c-'0',c=getchar();
        f?x=-x:0;
    }
    template<typename T>
    inline void write(T x){
        if(x==0){putchar('0');return ;}
        x<0?x=-x,putchar('-'):0;short st[50],top=0;
        while(x) st[++top]=x%10,x/=10;
        while(top) putchar(st[top--]+'0');
    }
    inline void read(char&c){c=getchar();while(isspace(c)) c=getchar();}
    inline void write(char c){putchar(c);}
    inline void read(string&s){s.clear();char c;read(c);while(!isspace(c)&&~c) s+=c,c=getchar();}
    inline void write(string s){for(int i=0,len=s.size();i<len;i++) putchar(s[i]);}
    template<typename T>inline void write(T*x){while(*x) putchar(*(x++));}
    template<typename T,typename...T2> inline void read(T&x,T2&...y){read(x),read(y...);}
    template<typename T,typename...T2> inline void write(const T x,const T2...y){write(x),putchar(' '),write(y...),sizeof...(y)==1?putchar('\n'):0;}
}using namespace IO;
#define int long long
const int maxn=50010;
const double PI=3.14159265358979323846;
int x[maxn],y[maxn];
complex<double>a[maxn*10],b[maxn*10];
void fft(complex<double>a[],int n,bool flag){
    if(n==1) return ;
    complex<double>a0[n/2],a1[n/2];
    for(int i=0;i<n/2;i++) a0[i]=a[i*2],a1[i]=a[i*2+1];
    fft(a0,n/2,flag),fft(a1,n/2,flag);
    complex<double>w=1,dwg;
    if(flag) dwg.real(cos(-2*PI/n)),dwg.imag(sin(-2*PI/n));
    else dwg.real(cos(2*PI/n)),dwg.imag(sin(2*PI/n));
    for(int i=0;i<n/2;i++){
        a[i]=a0[i]+w*a1[i];
        a[i+n/2]=a0[i]-w*a1[i];
        w*=dwg;
    }
}
void mul(complex<double>a[],int n,complex<double>b[],int m){
    int len=1;
    while(len<=n+m) len*=2;
    fft(a,len,0),fft(b,len,0);
    for(int i=0;i<len;i++) a[i]*=b[i];
    fft(a,len,1);
    for(int i=0;i<len;i++) a[i]/=len;
}
signed main(){
    int n,m,ans=0,h=0;read(n,m);
    for(int i=1;i<=n;i++) read(x[i]),ans+=x[i]*x[i],h+=x[i];
    for(int i=1;i<=n;i++) a[n-i]=x[i];
    for(int i=1;i<=n;i++) read(y[i]),ans+=y[i]*y[i],h-=y[i],b[i]=b[n+i]=y[i];
    mul(a,n+n+1,b,n+n+1);
    int maxx=0,minn=1000000000000000000;
    for(int i=-m;i<=m;i++) minn=min(minn,n*i*i+2*h*i);
    ans+=minn;
    for(int i=n;i<=n*2;i++) maxx=max(maxx,(int)round(a[i].real()));
    ans-=2*maxx;
    write(ans);
    return 0;
}

C - 火星商店问题

赛时思路

建可持久化 \(Trie\) 树,版本表示天,每个点维护一个线段树,线段树的下标是位置,值存 \(cnt\),及被遍历到的次数,表示祂被经过的次数。查询在 \(now\)\(now-d+1\) 两个版本上走,贪心的选,尽量选大的,如果大的那边 \(cnt\) 差值为 \(0\) 就走小的那边。
但是这样是错的,因为新版本的线段树需要和旧版本的合并,但这是困难的。
题解

总结

思考问题要仔细,把字节想清楚,不要再实现的时候才发现思路错了。

posted @ 2026-04-22 21:12  Link-Cut_Trees  阅读(6)  评论(0)    收藏  举报