「2017 山东一轮集训 Day5」苹果树
「2017 山东一轮集训 Day5」苹果树
\(n\leq 40\)
折半搜索+矩阵树定理。
没有想到折半搜索。
首先我们先枚举\(k\)个好点,我们让它们一定没有用的。要满足这个条件就要使它只能和坏点相连。其他的点没有要求。这样算出来了至少\(k\)个点没有用的生成树个数,我们要得到恰好\(k\)个点的生成树个数就简单容斥一下就好了。
然后我们要得到有\(k\)个点没有用的情况下的点集的方案数。看到\(40\)这个范围我们容易想到折半搜索。
然后就没了。
但是我没写容斥,写的枚举集合划分(被吊打
代码:
#include<bits/stdc++.h>
#define ll long long
#define N 45
using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;}
const ll mod=1e9+7;
int n;
ll lim;
int a[N];
ll c[N][N];
ll fac[N],ifac[N],g[N];
ll w[N][N];
ll sum;
ll ksm(ll t,ll x) {
ll ans=1;
for(;x;x>>=1,t=t*t%mod)
if(x&1) ans=ans*t%mod;
return ans;
}
ll f[N];
ll ned;
int good[N];
ll bad;
int num[N];
ll ans;
ll Gauss(ll a[][45],int n) {
ll ans=1;
for(int i=2;i<=n;i++) {
for(int j=i+1;j<=n;j++) {
if(a[j][i]) {
ans=ans*(mod-1)%mod;
swap(a[i],a[j]);
break;
}
}
if(!a[i][i]) return 0;
ans=ans*a[i][i]%mod;
for(int j=i+1;j<=n;j++) {
if(!a[j][i]) continue ;
ll tem=ksm(a[i][i],mod-2)*a[j][i]%mod;
for(int k=i;k<=n;k++) a[j][k]=(a[j][k]-tem*a[i][k]%mod+mod)%mod;
}
}
return ans;
}
vector<int>st;
ll cal() {
if(!f[num[1]]) return 0;
int tot=bad;
st.clear();
for(int i=1;i<=n;i++) for(int j=1;j<=num[i];j++) st.push_back(i);
for(int i=1;i<=n;i++) tot+=num[i];
memset(w,0,sizeof(w));
for(int i=0;i<st.size();i++) {
w[i+1][i+1]=st[i]*bad;
for(int j=st.size()+1;j<=tot;j++) {
w[i+1][j]=mod-st[i];
}
}
for(int i=st.size()+1;i<=tot;i++) {
w[i][i]=n-1;
for(int j=st.size()+1;j<=tot;j++)
if(i!=j) w[i][j]=mod-1;
for(int j=0;j<st.size();j++)
w[i][j+1]=mod-st[j];
}
ll ans=f[num[1]];
ans=ans*fac[n-bad-num[1]]%mod%mod;
for(int i=2;i<=n;i++) {
ans=ans*ksm(ifac[i],num[i])%mod;
ans=ans*ksm(g[i],num[i])%mod;
ans=ans*ifac[num[i]]%mod;
}
ll tem=Gauss(w,tot);
return ans*tem%mod;
}
int mid;
void dfs(int v,int lim,ll now,int cnt,vector<int>*a) {
if(v>lim) return ;
dfs(v+1,lim,now,cnt,a);
a[cnt+1].push_back(now+good[v]);
dfs(v+1,lim,now+good[v],cnt+1,a);
}
void meet_in_the_middle() {
mid=good[0]>>1;
vector<int>a[N],b[N];
dfs(1,mid,0,0,a);
dfs(mid+1,good[0],0,0,b);
for(int i=1;i<=mid;i++) sort(a[i].begin(),a[i].end());
for(int i=1;i<=mid;i++)
for(int j=0;j<a[i].size();j++)
if(a[i][j]>=ned) f[i]++;
for(int i=1;i<=good[0]-mid;i++) {
for(int j=0;j<b[i].size();j++) {
if(b[i][j]>=ned) f[i]++;
for(int k=1;k<=mid;k++) {
(f[i+k]+=a[k].end()-lower_bound(a[k].begin(),a[k].end(),ned-b[i][j]))%=mod;
}
}
}
}
int dep;
void dfs(int v,int res) {
if(!res) {
dep++;
(ans+=cal())%=mod;
return ;
}
if(!v) return ;
dfs(v-1,res);
for(int i=1;i*v<=res;i++) {
num[v]=i;
dfs(v-1,res-i*v);
num[v]=0;
}
}
bool cmp(int a,int b) {return a>b;}
int main() {
n=Get(),lim=Get();
fac[0]=1;
for(int i=1;i<=n;i++) fac[i]=fac[i-1]*i%mod;
ifac[n]=ksm(fac[n],mod-2);
for(int i=n-1;i>=0;i--) ifac[i]=ifac[i+1]*(i+1)%mod;
for(int i=0;i<=n;i++)
for(int j=0;j<=i;j++)
c[i][j]=(!j||i==j)?1:(c[i-1][j-1]+c[i-1][j])%mod;
for(int i=1;i<=n;i++) a[i]=Get();
for(int i=1;i<=n;i++) {
if(a[i]>0) {
good[++good[0]]=a[i];
sum+=a[i];
} else bad++;
}
ned=sum-lim;
if(sum<=lim) {
cout<<ksm(n,n-2)%mod;
return 0;
}
if(!bad) {cout<<0;return 0;}
sort(good+1,good+good[0]+1,cmp);
meet_in_the_middle();
g[0]=g[1]=1;
for(int i=2;i<=n;i++) g[i]=ksm(i,i-2)%mod;
dfs(good[0],good[0]);
cout<<ans;
return 0;
}
「2017 山东一轮集训 Day5」苹果树的更多相关文章
- 题解 「2017 山东一轮集训 Day5」苹果树
题目传送门 题目大意 给出一个 \(n\) 个点的图,每个点都有一个权值 \(f_i\) ,如果 \(f_i=-1\) 表示 \(i\) 这个点是坏的.定义一个点是有用的当且仅当它不是坏的,并且它连的 ...
- Loj #6073.「2017 山东一轮集训 Day5」距离
Loj #6073.「2017 山东一轮集训 Day5」距离 Description 给定一棵 \(n\) 个点的边带权的树,以及一个排列$ p\(,有\)q $个询问,给定点 \(u, v, k\) ...
- loj#6073. 「2017 山东一轮集训 Day5」距离(树链剖分 主席树)
题意 题目链接 Sol 首先对询问差分一下,我们就只需要统计\(u, v, lca(u, v), fa[lca(u, v)]\)到根的路径的贡献. 再把每个点与\(k\)的lca的距离差分一下,则只需 ...
- loj#6073. 「2017 山东一轮集训 Day5」距离(费用流)
题意 题目链接 Sol 我们可以把图行列拆开,同时对于行/列拆成很多个联通块,然后考虑每个点所在的行联通块/列联通块的贡献. 可以这样建边 从S向每个行联通块连联通块大小条边,每条边的容量为1,费用为 ...
- 「2017 山东一轮集训 Day5」字符串 (后缀自动机, 拓扑排序)
/** 首先通过SAM求出每个串本质不同的子串 然后发现转移不好处理整体的本质不同 形如串A可能状态有a,b,ab,空,串B可能状态有b,空两种, 那么我们需要处理ab + 空 和 a + b的情况 ...
- 「2017 山东一轮集训 Day5」距离
/* 写完开店再写这个题目顿时神清气爽, 腰也不疼了, 眼也不花了 首先考虑将询问拆开, 就是查询一些到根的链和点k的关系 根据我们开店的结论, 一个点集到一个定点的距离和可以分三部分算 那么就很简单 ...
- 「2017 山东一轮集训 Day5」字符串
题目 比较神仙的操作啊 首先先考虑一个串的做法,我们有两种:SA或SAM,其中SAM又有两种,拓扑图上的\(dp\)和\(parent\)上随便统计一下 显然这道题\(SA\)和\(parent\)树 ...
- LOJ6071. 「2017 山东一轮集训 Day5」字符串 [SAM]
LOJ 思路 这种计数题显然是要先把每一个合法的串用唯一的方法表示出来.(我连这都没想到真是无可救药了) 如何唯一?容易想到把前缀尽可能多地在第一个串填掉,然后填第二个,第三个-- 如何做到这样?可以 ...
- Loj #6069. 「2017 山东一轮集训 Day4」塔
Loj #6069. 「2017 山东一轮集训 Day4」塔 题目描述 现在有一条 $ [1, l] $ 的数轴,要在上面造 $ n $ 座塔,每座塔的坐标要两两不同,且为整点. 塔有编号,且每座塔都 ...
随机推荐
- SQL Server远程连接 provider: Named Pipes Provider, error: 40 解决方法
置SQLServer,允许远程连接 按照上面的文章一步步配置后,远程连接出现下面所示的报错(Navicat 和 SQL Server Management Studio) SQL Server Man ...
- 【Java每日一题】20170301
20170228问题解析请点击今日问题下方的“[Java每日一题]20170301”查看(问题解析在公众号首发,公众号ID:weknow619) package Mar2017; public cla ...
- Maven(五)Eclipse配置Maven插件
较新版本的Eclipse内置Maven插件 1. 设置installation 一般不适用内置的 指定核心程序的位置 2. 设置user settings 指定本地仓库的位置 1.如果setting. ...
- 6.方法_EJ
第38条: 检查参数的有效性 对于这一条,最常见的莫过于检查参数是否为null. 有时出现调用方未检查传入的参数是否为空,同时被调用方也没有检查参数是否为空,结果这就导致两边都没检查以至于出现null ...
- laravel中请求用例$request可用的一些方法小结
laravel中$request可用的一些方法小结 1,请求方法的获取 $method = $request->method(); 2,检测请求方法 $res = $request->is ...
- canvas-a12ellipse.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- window.requestAnimationFrame与Tween.js配合使用实现动画缓动效果
window.requestAnimationFrame 概述 window.requestAnimationFrame()这个方法是用来在页面重绘之前,通知浏览器调用一个指定的函数,以满足开发者操作 ...
- git 入门教程之变基合并
git 鼓励大量使用分支---"早建分支!多用分支!",这是因为即便创建再多的分支也不会造成存储或内存开销,并且分支的作用有助于我们分解逻辑工作,这样一样其实比维护单一臃肿分支要简 ...
- shell编程—变量(三)
在shell脚本中,变量分两种,系统变量和自定义变量. 系统默认变量是系统自带的一些变量,如path为路径变量 用户自定义变量为在编写吧脚本的时候自己定义的一些变量 变量名命名规则 首个字符必须为字母 ...
- mssql sql server上如何建一个只读视图–视图锁定的另类解决方案
转自:http://www.maomao365.com/?p=4508 <span style="color:red;font-weight:bold;">我们熟知一个 ...