题目链接

BZOJ2322

题解

鉴于BZOJ2115,要完成此题,就简单得多了

对图做一遍\(dfs\),形成\(dfs\)树,从根到每个点的路径形成一个权值,而每个返祖边形成一个环

我们从根出发去走一个环再回到根,最终会异或上环的权值而又回到根

所以环是可以任意选的

我们把环的权值丢进线性基,记线性基中有\(tot\)各位置,那么环的权值异或和方案数就是\(2^{tot}\)

我们将剩余路径权值放入线性基中消元后去重,剩余的路径权值是和线性基中的权值线性无关的

但路径只能选一个,记有\(x\)个线性无关的非\(0\)路径

除去\(0\),那么答案就是

\[(x + 1)2^{tot} - 1
\]

但这题动态删边,常规操作,离线改为加边

加边后有\(3\)种情况:

  1. 两端点都没访问过,那么不会产生任何影响
  2. 有一个端点访问过,那么现在就可以继续\(dfs\)到另一个没访问过的端点
  3. 两端点都访问过,就形成一个环,更新线性基即可

每次线性基被更新时,都要取出所有路径消一次元

每次路径加入集合前,都要放入线性基消元

集合去重用\(set\)即可

由于线性基只会被更新\(O(logw)\)次,路径只有\(O(nlogn)\)个

所以复杂度为\(O(nlognlogw)\)

#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<vector>
#include<queue>
#include<cmath>
#include<map>
#include<set>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define cls(s,v) memset(s,v,sizeof(s))
#define mp(a,b) make_pair<int,int>(a,b)
#define cp pair<int,int>
using namespace std;
const int maxn = 50005,maxm = 100005,INF = 0x3f3f3f3f;
inline LL read(){
LL out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = 0; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 1) + (out << 3) + c - 48; c = getchar();}
return flag ? out : -out;
}
const int N = 63;
set<LL> S;
set<LL>::iterator it;
LL bin[maxn],d[maxn],A[70],c[maxn],ci,ansi,ans[maxn];
int tot,flag;
int n,m,Q,pos[maxn],out[maxn],q[maxn],vis[maxn];
int h[maxn],ne = 1,fa[maxn];
struct EDGE{int to,nxt,id; LL w;}ed[maxn << 1];
inline void build(int u,int v,int i,LL w){
ed[++ne] = (EDGE){v,h[u],i,w}; h[u] = pos[i] = ne;
ed[++ne] = (EDGE){u,h[v],i,w}; h[v] = ne;
}
bool ins(LL x){
for (int i = N; ~i; i--){
if (x & bin[i]){
if (!A[i]){A[i] = x; tot++; return true;}
else x ^= A[i];
}
}
return false;
}
LL check(LL x){
for (int i = N; ~i; i--) if ((x & bin[i]) && A[i]) x ^= A[i];
return x;
}
void dfs(int u){
vis[u] = true; LL x;
if (x = check(d[u])) S.insert(x);
Redge(u) if (!out[ed[k].id] && (to = ed[k].to) != fa[u]){
if (!vis[to]) fa[to] = u,d[to] = d[u] ^ ed[k].w,dfs(to);
else if (ins(d[to] ^ d[u] ^ ed[k].w)) flag = true;
}
}
void upd(){
if (!flag) return;
it = S.begin(); ci = 0;
while (it != S.end()) c[++ci] = *it,it++;
S.clear(); LL x;
for (int i = 1; i <= ci; i++)
if (x = check(c[i])) S.insert(x);
}
void rebuild(int k){
int u = ed[pos[k] ^ 1].to,v = ed[pos[k]].to; LL w = ed[pos[k]].w;
out[k] = false; flag = false;
if (!vis[u] && !vis[v]) return;
else if (vis[u]) fa[v] = u,d[v] = d[u] ^ w,dfs(v),upd();
else if (vis[v]) fa[u] = v,d[u] = d[v] ^ w,dfs(u),upd();
else ins(d[v] ^ d[u] ^ w),upd();
}
void print(){ans[ansi--] = 1ll * (S.size() + 1) * bin[tot] - 1;}
void work(){
upd(); print();
for (int i = Q; i; i--){
rebuild(q[i]);
print();
}
}
int main(){
bin[0] = 1; for (int i = 1; i <= N; i++) bin[i] = bin[i - 1] << 1ll;
n = read(); m = read(); Q = read();
int a,b; LL w;
REP(i,m){
a = read(); b = read(); w = read();
build(a,b,i,w);
}
REP(i,Q) out[q[i] = read()] = true;
ansi = Q; dfs(1);
work();
for (int i = 0; i <= Q; i++) printf("%lld\n",ans[i]);
return 0;
}

BZOJ2322 [BeiJing2011]梦想封印 【set + 线性基】的更多相关文章

  1. 【线性基】bzoj2322: [BeiJing2011]梦想封印

    线性基的思维题+图常见套路 Description 渐渐地,Magic Land上的人们对那座岛屿上的各种现象有了深入的了解. 为了分析一种奇特的称为梦想封印(Fantasy Seal)的特技,需要引 ...

  2. 【BZOJ 2322】[BeiJing2011]梦想封印 利用"环基"+线性基特征值

    很容易想到离线加边并且把环和链拆开搞(就是对于每个终点求出起点到他的路径(其实就是dfs树),然后bzoj2115),而且维护也很简单,然而我们发现不同的终点可能得到相同的值,这就是我们遇到的最大的问 ...

  3. BZOJ2322: [BeiJing2011]梦想封印

    Description 渐渐地,Magic Land上的人们对那座岛屿上的各种现象有了深入的了解. 为了分析一种奇特的称为梦想封印(Fantasy Seal)的特技,需要引入如下的概念: 每一位魔法的 ...

  4. 【BZOJ2322】[BeiJing2011]梦想封印 高斯消元求线性基+DFS+set

    [BZOJ2322][BeiJing2011]梦想封印 Description 渐渐地,Magic Land上的人们对那座岛屿上的各种现象有了深入的了解. 为了分析一种奇特的称为梦想封印(Fantas ...

  5. [BZOJ 2322][BeiJing2011]梦想封印

    梦想封印 题意 原题面: Problem 2322. -- [BeiJing2011]梦想封印 2322: [BeiJing2011]梦想封印 Time Limit: 20 Sec  Memory L ...

  6. BZOJ 2460: [BeiJing2011]元素 贪心,线性基

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2460 解法:从大到小排序,依次贪心的添加到当前集合就可以了,需要动态维护线性基.用拟阵证明 ...

  7. BZOJ_2460_[BeiJing2011]元素_线性基

    BZOJ_2460_[BeiJing2011]元素_线性基 Description 相传,在远古时期,位于西方大陆的 Magic Land 上,人们已经掌握了用魔 法矿石炼制法杖的技术.那时人们就认识 ...

  8. [BeiJing2011]元素[贪心+线性基]

    2460: [BeiJing2011]元素 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1245  Solved: 652[Submit][Stat ...

  9. BZOJ2460 [BeiJing2011]元素 【线性基】

    2460: [BeiJing2011]元素 Time Limit: 20 Sec  Memory Limit: 128 MB Submit: 1675  Solved: 869 [Submit][St ...

随机推荐

  1. python之Django实现商城从0到1

    dailyfresh-B2Cdailyfresh mall based on B2C model 基于B2C的天天生鲜商城 项目托管地址:https://github.com/Ylisen/daily ...

  2. 20155207 《网络对抗》exp4 恶意代码分析 学习总结

    20155207 <网络对抗> 恶意代码分析 学习总结 实践目标 1.是监控你自己系统的运行状态,看有没有可疑的程序在运行. 2.是分析一个恶意软件,就分析Exp2或Exp3中生成后门软件 ...

  3. 20155209林虹宇Exp4 恶意代码分析

    Exp4 恶意代码分析 系统运行监控 使用schtasks指令监控系统运行 新建一个txt文件,然后将txt文件另存为一个bat格式文件 在bat格式文件里输入以下信息 然后使用管理员权限打开cmd, ...

  4. 2017-2018 Exp8 Web基础 20155214

    目录 Exp8 Web基础 实验内容 建站过程 SQL注入 知识点 Exp8 Web基础 实验内容 实验环境 主机 Kali 靶机 Kali 实验工具 后台语言 'PHP' 服务器 'Apache' ...

  5. 2017-2018-2 20155230《网络对抗技术》实验1:PC平台逆向破解(5)M

    1.直接修改程序机器指令,改变程序执行流程 2.通过构造输入参数,造成BOF攻击,改变程序执行流 3.注入Shellcode并执行 4.实验感想 注:因为截图是全屏所以右键图片在新的标签页打开观看更加 ...

  6. 20155238 《JAVA程序设计》实验三(敏捷开发与XP实践)实验报告

    实验内容 敏捷开发与XP实践 XP基础 XP核心实践 相关工具 实验要求 1.没有Linux基础的同学建议先学习<Linux基础入门(新版)><Vim编辑器> 课程 2.完成实 ...

  7. Oracle出现与并行相关的ORA-00600时的调查方法

    出现了 ORA-00600[kxfpqsod_qc_sod], 如何调查呢? 例如:从trace 文件的 Call Stack,可以看到 Error: ORA-600 [kxfpqsod_qc_sod ...

  8. VS Code使用Git管理代码

    Visual Studio Code(简称VS Code)是一个轻量级且强大的代码编辑器,后台是微软,支持Windows.Mac和Linux操作系统,拥有丰富的插件生态系统,可通过安装插件来支持C++ ...

  9. Android Studio Xposed模块编写(二)

    阅读本文前,假设读者已经看过Android Studio Xposed模块编写(一)  相关环境已经搭建完成.本文演示案例与上文环境一致,不在赘述. 1.概述 Xposed是非常牛叉的一款hook框架 ...

  10. [穷尽]ADO.NET连接字符串

    微软提供的四种数据库连接方式: System.Data.OleDb.OleDbConnection System.Data.SqlClient.SqlConnection System.Data.Od ...