【CF446C】DZY Loves Fibonacci Numbers (线段树 + 斐波那契数列)
Description
 看题戳我 给你一个序列,要求支持区间加斐波那契数列和区间求和。\(~n \leq 3 \times 10 ^ 5, ~fib_1 = fib_2 = 1~\).
Solution
 先来考虑一段斐波那契数列如何快速求和,根据性质有
fib_n &= fib_{n - 1} + fib_{n - 2} \\
&= fib_ {n - 2} + fib_{n - 3} + fib_{n - 2} \\
&= fib_{n - 3} + fib_{n - 4} + fib_{n - 3} + fib_{n - 2} \\
&= \dots \\
&= fib_2 + \sum_{i = 1}^{n - 2} {fib_i}
\end {align}
\]
 可以发现这里有个\(~\sum_{i = 1} ^ {n - 2} {fib_i}\),转换一下就是\(~\sum_{i = 1} ^ {n}fib_i = fib_{n + 2} - fib_2\).而两个斐波那契数列对应项加起来之后还是一个类斐波那契数列,记为\(~S_i\),设这个类斐波那契数列的起始项\(S_1 = a, S_2 = b\),显然有\(~S_i = a \times fib_{i - 2} + b \times fib_{i - 1}\).那么对于一段类斐波那契数列的求和,我们只要记起始的两项和这段数列的长度即可。现在可以用简单的线段树区间加来维护了,\(~PushDown~\)操作有一点细节,注意要分开算区间的前两项。具体看代码。。
Code
#include<bits/stdc++.h>
#define For(i, j, k) for(int i = j; i <= k; ++i)
#define Forr(i, j, k) for(int i = j; i >= k; --i)
using namespace std;
inline int read() {
	int x = 0, p = 1; char c = getchar();
	for(; !isdigit(c); c = getchar()) if(c == '-') p = -1;
	for(; isdigit(c); c = getchar()) x = (x << 1) + (x << 3) + (c ^ 48);
	return x *= p;
}
inline void File() {
	freopen("cf446c.in", "r", stdin);
	freopen("cf446c.out", "w", stdout);
}
const int N = 3e5 + 10, mod = 1e9 + 9;
int n, m, fib[N];
inline int add(int a, int b) { return (a += b) >= mod ? a - mod : a; }
namespace SGT {
#define lc (rt << 1)
#define rc (rt << 1 | 1)
#define mid (l + r >> 1)
#define lson lc, l, mid
#define rson rc, mid + 1, r
	int tr[N << 2], t1[N << 2], t2[N << 2];
	inline void pushup(int rt) { tr[rt] = (tr[lc] + tr[rc]) % mod; }
	inline int S(int a, int b, int x) {
		return x == 1 ? a : (x == 2 ? b : (1ll * a * fib[x - 2] + 1ll * b * fib[x - 1]) % mod);
	} 
	inline int sum(int a, int b, int x) {
		return x == 1 ? a : (x == 2 ? add(a, b) : (S(a, b, x + 2) - b + mod) % mod);
	}
	inline void pushdown(int rt, int l, int r) {
		if (t1[rt]) {
			t1[lc] = add(t1[lc], t1[rt]), t2[lc] = add(t2[lc], t2[rt]);
			tr[lc] = add(tr[lc], sum(t1[rt], t2[rt], mid - l + 1));
			int T1 = S(t1[rt], t2[rt], mid - l + 2), T2 = S(t1[rt], t2[rt], mid - l + 3);
			t1[rc] = add(t1[rc], T1), t2[rc] = add(t2[rc], T2);
			tr[rc] = add(tr[rc], sum(T1, T2, r - mid));
			t1[rt] = t2[rt] = 0;
		}
	}
	inline void build(int rt, int l, int r) {
		if (l == r) tr[rt] = read();
		else build(lson), build(rson), pushup(rt);
	}
	inline void update(int rt, int l, int r, int L, int R) {
		if (L <= l && r <= R) {
			tr[rt] = add(tr[rt], sum(fib[l - L + 1], fib[l - L + 2], r - l + 1));
			t1[rt] = add(t1[rt], fib[l - L + 1]); t2[rt] = add(t2[rt], fib[l - L + 2]);
			return ;
		}
		pushdown(rt, l, r);
		if (L <= mid) update(lson, L, R);
		if (R > mid) update(rson, L, R);
		pushup(rt);
	}
	inline int query(int rt, int l, int r, int L, int R) {
		if (L <= l && r <= R) return tr[rt];
		pushdown(rt, l, r); int res = 0;
		if (L <= mid) res = add(res, query(lson, L, R));
		if (R > mid) res = add(res, query(rson, L, R));
		return pushup(rt), res;
	}
#undef lc
#undef rc
#undef mid
#undef lson
#undef rson
}
int main() {
	File();
	n = read(), m = read();
	fib[1] = fib[2] = 1;
	For(i, 3, n + 5) fib[i] = (fib[i - 1] + fib[i - 2]) % mod;
	using namespace SGT;
	build(1, 1, n);
	while (m --) {
		int opt = read(), l = read(), r = read();
		opt == 1 ? update(1, 1, n, l, r), 1 : printf("%d\n", query(1, 1, n, l, r)), 1;
	}
	return 0;
}
【CF446C】DZY Loves Fibonacci Numbers (线段树 + 斐波那契数列)的更多相关文章
- CF446C DZY Loves Fibonacci Numbers 线段树 + 数学
		有两个性质需要知道: $1.$ 对于任意的 $f[i]=f[i-1]+f[i-2]$ 的数列,都有 $f[i]=fib[i-2]\times f[1]+fib[i-1]\times f[2]$ 其中 ... 
- [Codeforces 316E3]Summer Homework(线段树+斐波那契数列)
		[Codeforces 316E3]Summer Homework(线段树+斐波那契数列) 顺便安利一下这个博客,给了我很大启发(https://gaisaiyuno.github.io/) 题面 有 ... 
- Codeforces 446-C  DZY Loves Fibonacci Numbers 同余 线段树 斐波那契数列
		C. DZY Loves Fibonacci Numbers time limit per test 4 seconds memory limit per test 256 megabytes inp ... 
- ACM学习历程—Codeforces 446C DZY Loves Fibonacci Numbers(线段树 && 数论)
		Description In mathematical terms, the sequence Fn of Fibonacci numbers is defined by the recurrence ... 
- codeforces 446C DZY Loves Fibonacci Numbers 线段树
		假如F[1] = a, F[2] = B, F[n] = F[n - 1] + F[n - 2]. 写成矩阵表示形式可以很快发现F[n] = f[n - 1] * b + f[n - 2] * a. ... 
- Codeforces446C DZY Loves Fibonacci Numbers(线段树 or 分块?)
		第一次看到段更斐波那契数列的,整个人都不会好了.事后看了题解才明白了一些. 首先利用二次剩余的知识,以及一些数列递推式子有下面的 至于怎么解出x^2==5(mod 10^9+9),我就不知道了,但是要 ... 
- Codeforces 446C DZY Loves Fibonacci Numbers [线段树,数论]
		洛谷 Codeforces 思路 这题知道结论就是水题,不知道就是神仙题-- 斐波那契数有这样一个性质:\(f_{n+m}=f_{n+1}f_m+f_{n}f_{m-1}\). 至于怎么证明嘛-- 即 ... 
- 【思维题  线段树】cf446C. DZY Loves Fibonacci Numbers
		我这种maintain写法好zz.考试时获得了40pts的RE好成绩 In mathematical terms, the sequence Fn of Fibonacci numbers is de ... 
- cf446C DZY Loves Fibonacci Numbers
		C. DZY Loves Fibonacci Numbers time limit per test 4 seconds memory limit per test 256 megabytes inp ... 
随机推荐
- c++入门之类——进一步剖析
			通常的,关于一个类,包含了下面几个方面: 1 声明类成员和接口:2 定义类接口函数(方法)3通过接口调用类 下面先给出第一条:声明类成员和接口 # ifndef MYTIME0_H_ # defin ... 
- 2018软工实践K班总结
			再回首一学期的软工实践,首先还是要感谢两位助教童鞋帮我承担了作业发布.打分以及与学生的问题沟通等.从这次的软工实践80人+开始,之后的实践课变为必修,故如何能更有效地组织大班实践环节是一个需要持续探讨 ... 
- JSP页面的基本元素
			JSP页面元素构成:静态内容.指令.表达式.小脚本.声明.注释. JSP指令包括: page指令:通常位于jsp页面的顶端,同一个页面可以有多个page指令. include指令:将一个外部文件嵌入到 ... 
- MySQL中有关NULL的计算
			mysql> select NULL=NULL; #判断两个NULL是否相等,结果不是1也不是0 +-----------+ | NULL=NULL | +-----------+ | NULL ... 
- Python中Celery 的基本用法以及Django 结合 Celery 的使用和实时监控进程
			celery是什么 1 celery是一个简单,灵活且可靠的,处理大量消息的分布式系统 2 专注于实时处理的异步任务队列 3 同时也支持任务调度 执行流程 Celery 基本使用 tasks.py i ... 
- Prometheus & SoundCloud
			Prometheus 系统监控方案 一 - Vovolie - 博客园https://www.cnblogs.com/vovlie/p/Prometheus_CONCEPTS.html Prometh ... 
- python生成个性二维码学习笔记
			在linux环境下进行编码 1.先进家目录,自行创建Code文件夹 cd Code 2.下载MyQR库 sudo pip3 install MyQR 3.下载所需资源文件并解压 Code/ $ wge ... 
- IDEA将项目上传至码云/GitHub托管
			怎么将本地的项目放到码云或者GitHub去托管了?(以码云为例) 一.创建远程项目 第一步:点击创建项目 第二步:填写项目相关信息 第三步:复制远程的项目地址,注意:此处码云官方已经给出上传项目方法, ... 
- POI解析Excel代码
			// 批量区域数据导入 @Action(value = "area_batchImport") public String batchImport() throws IOExcep ... 
- java随笔1  Ctrl+1补全
			Ctrl+1补全变量时,如果补全后的不是自己想要的, 比如:补全后这样 修改后 这时要对更改变量进行Ctrl+1补全路径 并且后者要进行Ctrl+1强转 
