首页 > 其他分享 >P7453 大魔法师 Sol

P7453 大魔法师 Sol

时间:2022-11-01 23:14:23浏览次数:85  
标签:space0 space int Sol times 魔法师 space1 P7453 bmatrix

从 OI - Wiki 上跑过来的。

首先看到区间操作,考虑线段树。

看到轮换式这样朴素线段树维护不了的东西,考虑矩阵乘法。

因为矩阵乘法具有结合律,所以线段树区间合并支持矩阵合并。

状态很好设,显然三个数组以及一个常数,\([A,B,C,x]\)。

下面令其每个元素的位置分别为 \(1,2,3,4\)。

这题大致解法就出来了,下面考虑细节。


\(1\) 至 \(3\) 操作如何解决

对于 \(1\) 操作,有 \([A,B,C,x]\leftarrow[A+B,B,C,x]\)。

考虑一个 \(4\times4\) 的矩阵 \(M\) 和原状态 \([A,B,C,x]\) 的关系。

对于第一列来说,第一列从上往下第 \(i\) 个元素 \(a_{i,1}\) 表示往目标状态的第一个位置添加 \(a_{i,1}\times A\)。

同理可推出 \(B,C,x\) 的规律。

那么很显然,当前矩阵应该为

\(\begin{bmatrix}1\space\space0\space\space0\space\space0\\1\space\space1\space\space0\space\space0\\0\space\space0\space\space1\space\space0\\0\space\space0\space\space0\space\space1\end{bmatrix}\)

目标位置 \(1\) 为 \(1\times A+1\times B+0\times C+0\times x=A+B\)。

即终止状态为 \([A+B,B,C,x]\)。

同理可以推出 \(2\) 操作和 \(3\) 操作的矩阵。

\(\begin{bmatrix}1\space\space0\space\space0\space\space0\\0\space\space1\space\space0\space\space0\\0\space\space1\space\space1\space\space0\\0\space\space0\space\space0\space\space1\end{bmatrix}\)

\(\begin{bmatrix}1\space\space0\space\space1\space\space0\\0\space\space1\space\space0\space\space0\\0\space\space0\space\space1\space\space0\\0\space\space0\space\space0\space\space1\end{bmatrix}\)


\(4\) 至 \(7\) 操作如何解决

同理,\(4\) 操作为 \([A,B,C,x]\leftarrow[A+v,B,C,x]\)。

考虑利用常数项 \(x\)。

特殊地,在线段树的叶子节点,\(x=1\)。

那么考虑 \([A,B,C,1]\leftarrow[A+v,B,C,1]\)。

考虑在目标位置 \(1\) 加入系数为 \(v\) 的常数项 \(1\)。具体地,矩阵长这样。

\(\begin{bmatrix}1\space\space0\space\space0\space\space0\\0\space\space1\space\space0\space\space0\\0\space\space0\space\space1\space\space0\\v\space\space0\space\space0\space\space1\end{bmatrix}\)

那么目标位置 \(1\) 为 \(1\times A+0\times B+0\times C+v\times 1=A+v\),搞定。

对于 \(5\) 操作,变为乘法,考虑把目标位置 \(2\) 变为系数为 \(v\) 的 \(B\),矩阵长这样。

\(\begin{bmatrix}1\space\space0\space\space0\space\space0\\0\space\space v\space\space0\space\space0\\0\space\space0\space\space1\space\space0\\0\space\space0\space\space0\space\space1\end{bmatrix}\)

目标位置 \(2\) 为 \(0\times A+v\times B+0\times C+0\times 1=v\times B\)。

同理有 \(6\) 操作,直接把目标位置 \(2\) 变为系数为 \(v\) 的常数项 \(1\)。

\(\begin{bmatrix}1\space\space0\space\space0\space\space0\\0\space\space 1\space\space0\space\space0\\0\space\space0\space\space0\space\space 0\\0\space\space0\space\space v\space\space1\end{bmatrix}\)

目标位置 \(3\) 为 \(0\times A+0\times B+0\times C+v\times 1=v\)。

另外线段树 pushup 时区间的长度(即操作自带的系数)为两个子区间的长度和而非 \(1\)。

考虑在 pushup 的时候更新常数项 \(x=r-l+1\),即区间长度,作为区间全体操作的系数。

由于所有要维护的信息都是可以直接合并的,所以对于 \(4\) 个元素 \([A,B,C,x]\) 直接在 pushup 时求和即可。

\(7\) 询问操作直接求区间和即可,返回一个结构体并且输出。


pushdown 怎么写

由于矩阵乘法具有结合律,所以在 pushdown 的时候可以直接把当前节点的懒标记乘进子结点的懒标记。

另外,初始化的时候记得将懒标记赋为单位矩阵,这样才满足初始时和一个序列做乘法不会错。

pushdown 完成后也要记得清空为单位矩阵。


另外,这题真的不卡常。

#include <bits/stdc++.h>
#define int long long
#define ls ((rt) << 1)
#define rs ((rt) << 1 | 1)
using namespace std;

const int sz = 4, N = 2.5e5 + 10;
const int Mod = 998244353; int n, Q; bool ok[N << 2];

namespace fast_io {
  int it, ed, ot, t; char stk[20], bf[N + 50], ob[N + 50];
  #define gc (it == ed && (ed = (it = 0) + fread(bf, 1, N, stdin), it == ed))? EOF : bf[it++]
  template <typename T> inline void read(T &x) {
    x = 0; char ch = gc; int f = 1;
    for (; !isdigit(ch); ch = gc) if (ch == '-') f = -1;
    for (; isdigit(ch); ch = gc) x = x * 10 + (ch ^ 48); x *= f; return ;
  } template <typename T, typename ...Args>
  inline void read(T &x, Args &...args) { read(x), read(args...); }
  inline void fls() { fwrite(ob, 1, ot, stdout), ot = 0; }
  template <typename T> inline void write(T x, char opt) {
    while (x > 9) stk[++t] = 48 ^ (x % 10), x /= 10;
    for (ob[ot++] = 48 ^ x; t; ob[ot++] = stk[t--]);
    ob[ot++] = opt; if (ot > N) fls(); return ;
  }
} using fast_io::read; using fast_io::write;

struct Matrix {
  int v[sz][sz]; Matrix() { memset(v, 0, sizeof(v)); }
  inline void cl() { memset(v, 0, sizeof(v)); }
  inline void dl() { for (int i = 0; i < sz; ++i) v[i][i] = 1; }
  inline Matrix operator *(const Matrix &X) const {
    Matrix res; for (int i = 0; i < sz; ++i)
      for (int k = 0; k < sz; ++k) for (int j = 0; j < sz; ++j)
        res.v[i][j] = (res.v[i][j] + v[i][k] * X.v[k][j] % Mod) % Mod;
    return res;
  }
} ml[7], tag[N << 2];

struct Node {
  int st[sz]; Node() { fill(st, st + sz, 0); }
  inline void init(int a, int b, int c) {
    st[0] = a, st[1] = b, st[2] = c, st[3] = 1;
  } inline Node operator *(const Matrix &X) {
    Node res; for (int j = 0; j < sz; ++j)
      for (int i = 0; i < sz; ++i)
        res.st[i] = (res.st[i] + st[j] * X.v[j][i] % Mod) % Mod;
    return res;
  } inline Node operator +(const Node &X) const {
    Node res; for (int i = 0; i < sz; ++i)
      res.st[i] = (st[i] + X.st[i]) % Mod;
    return res;
  }
} t[N << 2];

inline void init() {
  for (int i = 1; i <= 6; ++i) {
    for (int j = 0; j < sz; ++j) ml[i].v[j][j] = 1;
    if (i == 1) ml[i].v[1][0] = 1;
    else if (i == 2) ml[i].v[2][1] = 1;
    else if (i == 3) ml[i].v[0][2] = 1;
  }
  return ;
}

inline void pushdown(int rt, int l, int r) {
  if (!ok[rt]) return ; ok[ls] = ok[rs] = ok[rt];
  t[ls] = t[ls] * tag[rt], t[rs] = t[rs] * tag[rt];
  tag[ls] = tag[ls] * tag[rt], tag[rs] = tag[rs] * tag[rt];
  ok[rt] = false; tag[rt].cl(), tag[rt].dl(); return ;
}

inline void build(int rt, int l, int r) {
  tag[rt].dl();
  if (l == r) {
    int a, b, c; read(a, b, c);
    t[rt].init(a, b, c); return ;
  }
  int mid = (l + r) >> 1;
  build(ls, l, mid), build(rs, mid + 1, r);
  t[rt] = t[ls] + t[rs]; return ;
}

inline void upd(int rt, int l, int r, int L, int R, int typ) {
  if (L <= l && r <= R) {
    t[rt] = t[rt] * ml[typ]; ok[rt] = true;
    tag[rt] = tag[rt] * ml[typ]; return ;
  }
  pushdown(rt, l, r); int mid = (l + r) >> 1;
  if (L <= mid) upd(ls, l, mid, L, R, typ); if (R > mid) upd(rs, mid + 1, r, L, R, typ);
  t[rt] = t[ls] + t[rs]; return ;
}

inline Node query(int rt, int l, int r, int L, int R) {
  if (L <= l && r <= R) return t[rt];
  pushdown(rt, l, r); int mid = (l + r) >> 1;
  int tar = 0; Node lef, rig;
  if (L <= mid) ++tar, lef = query(ls, l, mid, L, R);
  if (R > mid) tar += 2, rig = query(rs, mid + 1, r, L, R);
  if (tar == 1) return lef; if (tar == 2) return rig;
  return lef + rig;
}

inline void solve() {
  int op, l, r, vl; read(op, l, r);
  if (op <= 3) return upd(1, 1, n, l, r, op), void();
  if (op <= 6) {
    read(vl); if (op == 4) ml[op].v[3][0] = vl;
    else if (op == 5) ml[op].v[1][1] = vl;
    else ml[op].v[2][2] = 0, ml[op].v[3][2] = vl;
    return upd(1, 1, n, l, r, op), void();
  }
  Node tmp = query(1, 1, n, l, r);
  for (int i = 0; i < sz - 1; ++i) write(tmp.st[i], ' ');
  fast_io::ob[fast_io::ot++] = '\n';
  if (fast_io::ot > N) fast_io::fls(); return ;
}

signed main() {
  init(); read(n); build(1, 1, n);
  read(Q); while (Q--) solve();
  fast_io::fls(); return 0;
}

标签:space0,space,int,Sol,times,魔法师,space1,P7453,bmatrix
From: https://www.cnblogs.com/MistZero/p/P7453-Sol.html

相关文章

  • solr
    目标:solr的概念solr服务器的搭建和使用solr中导入数据库数据项目中怎么使用solr实现商品搜索功能一.solr相关概念1.1什么是Solr?solr是一个独立的企业级搜索应用......
  • solr服务器
    目标:solr的概念solr服务器的搭建和使用solr中导入数据库数据项目中怎么使用solr实现商品搜索功能一.solr相关概念1.1什么是Solr?solr是一个独立的企业级搜索应用服务器,它......
  • Efforts on resolving compiling issues upon installing R packages on the Mac M1 c
    ResolvecompilingproblemsuponinstallingRpackagesonM1chiphttps://mac.r-project.org/openmp/#dohttps://github.com/Rdatatable/data.table/wiki/Installa......
  • 25Jmeter之服务器性能资源监测-Jconsole &Linux命令
    一.通过Jconsole进行监控服务器资源情况Jconsole是一个内置Java性能分析器,可以轻松地使用JConsole来监控Java应用程序性能和跟踪Java中的代码。(1)开始—运行—输......
  • NPM INSTALL 的时候提示“ERESOLVE could not resolve”
    我们通常使用npminstall来为vue项目导入组件库,但是有时候会莫名报错,如下:npmERR!codeERESOLVEnpmERR!ERESOLVEcouldnotresolvenpmERR!npmERR!Whilere......
  • Solr 8.11入门教程(4)中文分词
    中文分词默认对中文分词的效果并不好,我们添加IK分词。下载重新下载:先下载solr8版本对应的ik分词器,分词器GitHub源码地址:https://github.com/magese/ik-analyzer-solr添......
  • Solr 8.11入门教程(2)创建core
    新建core添加core命令添加使用命令比较简单~$bin/solrcreate-cmytest[core名称]这样就添加完了。CoreAdmin就可以看到了。手动添加手动添加相对复杂一些,需要提......
  • Solr 8.11入门教程(2)创建core
    新建core添加core命令添加使用命令比较简单~$bin/solrcreate-cmytest[core名称]这样就添加完了。CoreAdmin就可以看到了。手动添加手动添加相对复杂一些,需要提前创建目......
  • Solr 8.11入门教程(2)新建core
    Solr8.11入门教程(2)新建core添加core命令添加使用命令比较简单~$bin/solrcreate-cmytest[core名称]这样就添加完了。CoreAdmin就可以看到了。手动添加手动......
  • Cannot resolve Servlet 'DispatcherServlet'
    问题:    明明自己包导入(org.springframework.web.servlet.DispatcherServlet)的没有问题,按ctrl键可以直接进入DispatcherServlet,却一直有红色警告。后来通过清除缓存......