POJ1737 Connected Graph 高精度+递推

题意:求出\(n\)个点的无向完全图的子图中有多少个是连通图.

思路:

令\(f_i\)表示\(i\)个点时的连通图个数.

考虑如果图不连通,那么必定存在某些点与\(1\)号点不在一个连通分量中.

如果我们要计算\(f_n\),那就枚举和\(1\)号点在一个连通分量中中的有几个点,那么与这个连通分量无关的点的集合内部就可以随便连边了.这种情况下我们要考虑是哪些点和\(1\)号点在一个连通分量中,另外还要乘上这个连通分量的内部连边数.这个是之前已经搞出来的.

随后利用高精度预处理一下直接输出.

#include<cstdio>
#include<cstring>
#include<cctype>
#include<iostream>
#include<algorithm>
using namespace std;

static const int mod=1e4;
struct Int{
	int d[500],l;
	Int():l(1){memset(d,0,sizeof d);}
	inline void operator=(const int&x){memset(d,0,sizeof d),l=1,d[0]=x;}
	inline void operator+=(const Int&B){
		l=l>B.l?l:B.l;for(int i=0;i<l;++i)if((d[i]+=B.d[i])>=mod)d[i]-=mod,d[i+1]++;
		if(d[l])l++;
	}
	inline void operator*=(const Int&B){
		Int C;
		for(int i=0;i<l;++i)for(int j=0;j<B.l;++j){
			C.d[i+j]+=d[i]*B.d[j];if(C.d[i+j]>=mod)C.d[i+j+1]+=C.d[i+j]/mod,C.d[i+j]%=mod;
		}
		for(C.l=l+B.l+1;!C.d[C.l-1];--C.l);
		*this=C;
	}
	inline void operator-=(const Int&B){
		for(int i=0;i<B.l;++i){if((d[i]-=B.d[i])<0){d[i]+=mod,d[i+1]--;}}
		while(l>1&&d[l-1]==0)--l;
	}
	Int operator*(const Int&B)const{Int r=*this;r*=B;return r;}
	inline void print(){
		printf("%d",d[l-1]);for(int i=l-2;i>=0;--i)printf("%04d",d[i]);
	}
};
Int pow(int y){
	Int t,r;for(t=2,r=1;y;y>>=1,t*=t)if(y&1)r*=t;return r;
}
Int C[51][51];
inline void Init(){
	C[0][0]=1,C[1][0]=1,C[1][1]=1;
	for(int i=2;i<=50;++i){
		C[i][0]=1,C[i][i]=1;
		for(int j=1;j<i;++j)C[i][j]=C[i-1][j-1],C[i][j]+=C[i-1][j];
	}
}
Int f[51];

int main(){
	f[1]=1;Init();
	for(int i=2;i<=50;++i){
		f[i]=pow(i*(i-1)>>1);
		for(int j=0;j<i-1;++j)f[i]-=C[i-1][j]*f[j+1]*pow((i-j-1)*(i-j-2)>>1);
	}
	int x;while(scanf("%d",&x)&&x)f[x].print(),puts("");
	return 0;
}

hdu3932 Groundhog Build Home 计算几何+最小圆覆盖

思路:

本来想写的是求出凸包然后在凸包上\(O(n^3)\)进行暴力的算法.虽然平面上的随机点形成的凸包点数很少,不过显然如果直接给出一个凸包就直接卡掉了.

于是我学习的是随机增量法求最小圆覆盖.

网上题解很多,就不解释了(还不太懂),如果随机打乱一下的话,好像是能够做到期望\(O(n)\),相当厉害的算法.

一个麻烦就是求出三角形的外心,用了一大堆叉积来算好像很傻逼...

#include<cstdio>
#include<cstring>
#include<cctype>
#include<cmath>
#include<iostream>
#include<algorithm>
using namespace std;

#define N 1010

#define eps 1e-8
typedef double f2;
struct Point{
    f2 x,y;
    Point(){}
    Point(f2 _x,f2 _y):x(_x),y(_y){}
    bool operator<(const Point&B)const{return x<B.x;}
    Point operator+(const Point&B)const{return Point(x+B.x,y+B.y);}
    Point operator-(const Point&B)const{return Point(B.x-x,B.y-y);}
    Point operator*(const f2&p){return Point(p*x,p*y);}
    Point operator/(const f2&p){return Point(x/p,y/p);}
}P[N];
inline f2 sqr(const f2&x){return x*x;}
f2 Cross(const Point&a,const Point&b){return a.x*b.y-a.y*b.x;} 
f2 getdist(const Point&a,const Point&b){return sqrt(sqr(a.x-b.x)+sqr(a.y-b.y));}

struct Line{
    Point p,v;
    Line(){}
    Line(Point _p,Point _v):p(_p),v(_v){}
};
Point GetLineIntersection(Line L1,Line L2){
    return L1.p+L1.v*(Cross(L1.p-L2.p,L2.v)/Cross(L1.v,L2.v));
}
Point Getvert(const Point&a){return Point(-a.y,a.x);}
Point Getmid(const Point&a,const Point&b){return Point((a.x+b.x)/2,(a.y+b.y)/2);}

Line GetLine(const Point&a,const Point&b){
    return Line(Getmid(a,b),Getvert(a-b));
}
Point Getcenter(Point a,Point b,Point c){
    Point P[3]={a,b,c};sort(P,P+3);a=P[0],b=P[1],c=P[2];
    if(fabs(Cross(a-b,b-c))<eps)return Getmid(a,c);
    else return GetLineIntersection(GetLine(a,b),GetLine(b,c));
}
    
int main(){
    int X,Y,n;register int i,j,k;
    while(scanf("%d%d%d",&X,&Y,&n)!=EOF){
        for(i=1;i<=n;++i)scanf("%lf%lf",&P[i].x,&P[i].y);
        random_shuffle(P+1,P+n+1);
        Point center=P[1];f2 r=0;
        for(i=2;i<=n;++i){
            if(getdist(center,P[i])>r){
                center=P[i],r=0;
                for(j=1;j<i;++j){
                    if(getdist(center,P[j])>r){
                        center=Getmid(P[i],P[j]),r=getdist(P[i],P[j])*.5;
                        for(k=1;k<j;++k)
                            if(getdist(center,P[k])>r)center=Getcenter(P[i],P[j],P[k]),r=getdist(center,P[i]);
                    }
                }
            }
        }
        printf("(%.1lf,%.1lf).\n%.1lf\n",center.x,center.y,r);
    }
    return 0;
}

BZOJ2956:模积和 数学

思路:

其实是一道简单题.关键是要注意到题目中的一个条件:\(i\not=j\)!!!

我们考虑用所有的情况减去\(i=j\)的情况.

所有的情况就很好求了.令\(f(n)=\sum_{i=1}^{n}n\%i\),则所有的情况为\(f(n)*{f(m)}\).

这玩意很容易分块求.

然后\(i=j\)的情况的贡献是:

\((n\%i)*(m\%i)=(n-i*\frac{n}{i})*(m-i*\frac{m}{i})=nm+i^2\frac{n}{i}\frac{m}{i}-n*i\frac{m}{i}-m*i\frac{n}{i}\)

这样发现分开之后都可以分块求.

于是就\(O(\sqrt{n})\)水过.

#include<cstdio>
#include<cstring>
#include<cctype>
#include<iostream>
#include<algorithm>
using namespace std;
 
typedef long long LL;
 
static const int mod=19940417,rev2=(mod+1)>>1,rev6=3323403;
 
inline void mul(int&x,int y){x=(LL)x*y%mod;}
inline void inc(int&x,int y){if((x+=y)>=mod)x-=mod;}
inline void dec(int&x,int y){x=x<y?x+mod-y:x-y;}
inline int calc(int n){//return sigma i^2
    int res=n;mul(res,(LL)(n+1)*(2*n+1)%mod),mul(res,rev6);return res;
}
inline int cal(int l,int r){
    int cl=calc(l-1),cr=calc(r);
    return cr<cl?cr+mod-cl:cr-cl;
}
 
inline int solve1(int n,int lim){//return sigma i*(n/i)
    int lst,t,res=0;
    for(int i=1;i<=lim;i=lst+1){
        lst=min(lim,n/(n/i));
        mul(t=n/i,(LL)(i+lst)*(lst-i+1)%mod),mul(t,rev2);
        inc(res,t);
    }return res;
}
inline int solve2(int n){//return sigma n%i
    return ((LL)n*n%mod+mod-solve1(n,n))%mod;
}
inline int solve3(int n,int m,int lim){//return sigma i^2(n/i)(m/i)
    int lst,t,res=0;
    for(int i=1;i<=lim;i=lst+1){
        lst=min(n/(n/i),m/(m/i)),lst=min(lst,lim);
        mul(t=(LL)(n/i)*(m/i)%mod,cal(i,lst));
        inc(res,t);
    }return res;
}
 
int main(){
    int n,m;scanf("%d%d",&n,&m);register int i,j;
    int res=(LL)solve2(n)*solve2(m)%mod;
    dec(res,(LL)min(n,m)*n%mod*m%mod);
    inc(res,(LL)solve1(n,min(n,m))*m%mod);
    inc(res,(LL)solve1(m,min(n,m))*n%mod);
    dec(res,solve3(n,m,min(n,m)));
    printf("%d",res);
    return 0;
}

BZOJ1189:[HNOI2007]紧急疏散evacuate 二分+最大流

思路:

首先bfs求出每个空地到每个门的最短距离,注意中间不能经过任何门.好像是原题说过只要经过门就算是退出了.

随后我们二分答案,从原点到每块空地连容量为1的边,从每扇门到汇点连容量为(当前二分的答案)时间的边,对于每块空地和每扇门,若距离不超过时间,则从空地到门连一条容量为1的边.

好像唯一的trick就是从门到汇点的边?不过大家自己YY一下吧.

反正不难.

#include<cstdio>
#include<cstring>
#include<queue>
#include<cctype>
#include<iostream>
#include<algorithm>
using namespace std;
 
#define INF 0x3f3f3f3f
 
#define N 21
#define M 21
int n,m;
char s[M];
 
int ty[N][M];
 
int d[N][M];
 
static const int dx[]={-1,1,0,0},dy[]={0,0,1,-1};
struct Ins{
    int x,y;
    Ins(){}
    Ins(int _x,int _y):x(_x),y(_y){}
};
void bfs(int x,int y){
    static queue<Ins>q;
    d[x][y]=0,q.push(Ins(x,y));
    while(!q.empty()){
        Ins tmp=q.front();q.pop();
        for(int k=0;k<4;++k){
            int nx=tmp.x+dx[k],ny=tmp.y+dy[k];
            if(nx>=1&&nx<=n&&ny>=1&&ny<=m&&ty[nx][ny]==1&&d[nx][ny]==-1)
                d[nx][ny]=d[tmp.x][tmp.y]+1,q.push(Ins(nx,ny));
        }
    }
}
 
struct MaxflowSolver{
    static const int V=1010,E=200010;
    int head[V],next[E],end[E],flow[E],ind,d[V],inq[V];
    inline void reset(){ind=0,memset(head,-1,sizeof head);}
    inline void addedge(int a,int b,int f){int q=ind++;end[q]=b,next[q]=head[a],head[a]=q,flow[q]=f;}
    inline void make(int a,int b,int f){addedge(a,b,f),addedge(b,a,0);}
    inline bool bfs(int S,int T){
        static int q[V];int fr=0,ta=0;
        memset(d,-1,sizeof d),d[S]=0,q[ta++]=S;
        while(fr^ta){
            int i=q[fr++];for(int j=head[i];j!=-1;j=next[j])if(flow[j]&&d[end[j]]==-1)d[end[j]]=d[i]+1,q[ta++]=end[j];
        }return d[T]!=-1;
    }
    inline int dinic(int p,int T,int Maxflow){
        if(p==T)return Maxflow;
        int last=Maxflow;
        for(int j=head[p];j!=-1;j=next[j])if(flow[j]&&d[end[j]]==d[p]+1){
            int to=dinic(end[j],T,last>flow[j]?flow[j]:last);
            if(to){flow[j]-=to,flow[j^1]+=to;if(!(last-=to))return Maxflow;}
        }d[p]=-1;return Maxflow-last;
    }
    inline int Maxflow(int S,int T){
        int res=0;while(bfs(S,T))res+=dinic(S,T,INF);return res;
    }
}Stg;
 
int lab[N][M];
 
int main(){
    scanf("%d%d",&n,&m);register int i,j;
    for(i=1;i<=n;++i){
        scanf("%s",s+1);
        for(j=1;j<=m;++j)if(s[j]=='.')ty[i][j]=1;else if(s[j]=='D')ty[i][j]=2;
    }
     
    int id=0;for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(ty[i][j])lab[i][j]=++id;
     
    int L=0,R=1<<30,mid;
    while(L<R){
        mid=(L+R)>>1;
        Stg.reset();
        int cnt=0;
        for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(ty[i][j]==1)Stg.make(0,lab[i][j],1),++cnt;
        for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(ty[i][j]==2)Stg.make(lab[i][j],n*m+1,mid);
        for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(ty[i][j]==2){
            memset(d,-1,sizeof d),bfs(i,j);
            for(int p=1;p<=n;++p)for(int q=1;q<=m;++q)if(ty[p][q]==1&&d[p][q]!=-1)
                if(d[p][q]<=mid)Stg.make(lab[p][q],lab[i][j],1);
        }
        if(Stg.Maxflow(0,n*m+1)==cnt)R=mid;else L=mid+1;
    }
     
    if(L==1<<30)puts("impossible");else printf("%d",L);
    return 0;
}

BZOJ2786:Ural1142 Relation 递推+高精度

思路:

令\(f[i][j]\)表示将\(i\)个数划分为\(j\)个集合的方案数.

则有递推式:

\[f[i][j]=f[i-1][j]*j+f[i-1][j-1]\]

于是\(ans(n)=\sum_{i=1}^{n}f[n][i]*fac(i)\)

数据范围小,随便写写.

#include<cstdio>
#include<cstring>
#include<cctype>
#include<iostream>
#include<algorithm>
using namespace std;
 
static const int mod=1e4;
struct Int{
    int d[100],l;
    Int():l(1){memset(d,0,sizeof d);}
    inline void operator=(const int&x){memset(d,0,sizeof d),l=1,d[0]=x;}
    inline void operator+=(const Int&B){
        l=l>B.l?l:B.l;
        for(int i=0;i<l;++i){
            d[i]+=B.d[i];
            if(d[i]>=mod)d[i+1]+=d[i]/mod,d[i]%=mod;
        }if(d[l])l++;
    }
    inline void operator*=(const int&x){
        int p=0;
        for(int i=0;i<l;++i)d[i]=(p+=d[i]*x)%mod,p/=mod;
        if(p)d[l++]=p;
    }
    inline Int operator*(const Int&B){
        Int C;
        for(int i=0;i<l;++i)for(int j=0;j<B.l;++j){
            C.d[i+j]+=d[i]*B.d[j];if(C.d[i+j]>=mod)C.d[i+j+1]+=C.d[i+j]/mod,C.d[i+j]%=mod;
        }
        for(C.l=l+B.l+1;!C.d[C.l-1];--C.l);
        return C;
    }
    inline void output(){
        printf("%d",d[l-1]);for(int i=l-2;i>=0;--i)printf("%04d",d[i]);
    }
};
 
Int f[51][51],fac[51],res[51];
 
int main(){
    int i,j;
    for(fac[0]=1,fac[1]=1,i=2;i<=50;++i)fac[i]=fac[i-1],fac[i]*=i;
    for(i=1;i<=50;++i){
        f[i][1]=1,f[i][i]=1;
        for(j=2;j<i;++j)f[i][j]=f[i-1][j],f[i][j]*=j,f[i][j]+=f[i-1][j-1];
    }
     
    for(i=1;i<=50;++i)for(j=1;j<=i;++j)res[i]+=f[i][j]*fac[j];
     
    int T;scanf("%d",&T);
    int x;while(T--)scanf("%d",&x),res[x].output(),puts("");
     
    return 0;
}

I'm 666?No,I'm wu.

通过最近疯狂的刷水,刷到了一个吉祥的数字.这就算留个纪念?

可是我真的是666么?不,我是污的不行(雾).

BZOJ2212:[Poi2011]Tree Rotations 平衡树启发式合并

思路:

显然左右子树内的逆序对并不会互相影响,那么会产生影响的仅仅是是否交换左右子树.

我们利用平衡树启发式合并,每一次用小的平衡树在大的平衡树里面询问小的有多少以及大的有多少决定是否交换.

然后暴力将小的平衡树合并到大的平衡树里面.

(貌似还可以用线段树启发式合并?不过我并不想写,虽然常数小)

#include<cstdio>
#include<cstring>
#include<cctype>
#include<climits>
#include<iostream>
#include<algorithm>
using namespace std;

typedef long long LL;

#define l ch[0]
#define r ch[1]

#define N 200010
struct Node{
	Node*ch[2];
	int v,siz,pr;
	Node():siz(0){}
	inline void up(){siz=l->siz+r->siz+1;}
}mem[4000010],*C=mem,Tnull,*null=&Tnull;
inline Node*Newnode(int _v){
	C->l=C->r=null;C->v=_v,C->siz=1,C->pr=rand();return C++;
}
inline void Rot(Node*&p,bool d){
	Node*q=p->ch[!d];p->ch[!d]=q->ch[d],q->ch[d]=p,p->up(),q->up(),p=q;
}
inline void Insert(Node*&p,int x){
	if(p==null){p=Newnode(x);return;}
	if(x<p->v){Insert(p->l,x);if(p->l->pr<p->pr)Rot(p,1);}
	else{Insert(p->r,x);if(p->r->pr<p->pr)Rot(p,0);}
	p->up();
}
inline int QueryLess(Node*p,int x){
	if(p==null)return 0;
	if(p->v>x)return QueryLess(p->l,x);
	else return p->l->siz+1+QueryLess(p->r,x);
}
inline int QueryMore(Node*p,int x){
	if(p==null)return 0;
	if(p->v<x)return QueryMore(p->r,x);
	else return p->r->siz+1+QueryMore(p->l,x);
}

int L[N<<1],R[N<<1],cnt,x,w[N<<1];
inline int read(){
	int q=++cnt;
	scanf("%d",&x);
	if(!x)L[q]=read(),R[q]=read();else w[q]=x;
	return q;
}

LL res=0,tans1,tans2;

Node*Root[N<<1];
static Node*q[N<<1];int fr,ta;
void dfs(int x){
	if(w[x]){Root[x]=Newnode(w[x]);return;}
	dfs(L[x]),dfs(R[x]);
	tans1=tans2=0;
	if(Root[L[x]]->siz<Root[R[x]]->siz){
		fr=ta=0;q[ta++]=Root[L[x]];
		while(fr^ta){
			Node*p=q[fr++];
			tans1+=QueryLess(Root[R[x]],p->v),tans2+=QueryMore(Root[R[x]],p->v);
			if(p->l!=null)q[ta++]=p->l;if(p->r!=null)q[ta++]=p->r;
		}
		res+=min(tans1,tans2);
		fr=ta=0;q[ta++]=Root[L[x]];
		while(fr^ta){
			Node*p=q[fr++];
			Insert(Root[R[x]],p->v);
			if(p->l!=null)q[ta++]=p->l;if(p->r!=null)q[ta++]=p->r;
		}
		Root[x]=Root[R[x]];
	}
	else{
		fr=ta=0;q[ta++]=Root[R[x]];
		while(fr^ta){
			Node*p=q[fr++];
			tans1+=QueryLess(Root[L[x]],p->v),tans2+=QueryMore(Root[L[x]],p->v);
			if(p->l!=null)q[ta++]=p->l;if(p->r!=null)q[ta++]=p->r;
		}
		res+=min(tans1,tans2);
		fr=ta=0;q[ta++]=Root[R[x]];
		while(fr^ta){
			Node*p=q[fr++];
			Insert(Root[L[x]],p->v);
			if(p->l!=null)q[ta++]=p->l;if(p->r!=null)q[ta++]=p->r;
		}
		Root[x]=Root[L[x]];
	}
}

int main(){
#ifndef ONLINE_JUDGE
	freopen("tt.in","r",stdin);
#endif
	scanf("%d",&x);read();
	
	dfs(1);
	
	cout<<res<<endl;
	
	return 0;
}

BZOJ1187:[HNOI2007]神奇游乐园 插头dp

思路:

非常裸的插头dp,只不过这道题是只有一条回路,这个很容易实现,当遇到左右插头合并的时候直接更新答案,而并不是向下转移.

然后这个是要维护一个左右插头,我们用三进制来维护.

具体转移时有几个细节:

不能在边界上插插头!

更新答案时必须确认当前只有左右插头各一个!

我写了一个结构体来维护插头,还是蛮好写的.

#include<cstdio>
#include<cstring>
#include<cctype>
#include<climits>
#include<iostream>
#include<algorithm>
using namespace std;

int l;
struct Line{
	int d[7];
	Line(){memset(d,0,sizeof d);}
	inline int hash(){
		int re=0,mi=1;for(int i=0;i<l;++i,mi*=3)re+=d[i]*mi;return re;
	}
	inline void flush(){
		for(int i=l-1;i>=1;--i)d[i]=d[i-1];d[0]=0;
	}
	inline int getleftins(int x){
		int sav=0;
		for(int i=x;i>=0;--i){
			if(d[i]==2)++sav;else if(d[i]==1)--sav;
			if(sav==0)return i;
		}
	}
	inline int getrightins(int x){
		int sav=0;
		for(int i=x;i<l;++i){
			if(d[i]==1)++sav;else if(d[i]==2)--sav;
			if(sav==0)return i;
		}
	}
	Line change(int ins,int x){
		Line re=*this;re.d[ins]=x;return re;
	}
	inline int getlnum(){
		int re=0;for(int i=0;i<l;++i)re+=d[i]==1;return re;
	}
	inline int getrnum(){
		int re=0;for(int i=0;i<l;++i)re+=d[i]==2;return re;
	}
};
struct Hashset{
	int ins[2187],w[2187],ind;
	Line s[2187];
	inline void reset(){ind=0,memset(ins,-1,sizeof ins);}
	inline void insert(Line l,int _w){
		int hash=l.hash();
		if(ins[hash]==-1)ins[hash]=ind,w[ind]=_w,s[ind++]=l;
		else w[ins[hash]]=max(w[ins[hash]],_w);
	}
}S[2];

#define N 110
#define M 10
int w[N][M];

int main(){
	int n,m;scanf("%d%d",&n,&m);register int i,j,k;l=m+1;
	for(i=1;i<=n;++i)for(j=1;j<=m;++j)scanf("%d",&w[i][j]);
	
	int now=0,last=1;
	Line begin;S[now].reset(),S[now].insert(begin,0);
	
	int res=-1<<30;
	Line nowst;int nowf,d1,d2,ins1,ins2;
	for(i=1;i<=n;++i){
		for(j=1;j<=m;++j){
			swap(now,last),S[now].reset();
			for(k=0;k<S[last].ind;++k){
				nowst=S[last].s[k];nowf=S[last].w[k];
				d1=nowst.d[j-1],d2=nowst.d[j];
				if(d1==0&&d2==0){
					S[now].insert(nowst,nowf);
					if(i!=n&&j!=m)S[now].insert(nowst.change(j-1,1).change(j,2),nowf+w[i][j]);
				}
				else if(d1==0||d2==0){
					if(i!=n)S[now].insert(nowst.change(j-1,d1+d2).change(j,0),nowf+w[i][j]);
					if(j!=m)S[now].insert(nowst.change(j-1,0).change(j,d1+d2),nowf+w[i][j]);
				}
				else if(d1==1&&d2==1){
					ins1=nowst.getrightins(j),ins2=nowst.getrightins(j-1);
					S[now].insert(nowst.change(j-1,0).change(j,0).change(ins1,1).change(ins2,2),nowf+w[i][j]);
				}
				else if(d1==1&&d2==2){
					if(nowst.getlnum()==1&&nowst.getrnum()==1)res=max(res,nowf+w[i][j]);
				}
				else if(d1==2&&d2==1){
					S[now].insert(nowst.change(j-1,0).change(j,0),nowf+w[i][j]);
				}
				else if(d1==2&&d2==2){
					ins1=nowst.getleftins(j),ins2=nowst.getleftins(j-1);
					S[now].insert(nowst.change(j-1,0).change(j,0).change(ins1,1).change(ins2,2),nowf+w[i][j]);
				}
			}
		}
		swap(now,last),S[now].reset();
		for(k=0;k<S[last].ind;++k){
			nowst=S[last].s[k],nowf=S[last].w[k];
			nowst.flush();
			S[now].insert(nowst,nowf);
		}
	}
	
	printf("%d",res);
	
	return 0;
}
		

BZOJ1822:[JSOI2010]Frozen Nova 冷冻波 二分+计算几何+网络流

思路:

一眼网络流,套一个二分就行了.

关键是如何判断巫妖和精灵之间能否有圆阻挡.

一开始我是进行了复杂的判断:如果线段平行于x轴或者平行于y轴直接特判;

否则求出圆心到直线的距离,如果距离不超过半径,则用一个简单的公式求出圆心到直线最近点的横坐标,利用这个横坐标与线段两个端点的横坐标比较,得出结果.

可是数据跟我想的根本不是一回事啊啊啊!

根本只需要判断距离啊啊啊!!而且相切的情况不算相交啊啊啊!!

什么坑爹数据,我是领教到了.

#include<cstdio>
#include<cstring>
#include<climits>
#include<iostream>
#include<algorithm>
using namespace std;

typedef long long LL;

#define INF 0x3f3f3f3f

#define N 210
#define M 210
#define P 210

struct MaxflowSolver{
	static const int V=410;
	static const int E=100010;
	int head[V],next[E],end[E],flow[E],ind,d[V];
	inline void reset(){ind=0,memset(head,-1,sizeof head);}
	inline void addedge(int a,int b,int f){int q=ind++;end[q]=b,next[q]=head[a],head[a]=q,flow[q]=f;}
	inline void make(int a,int b,int f){addedge(a,b,f),addedge(b,a,0);}
	inline bool bfs(int S,int T){
		static int q[V];int fr=0,ta=0;
		memset(d,-1,sizeof d),d[S]=0,q[ta++]=S;
		while(fr^ta){
			int i=q[fr++];for(int j=head[i];j!=-1;j=next[j])if(flow[j]&&d[end[j]]==-1)d[end[j]]=d[i]+1,q[ta++]=end[j];
		}return d[T]!=-1;
	}
	inline int dinic(int p,int T,int Maxflow){
		if(p==T)return Maxflow;
		int last=Maxflow;
		for(int j=head[p];j!=-1;j=next[j])if(flow[j]&&d[end[j]]==d[p]+1){
			int to=dinic(end[j],T,last>flow[j]?flow[j]:last);
			if(to){flow[j]-=to,flow[j^1]+=to;if(!(last-=to))return Maxflow;}
		}d[p]=-1;return Maxflow-last;
	}
	inline int Maxflow(int S,int T){
		int res=0;while(bfs(S,T))res+=dinic(S,T,INF);return res;
	}
}Stg;
int x1[N],y1[N],x2[M],y2[M],lim[N],cd[N],ox[P],oy[P],r[P];

int G[N][M];

inline int sqr(int x){return x*x;}



int main(){
	int n,m,p;scanf("%d%d%d",&n,&m,&p);register int i,j,k;
	for(i=1;i<=n;++i)scanf("%d%d%d%d",&x1[i],&y1[i],&lim[i],&cd[i]);
	for(i=1;i<=m;++i)scanf("%d%d",&x2[i],&y2[i]);
	for(i=1;i<=p;++i)scanf("%d%d%d",&ox[i],&oy[i],&r[i]);
	
	int a,b,c;
	for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(sqr(x1[i]-x2[j])+sqr(y1[i]-y2[j])<=sqr(lim[i])){
		bool ok=1;
		for(k=1;k<=p&&ok;++k){
			a=y1[i]-y2[j],b=x2[j]-x1[i],c=(x1[i]-x2[j])*y1[i]-(y1[i]-y2[j])*x1[i];
			if((LL)(a*ox[k]+b*oy[k]+c)*(a*ox[k]+b*oy[k]+c)<(LL)r[k]*r[k]*(a*a+b*b))ok=0;
		}
		if(ok)G[i][j]=1;
	}
	
	int L=0,R=1<<30,mid;
	while(L<R){
		mid=(L+R)>>1;
		Stg.reset();
		for(i=1;i<=n;++i)Stg.make(0,i,mid/cd[i]+1);
		for(i=1;i<=m;++i)Stg.make(n+i,n+m+1,1);
		for(i=1;i<=n;++i)for(j=1;j<=m;++j)if(G[i][j])Stg.make(i,n+j,1);
		if(Stg.Maxflow(0,n+m+1)==m)R=mid;else L=mid+1;
	}
	
	if(L==1<<30)puts("-1");else printf("%d",L);
	return 0;
}

BZOJ3413:匹配 后缀自动机

思路:

这道题真心是一道字符串好题.我要不是膜拜了秦神的代码现在还看不出个所以然...(果然太弱不解释)

 

首先呢,题目的意思其实就是让我们求询问串与原串的每一个后缀的LCP之和.但是呢,有一个Trick,那就是找到整个串之后就不找啦!

那么先考虑一种简单的情况,就是原串中并不存在询问串.

那么,我们在逆序后缀树上走,每走到一个位置,就看一下当前串出现了几次.这样,实际上每次统计的都是这一位的贡献.

现在有限制.我们可以预先处理出询问串第一次出现的位置.那么我们可以离线处理,利用dfs序来维护逆序后缀树,然后将所有询问按照限制从小到大累加贡献.

 

有点不好说.详情见代码.

#include<cstdio>
#include<cstring>
#include<cctype>
#include<iostream>
#include<algorithm>
using namespace std;
  
typedef long long LL;
  
#define N 100010
#define M 3000010
#define Q 500010
int tranc[N<<1][10],pa[N<<1],len[N<<1],first[N<<1],ins[N],cnt,root,last;
inline int newnode(int l){
    len[++cnt]=l;return cnt;
}
  
char s[N];
  
namespace Fio{
    inline int getc(){
        static const int L=1<<15;static char buf[L],*S=buf,*T=buf;
        if(S==T){T=(S=buf)+fread(buf,1,L,stdin);if(S==T)return EOF;}
        return*S++;
    }
    inline bool digit(char c){return c>='0'&&c<='9';}
    template<typename T>inline void Get(T&x){
        int c;while(!digit(c=getc()));x=c-'0';while(digit(c=getc()))x=(x<<1)+(x<<3)+c-'0';
    }
    char buf[500010*12],*o=buf;
    template<typename T>inline void print(T x){
        static int stk[100];int top=0;
        if(!x)*o++=48;else{for(;x;x/=10)stk[++top]=x%10;for(int i=top;i>=1;--i)*o++=48+stk[i];}
        *o++='\n';
    }
    inline void flush(){fwrite(buf,1,o-buf,stdout);}
}
using namespace Fio;
  
namespace Tree{
    int head[N<<1],next[N<<1],end[N<<1];
    inline void addedge(int a,int b){
        static int q=1;end[q]=b,next[q]=head[a],head[a]=q++;
    }
    int in[N<<1],out[N<<1],tclock;
    inline void dfs(int x){
        in[x]=++tclock;
        for(int j=head[x];j;j=next[j])dfs(end[j]);
        out[x]=tclock;
    }
    int A[N<<1];
    inline int ask(int x){int r=0;for(;x;x-=x&-x)r+=A[x];return r;}
    inline void mdf(int x,int add){for(;x<=cnt;x+=x&-x)A[x]+=add;}
    inline void addnode(int x){mdf(in[x],1);}
    inline int asksubtree(int x){return ask(out[x])-ask(in[x]-1);}
}
 
namespace Query{
    int head[N],next[M],end[M],lab[M];
    inline void insert(int a,int b,int _lab){
        static int q=1;end[q]=b,next[q]=head[a],head[a]=q,lab[q++]=_lab;
    }
}
  
LL res[Q];
char Read[100010];
  
char ch;
int main(){
    int n;Get(n);register int i,j;
      
    int id=0;while(!digit(ch=getc()));s[++id]=ch;
    while(digit(ch=getc()))s[++id]=ch;  
     
    root=last=newnode(0);
    int p,q,np,nq,y;
    for(i=1;i<=n;++i){
        ins[i]=np=newnode(len[last]+1);
        for(y=s[i]-'0',p=last;p&&!tranc[p][y];p=pa[p])tranc[p][y]=np;
        if(!p)pa[np]=root;
        else{
            q=tranc[p][y];
            if(len[q]==len[p]+1)pa[np]=q;
            else{
                nq=newnode(len[p]+1),pa[nq]=pa[q],pa[q]=pa[np]=nq;
                memcpy(tranc[nq],tranc[q],sizeof tranc[q]);
                for(;p&&tranc[p][y]==q;p=pa[p])tranc[p][y]=nq;
            }
        }last=np;
    }
      
 
    for(i=1;i<=n;++i)for(p=ins[i];p;p=pa[p])if(!first[p])first[p]=i;else break;
     
    for(i=1;i<=cnt;++i)if(pa[i])Tree::addedge(pa[i],i);
    Tree::dfs(1);
     
     
     
    int cas;Get(cas);
    for(i=1;i<=cas;++i){
        int len=0;while(!digit(ch=getc()));Read[++len]=ch;
        while(digit(ch=getc()))Read[++len]=ch;
         
        bool find=1;int p=root;
        for(j=1;j<=len;++j){
            y=Read[j]-'0';if(tranc[p][y])p=tranc[p][y];else{find=0;break;}
        }
        if(!find){
            res[i]=n;
            for(p=root,j=1;j<=len;++j){
                y=Read[j]-'0';if(tranc[p][y])p=tranc[p][y],Query::insert(n,p,i);else break;
            }
        }
        else{
            res[i]=first[p]-len;
            int firstins=first[p];
            for(p=root,j=1;j<=len;++j){
                y=Read[j]-'0',Query::insert(firstins-len+j,p=tranc[p][y],i);
            }
        }
    }
     
    for(i=1;i<=n;++i){
        Tree::addnode(ins[i]);
        for(j=Query::head[i];j;j=Query::next[j])res[Query::lab[j]]+=Tree::asksubtree(Query::end[j]);
    }
      
    for(i=1;i<=cas;++i)Fio::print(res[i]);
 
    Fio::flush();
    return 0;
}