题目

首先我要吐槽,这题目就是坑,给那么多无用的信息,我还以为要根据提示才能做出来呢!

算法1

暴力,傻傻地跟着提示,纯暴力\(40\)分,高斯消元\(60\)分。

算法2

DP!一个显然的东西是,这个矩阵有很多地方都是\(0\),所以我们枚举的许多排列都是无用的。

设\(f(i,set)\),其中\(i\)表示计算到排列的第\(i\)个元素,或者说是到矩阵的第\(i\)行,\(set\)是一个集合,表示前一行哪些数字还没选,可知\(set\)的大小为\(2k\)(这样我们才能DP嘛)。\(f\)的值表示当前计算到的行列式的值。

然后转移的时候我们要统计新产生的逆序对,进而判断是否要乘\(-1\)。

时间复杂度\(O(2^{2k}nk)\)。

算法3

直接DP,不要想什么矩阵。

假设我们DP到了第\(i\)位,显然有用的信息只有\(i-k \sim i-1\)这\(k\)个点的连通性。

然后暴力出所有的状态(要最小表示法,只有\(52\)种状态),搞出它们之间的转移,然后直接矩阵乘法即可。时间复杂度\(O(52^3 \log n)\)。

代码

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <assert.h>
using namespace std; #ifdef debug
#define ep(...) fprintf(stderr, __VA_ARGS__)
#else
#define ep(...) assert(true)
#endif typedef long long i64; const int MAXK = 5;
const int MAXS = 60;
const int MOD = 65521;
const int LOGN = 60;
const int HASHSIZE = 1 << MAXK * 3; int cntS;
int k;
i64 n; struct MatrixB {
i64 A[MAXS][MAXS]; i64* operator [] (const int &x) {
return A[x];
}
}; struct MatrixA {
i64 A[MAXS]; i64 &operator [] (const int &x) {
return A[x];
}
}; void multi(MatrixB &A, MatrixB &B, MatrixB &C) {
for (int i = 0; i < cntS; i ++)
for (int j = 0; j < cntS; j ++) {
C[i][j] = 0;
for (int k = 0; k < cntS; k ++)
C[i][j] += A[i][k] * B[k][j];
C[i][j] %= MOD;
}
} void multi(MatrixA &A, MatrixB &B, MatrixA &C) {
for (int i = 0; i < cntS; i ++) {
C[i] = 0;
for (int j = 0; j < cntS; j ++)
C[i] += B[j][i] * A[j];
C[i] %= MOD;
}
} MatrixB B[LOGN]; struct Status {
int A[MAXK]; int &operator [] (const int &x) {
return A[x];
} int transform() {
int ret = 0;
for (int i = k - 1; i >= 0; i --) {
ret <<= 3;
ret |= A[i];
}
return ret;
} void show() {
#ifdef debug
for (int i = 0; i < k; i ++)
ep("%d ", A[i]);
ep("\n");
#endif
}
}; int hash[HASHSIZE]; #define test(s, i) (((s) >> (i)) & 1) Status transform(int x) {
Status ret;
for (int i = 0; i < k; i ++) {
ret[i] = x & 7;
x >>= 3;
}
return ret;
} int dfs(int f) {
Status cur = transform(f);
if (hash[f] == -1) {
hash[f] = cntS ++;
int cnt[MAXK];
fill(cnt, cnt + k, 0);
for (int i = 0; i < k; i ++)
cnt[cur[i]] ++;
for (int s = 0; s < 1 << k; s ++) {
int con = 1;
Status nxt = cur;
int in = -1; if (cnt[0] == 1 && test(s, 0) == 0) continue; for (int i = 0; i < k; i ++)
if (test(s, i)) {
con *= cnt[i];
if (in == -1) in = i;
else {
for (int j = 0; j < k; j ++)
if (nxt[j] == i) nxt[j] = in;
}
}
if (! con) continue; static int mapTo[MAXK];
fill(mapTo, mapTo + MAXK, -1);
int z = 0;
for (int i = 1; i < k; i ++)
if (mapTo[nxt[i]] == -1) {
mapTo[nxt[i]] = z ++;
}
for (int i = 0; i + 1 < k; i ++)
nxt[i] = mapTo[nxt[i + 1]];
nxt[k - 1] = in == -1 || mapTo[in] == -1 ? z : mapTo[in]; cur.show();
ep("%d %d\n", s, con);
nxt.show();
ep("--------------\n"); int h = nxt.transform();
int idx = dfs(h);
B[0][hash[f]][idx] += con;
B[0][hash[f]][idx] %= MOD;
}
}
return hash[f];
} MatrixA A, tA; void dfs2(int cur, Status s, int combination) {
if (cur == k) {
s.show();
ep("%d\n-----------\n", combination);
int idx = hash[s.transform()];
A[idx] += combination;
A[idx] %= MOD;
}
else {
int cnt[MAXK];
fill(cnt, cnt + cur + 1, 0);
for (int i = 0; i < cur; i ++)
cnt[s[i]] ++;
for (int S = 0; S < 1 << cur; S ++) {
int con = 1;
Status nxt = s;
nxt[cur] = -1;
for (int i = 0; i < cur; i ++)
if (test(S, i)) {
con *= cnt[i];
if (nxt[cur] == -1) nxt[cur] = i;
else {
for (int j = 0; j < cur; j ++)
if (nxt[j] == i) nxt[j] = nxt[cur];
}
}
if (! con) continue;
static int mapTo[MAXK];
int z = 0;
fill(mapTo, mapTo + cur, -1);
for (int i = 0; i < cur; i ++) {
if (mapTo[nxt[i]] == -1) {
mapTo[nxt[i]] = z ++;
}
nxt[i] = mapTo[nxt[i]];
} if (nxt[cur] == -1) {
int x = 0;
while (true) {
bool ok = true;
for (int i = 0; i < cur; i ++)
if (nxt[i] == x) {
ok = false;
break;
}
if (! ok) x ++;
else break;
}
nxt[cur] = x;
}
dfs2(cur + 1, nxt, combination * con);
}
}
} int main() {
#ifndef ONLINE_JUDGE
freopen("count.in", "r", stdin);
freopen("count.out", "w", stdout);
#endif scanf("%d%lld", &k, &n);
memset(hash, -1, sizeof hash);
dfs(0);
ep("find %d\n", cntS);
ep("-----------\n");
for (int i = 0; i < cntS; i ++) {
for (int j = 0; j < cntS; j ++)
ep("%d ", B[0][i][j]);
ep("\n");
}
ep("===================\n");
for (int i = 1; i < LOGN; i ++)
multi(B[i - 1], B[i - 1], B[i]); Status s;
dfs2(0, s, 1);
ep("-----------\n");
for (int i = 0; i < cntS; i ++)
ep("%d ", A[i]);
ep("\n");
ep("===================\n"); n -= k;
for (int i = 0; i < LOGN; i ++)
if (n >> i & 1) {
tA = A;
multi(tA, B[i], A);
} i64 ans = 0;
for (int i = 0; i < HASHSIZE; i ++)
if (hash[i] != -1) {
Status x = transform(i);
bool ok = true;
for (int j = 1; j < k; j ++)
if (x[j] != x[j - 1]) {
ok = false;
break;
}
if (ok) ans += A[hash[i]];
}
ans %= MOD;
printf("%d\n", (int) ans);
ep("%lld\n", ans); return 0;
}

NOI2007 生成树计数的更多相关文章

  1. BZOJ1494 [NOI2007]生成树计数

    题意 F.A.Qs Home Discuss ProblemSet Status Ranklist Contest 入门OJ ModifyUser  autoint Logout 捐赠本站 Probl ...

  2. [BZOJ1494][NOI2007]生成树计数 状压dp 并查集

    1494: [NOI2007]生成树计数 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 793  Solved: 451[Submit][Status][ ...

  3. [NOI2007]生成树计数环形版

    NOI2007这道题人类进化更完全之后出现了新的做法 毕姥爷题解: 于是毕姥爷出了一道环形版的这题(test0814),让我们写这个做法 环形的情况下,k=5的时候是162阶递推. 求这个递推可以用B ...

  4. [BZOJ1494]生成树计数

    [BZOJ1494] [NOI2007]生成树计数 Description 最近,小栋在无向连通图的生成树个数计算方面有了惊人的进展,他发现:·n个结点的环的生成树个数为n.·n个结点的完全图的生成树 ...

  5. 【BZOJ1494】【NOI2007】生成树计数(动态规划,矩阵快速幂)

    [BZOJ1494][NOI2007]生成树计数(动态规划,矩阵快速幂) 题面 Description 最近,小栋在无向连通图的生成树个数计算方面有了惊人的进展,他发现: ·n个结点的环的生成树个数为 ...

  6. 【BZOJ1002】【FJOI2007】轮状病毒(生成树计数)

    1002: [FJOI2007]轮状病毒 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1766  Solved: 946[Submit][Status ...

  7. SPOJ 104 HIGH - Highways 生成树计数

    题目链接:https://vjudge.net/problem/SPOJ-HIGH 解法: 生成树计数 1.构造 基尔霍夫矩阵(又叫拉普拉斯矩阵) n阶矩阵 若u.v之间有边相连 C[u][v]=C[ ...

  8. Luogu P5296 [北京省选集训2019]生成树计数

    Luogu P5296 [北京省选集训2019]生成树计数 题目链接 题目大意:给定每条边的边权.一颗生成树的权值为边权和的\(k\)次方.求出所有生成树的权值和. 我们列出答案的式子: 设\(E\) ...

  9. Loj 2320.「清华集训 2017」生成树计数

    Loj 2320.「清华集训 2017」生成树计数 题目描述 在一个 \(s\) 个点的图中,存在 \(s-n\) 条边,使图中形成了 \(n\) 个连通块,第 \(i\) 个连通块中有 \(a_i\ ...

随机推荐

  1. iOS 滤镜 转载,原文见正文首行链接

    转载自:http://blog.sina.com.cn/s/blog_5fb39f9101018gv7.html 直接上代码了: // // ViewController.m // 图片模糊处理 // ...

  2. UVA 674 Coin Change(dp)

    UVA 674  Coin Change  解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid=87730#problem/ ...

  3. Jetty 8.1 安装 运行 部署

    链接地址:http://blog.csdn.net/lego2816/article/details/42650545 Java + MySQL 从Jfinal开始,第一步先走完整个流程,主要是发布部 ...

  4. OpenCV配置使用版

    在VS2010环境中应用Opencv,网上找到了很多配置方法,但大多都是老版本的,很多新手面对最新版本的Opencv无从下手,就给新手童鞋写了这么一篇超级详细的配置攻略,贴上来共享.要强调一点的就是, ...

  5. Yii2 composer win7安装新建项目流程

    一.首先下载 Composer-Setup.exe ,安装. 问题1:openSSL 问题,在php.ini  内去掉":"注释 问题2:browscap 问题 ,在php.ini ...

  6. 怎样在VC里面使用graphics.h绘图

    网上很多绘图程序和小游戏程序都是用的 TC,在 VC 下编译时提示错误:fatal error C1083: Cannot open include file: 'graphics.h': No su ...

  7. android listview需要呈现多个布局

    android listview需要呈现多个布局 之前的做法很笨 在getView()方法里面,不仅将viewHolder作为tag属性设置给convertView 还将当前的position作为ta ...

  8. android 项目中log信息的正确处理

    我们在做项目中,调试程序的时候通常是debug或者打log,特别是看一些json数据之类的,都喜欢打log看看,可是在项目上线时log信息一定不能被打印出来,大家能够平时下载一些做的烂的app,连接e ...

  9. 最全的LBS手机定位技术说明

    随着手机技术的发展定位方式也发生了非常大的变化.获取手机位置有非常多种方式. 第一种:CELL-ID定位原理 通过移动网络获取设备当前所在的Cell信息来获取设备当前位置.当设备位置更新设备会向当前服 ...

  10. LeetCode——Path Sum II

    Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the given su ...