美登杯”上海市高校大学生程序设计邀请赛 Problem E 、 小 花梨 的数组 (线段树)
Problem E E 、 小 花梨 的数组
时间限制:1000ms 空间限制:512MB
Description
小花梨得到了一个长度为𝑜的数组𝑏,现在要对它进行三种操作:
⚫ 1 𝑠 ] ∗
⚫ 2 𝑚 𝑠 对所有的𝑗 ∈ [𝑚,𝑠],𝑏[𝑗] = 𝑏[𝑗] / 𝑛𝑗𝑜𝑞𝑠𝑗𝑛𝑓(𝑏[𝑗])
⚫ 3 𝑦 求𝑏[𝑦]的值
𝑛𝑗𝑜𝑞𝑠𝑗𝑛𝑓(𝑦) = {
1 (𝑦 = 1)
𝑦的最小素因子(𝑦 ≥ 2)
现在给出初始数组𝑏,对其进行𝑛次操作,对于第三种操作输出结果。
Input
第一两个正整数𝑜,𝑛,表示数组的长度以及操作次数(1 ≤ 𝑜,𝑛 ≤ 100000)
第二行输入𝑜个正整数表示数组𝑏(1 ≤ 𝑏 𝑗 ≤ 1000000)
接下来𝑛行表示𝑛次操作,每行输入格式为"1 𝑚 𝑠"或者"2 𝑚 𝑠",或者"3 𝑦",对应上述三种操作。
1 ≤ 𝑚,𝑠,𝑦 ≤ 𝑜,𝑚 ≤ 𝑠
Output
对于第三种操作输出答案即可,答案对10 9 + 7进行取模。
Example
Sample Input Sample Output
5 8
1 2 3 4 5
1 2 4
3 2
3 3
2 2 5
3 2
3 5
1 5 5
3 5
4
9
2
1
1
题意:
思路:
读入的时候用唯一分解定理将a[i]分解为每一个质因子放入一个堆中。
我们知道一个数乘以它的最小的质因子后,它的最小的质因子不会改变。
如果先乘后除以它的最小的质因子,那么就是除以抵消一次乘法。那么我们可以标记一个数当前需要操作1多少次laze1,而不去实际更新数值,那么如果需要操作2的时候,laze1>0,那么影响只是laze1--,否则就是 需要执行的操作2次数laze2++.
我们用线段树来处理区间操作的问题,
那么我们再来分析一下单点询问是,该如何得到答案,我们知道,如果一个点的laze2有数值,laze2也有数值,那么这些除法操作一定在乘法操作之前,至于为什么,看上面laze变化的讲解。所以我们可以先暴力除掉需要除掉的数,当数值等于1的时候停止。(因为没有意义,又因为唯一分解定理可以知道一个小于等于1e6的数最多被除以十几次而已),除以之后得到的剩余数值a[i] *它的最小质因子的laze1次幂就是当前的a[i]数值。
线段树的pushdown 同样根据上面的性质和优先级进行处理。
细节见代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <stack>
#include <map>
#include <set>
#include <vector>
#include <iomanip>
#define ALL(x) (x).begin(), (x).end()
#define sz(a) int(a.size())
#define all(a) a.begin(), a.end()
#define rep(i,x,n) for(int i=x;i<n;i++)
#define repd(i,x,n) for(int i=x;i<=n;i++)
#define pii pair<int,int>
#define pll pair<long long ,long long>
#define gbtb ios::sync_with_stdio(false),cin.tie(0),cout.tie(0)
#define MS0(X) memset((X), 0, sizeof((X)))
#define MSC0(X) memset((X), '\0', sizeof((X)))
#define pb push_back
#define mp make_pair
#define fi first
#define se second
#define eps 1e-6
#define gg(x) getInt(&x)
#define chu(x) cout<<"["<<#x<<" "<<(x)<<"]"<<endl
using namespace std;
typedef long long ll;
ll gcd(ll a, ll b) {return b ? gcd(b, a % b) : a;}
ll lcm(ll a, ll b) {return a / gcd(a, b) * b;}
ll powmod(ll a, ll b, ll MOD) {ll ans = 1; while (b) {if (b % 2) { ans = ans * a % MOD; } a = a * a % MOD; b /= 2;} return ans;}
inline void getInt(int *p);
const int maxn = 1000010;
const int inf = 0x3f3f3f3f;
/*** TEMPLATE CODE * * STARTS HERE ***/
// const int maxn = 1e7+50;
bool noprime[maxn + 50];
vector <int> p;
int getPrime()
{
// 华丽的初始化
memset(noprime, false, sizeof(noprime));
p.clear();
int m = (int)sqrt(maxn + 0.5);
// 优化的埃筛
for (int i = 2; i <= m; i++) {
if (!noprime[i]) {
for (int j = i * i; j <= maxn; j += i) {
noprime[j] = true;
}
}
}
// 把素数加到vector里
for (int i = 2; i <= maxn; i++) {
if (!noprime[i]) {
p.push_back(i);
}
}
//返回vector的大小
return p.size();
}
ll pf[105][2];// 0 -> value 1->count
const int N = 1e5 + 10;
priority_queue<int, vector<int>, greater<int> > q[N];
const ll mod = 1e9 + 7;
void getPrifac( ll n, int len, int id)
{
int pos = 0;
for (int i = 0; 1ll * p[i]*p[i] <= n && i < len; i++) {
if ( n % p[i] == 0) {
// 算质因数的幂数
while (n % p[i] == 0) {
q[id].push(p[i]);
n /= p[i];
}
// q[id].push(temp);
}
}
if ( n > 1) {
// pf[++pos][0] = n;
// pf[pos][1]=1;
q[id].push((n));
}
// return pos; // 优美的返回有多少个质因数
// 1~pos
}
int n;
ll a[N];
struct node {
int l, r;
int laze1;
int laze2;
} segment_tree[N << 2];
void build(int rt, int l, int r)
{
segment_tree[rt].l = l;
segment_tree[rt].r = r;
segment_tree[rt].laze2 = 0;
segment_tree[rt].laze1 = 0;
if (l == r) {
return ;
}
int mid = (l + r) >> 1;
build(rt << 1, l, mid);
build(rt << 1 | 1, mid + 1, r);
}
void pushdown(int rt)
{
int x = min(segment_tree[rt].laze2, segment_tree[rt << 1].laze1);
segment_tree[rt << 1].laze1 -= x;
segment_tree[rt << 1].laze2 += segment_tree[rt].laze2 - x;
segment_tree[rt << 1].laze1 += segment_tree[rt].laze1;
x = min(segment_tree[rt].laze2, segment_tree[rt << 1 | 1].laze1);
segment_tree[rt << 1 | 1].laze1 -= x;
segment_tree[rt << 1 | 1].laze2 += segment_tree[rt].laze2 - x;
segment_tree[rt << 1 | 1].laze1 += segment_tree[rt].laze1;
segment_tree[rt].laze1 = segment_tree[rt].laze2 = 0;
}
void update1(int rt, int l, int r)
{
if (segment_tree[rt].l >= l && segment_tree[rt].r <= r) {
segment_tree[rt].laze1++;
} else {
pushdown(rt);
int mid = (segment_tree[rt].l + segment_tree[rt].r) >> 1;
if (l <= mid) {
update1(rt << 1, l, r);
}
if (r > mid) {
update1(rt << 1 | 1, l, r);
}
}
}
void update2(int rt, int l, int r)
{
if (segment_tree[rt].l >= l && segment_tree[rt].r <= r) {
if (segment_tree[rt].laze1) {
segment_tree[rt].laze1--;
} else {
segment_tree[rt].laze2++;
}
} else {
pushdown(rt);
int mid = (segment_tree[rt].l + segment_tree[rt].r) >> 1;
if (l <= mid) {
update2(rt << 1, l, r);
}
if (r > mid) {
update2(rt << 1 | 1, l, r);
}
}
}
ll ask(int rt, int id)
{
if (segment_tree[rt].l == segment_tree[rt].r && segment_tree[rt].r == id) {
while (segment_tree[rt].laze2 && a[id] > 1 && q[id].size()) {
segment_tree[rt].laze2--;
a[id] /= q[id].top();
q[id].pop();
}
ll x = 1ll;
if (q[id].size()) {
x = q[id].top();
}
return (powmod(x, segment_tree[rt].laze1, mod) * a[id]) % mod;
} else {
pushdown(rt);
int mid = (segment_tree[rt].r + segment_tree[rt].l) >> 1;
if (id <= mid) {
return ask(rt << 1, id);
} else {
return ask(rt << 1 | 1, id);
}
}
}
int main()
{
//freopen("D:\\code\\text\\input.txt","r",stdin);
//freopen("D:\\code\\text\\output.txt","w",stdout);
int m;
int len = getPrime();
scanf("%d %d", &n, &m);
repd(i, 1, n) {
scanf("%lld", &a[i]);
getPrifac(a[i], len, i);
}
build(1, 1, n);
int op;
while (m--) {
scanf("%d", &op);
if (op == 1) {
int l, r;
scanf("%d %d", &l, &r);
update1(1, l, r);
} else if (op == 2) {
int l, r;
scanf("%d %d", &l, &r);
update2(1, l, r);
} else {
int x;
scanf("%d", &x);
printf("%lld\n", ask(1, x) );
}
}
return 0;
}
inline void getInt(int *p)
{
char ch;
do {
ch = getchar();
} while (ch == ' ' || ch == '\n');
if (ch == '-') {
*p = -(getchar() - '0');
while ((ch = getchar()) >= '0' && ch <= '9') {
*p = *p * 10 - ch + '0';
}
} else {
*p = ch - '0';
while ((ch = getchar()) >= '0' && ch <= '9') {
*p = *p * 10 + ch - '0';
}
}
}
本博客为本人原创,如需转载,请必须声明博客的源地址。
本人博客地址为:www.cnblogs.com/qieqiemin/
希望所写的文章对您有帮助。