zoukankan      html  css  js  c++  java
  • Ybtoj-排列计数【矩阵乘法,分块幂】

    正题

    题目链接:http://noip.ybtoj.com.cn/contest/596/problem/1


    题目大意

    (T)组询问给出(n)求有多少个(n)的排列满足第一个是(1)并且相邻的差不超过(2)

    (1leq Tleq 10^6,1leq nleq 10^9)


    解题思路

    考虑一下如果我们要不断向前填满前面的一段的话,那么填的方案就只有两种,((x,x+1))((x,x+2,x+1,x+3)),这样一下会跳(1)或者(3)

    然后还有一种方法是一直往前跳两格然后再跳回来,但是这样就直接结束了。

    (f_i)表示按照最前面那两种方法铺(i)个的方案,那么答案就是((sum_{i=1}^nf_{i})-f_{n-1}),减去(n-1)是因为会被(f_n)算重。

    然后矩阵乘法+分块预处理光速幂做就好了。

    时间复杂度(O(4^3(sqrt {10^9}+T)))


    code

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<vector>
    #define ll long long
    #define file(x) freopen("data"#x".in","r",stdin);freopen("data"#x".out","w",stdout);
    using namespace std;
    const ll N=5e5+10;
    struct node{
    	ll to,next;
    }a[N<<1];
    ll n,m,tot=1,ls[N],fa[N],dep[N],s[N],f[N],ans,num;
    vector<ll> T[N];
    void addl(ll x,ll y){
    	a[++tot].to=y;
    	a[tot].next=ls[x];
    	ls[x]=tot;return;
    }
    void dfs(ll x,ll from){
    	dep[x]=dep[fa[x]]+1;
    	for(ll i=ls[x];i;i=a[i].next){
    		ll y=a[i].to;
    		if(i==from)continue;
    		else if(dep[y]){
    			if(dep[y]<=dep[x])
    				s[x]++,s[fa[y]]--;
    			continue;
    		}
    		T[x].push_back(y);
    		fa[y]=x;dfs(y,i^1);
    		s[x]+=s[y];
    	}
    	num+=(s[x]==0);
    	return;
    }
    void calc(ll x){
    	f[x]=(s[x]==0);dep[x]=1;
    	for(ll i=0;i<T[x].size();i++){
    		ll y=T[x][i];
    		calc(y);
    		ans+=f[x]*dep[y]+f[y]*dep[x];
    		dep[x]+=dep[y];
    		f[x]+=(s[x]==0)*dep[y]+f[y];
    	}
    	return;
    }
    signed main()
    {
    	file(2);
    	scanf("%lld%lld",&n,&m);
    	for(ll i=1;i<=m;i++){
    		ll x,y;
    		scanf("%lld%lld",&x,&y);
    		addl(x,y);addl(y,x);
    	}
    	dfs(1,0);
    	calc(1);
    	printf("%lld
    ",n*(n-1)/2ll*num-ans);
    	return 0;
    }
    
  • 相关阅读:
    也用一下Windows Live Writer
    常用16种iOS视图切换动画
    iPad平板电脑程序开发基础规则(源址:http://www.cocoachina.com/bbs/simple/?t82559.html)
    NSXMLParser详解
    UIScrollView用法
    使用UIActivityIndicatorView 和多线程
    为视图加边框
    iPhone SDK开发基础之iPhone程序框架
    NSXMLParser详解(事例)
    iOS开发 小知识点
  • 原文地址:https://www.cnblogs.com/QuantAsk/p/15371467.html
Copyright © 2011-2022 走看看