Welcome to Subscribe On Youtube
1622. Fancy Sequence
Description
Write an API that generates fancy sequences using the append
, addAll
, and multAll
operations.
Implement the Fancy
class:
Fancy()
Initializes the object with an empty sequence.void append(val)
Appends an integerval
to the end of the sequence.void addAll(inc)
Increments all existing values in the sequence by an integerinc
.void multAll(m)
Multiplies all existing values in the sequence by an integerm
.int getIndex(idx)
Gets the current value at indexidx
(0-indexed) of the sequence modulo109 + 7
. If the index is greater or equal than the length of the sequence, return-1
.
Example 1:
Input ["Fancy", "append", "addAll", "append", "multAll", "getIndex", "addAll", "append", "multAll", "getIndex", "getIndex", "getIndex"] [[], [2], [3], [7], [2], [0], [3], [10], [2], [0], [1], [2]] Output [null, null, null, null, null, 10, null, null, null, 26, 34, 20] Explanation Fancy fancy = new Fancy(); fancy.append(2); // fancy sequence: [2] fancy.addAll(3); // fancy sequence: [2+3] -> [5] fancy.append(7); // fancy sequence: [5, 7] fancy.multAll(2); // fancy sequence: [5*2, 7*2] -> [10, 14] fancy.getIndex(0); // return 10 fancy.addAll(3); // fancy sequence: [10+3, 14+3] -> [13, 17] fancy.append(10); // fancy sequence: [13, 17, 10] fancy.multAll(2); // fancy sequence: [13*2, 17*2, 10*2] -> [26, 34, 20] fancy.getIndex(0); // return 26 fancy.getIndex(1); // return 34 fancy.getIndex(2); // return 20
Constraints:
1 <= val, inc, m <= 100
0 <= idx <= 105
- At most
105
calls total will be made toappend
,addAll
,multAll
, andgetIndex
.
Solutions
-
class Node { Node left; Node right; int l; int r; int mid; long v; long add; long mul = 1; public Node(int l, int r) { this.l = l; this.r = r; this.mid = (l + r) >> 1; } } class SegmentTree { private Node root = new Node(1, (int) 1e5 + 1); private static final int MOD = (int) 1e9 + 7; public SegmentTree() { } public void modifyAdd(int l, int r, int inc) { modifyAdd(l, r, inc, root); } public void modifyAdd(int l, int r, int inc, Node node) { if (l > r) { return; } if (node.l >= l && node.r <= r) { node.v = (node.v + (node.r - node.l + 1) * inc) % MOD; node.add = (node.add + inc) % MOD; return; } pushdown(node); if (l <= node.mid) { modifyAdd(l, r, inc, node.left); } if (r > node.mid) { modifyAdd(l, r, inc, node.right); } pushup(node); } public void modifyMul(int l, int r, int m) { modifyMul(l, r, m, root); } public void modifyMul(int l, int r, int m, Node node) { if (l > r) { return; } if (node.l >= l && node.r <= r) { node.v = (node.v * m) % MOD; node.add = (node.add * m) % MOD; node.mul = (node.mul * m) % MOD; return; } pushdown(node); if (l <= node.mid) { modifyMul(l, r, m, node.left); } if (r > node.mid) { modifyMul(l, r, m, node.right); } pushup(node); } public int query(int l, int r) { return query(l, r, root); } public int query(int l, int r, Node node) { if (l > r) { return 0; } if (node.l >= l && node.r <= r) { return (int) node.v; } pushdown(node); int v = 0; if (l <= node.mid) { v = (v + query(l, r, node.left)) % MOD; } if (r > node.mid) { v = (v + query(l, r, node.right)) % MOD; } return v; } public void pushup(Node node) { node.v = (node.left.v + node.right.v) % MOD; } public void pushdown(Node node) { if (node.left == null) { node.left = new Node(node.l, node.mid); } if (node.right == null) { node.right = new Node(node.mid + 1, node.r); } if (node.add != 0 || node.mul != 1) { Node left = node.left, right = node.right; left.v = (left.v * node.mul + (left.r - left.l + 1) * node.add) % MOD; right.v = (right.v * node.mul + (right.r - right.l + 1) * node.add) % MOD; left.add = (left.add * node.mul + node.add) % MOD; right.add = (right.add * node.mul + node.add) % MOD; left.mul = (left.mul * node.mul) % MOD; right.mul = (right.mul * node.mul) % MOD; node.add = 0; node.mul = 1; } } } class Fancy { private int n; private SegmentTree tree = new SegmentTree(); public Fancy() { } public void append(int val) { ++n; tree.modifyAdd(n, n, val); } public void addAll(int inc) { tree.modifyAdd(1, n, inc); } public void multAll(int m) { tree.modifyMul(1, n, m); } public int getIndex(int idx) { return idx >= n ? -1 : tree.query(idx + 1, idx + 1); } } /** * Your Fancy object will be instantiated and called as such: * Fancy obj = new Fancy(); * obj.append(val); * obj.addAll(inc); * obj.multAll(m); * int param_4 = obj.getIndex(idx); */
-
const int MOD = 1e9 + 7; class Node { public: Node* left; Node* right; int l; int r; int mid; long long v; long long add; long long mul; Node(int l, int r) { this->l = l; this->r = r; this->mid = (l + r) >> 1; this->left = this->right = nullptr; v = add = 0; mul = 1; } }; class SegmentTree { private: Node* root; public: SegmentTree() { root = new Node(1, 1e5 + 1); } void modifyAdd(int l, int r, int inc) { modifyAdd(l, r, inc, root); } void modifyAdd(int l, int r, int inc, Node* node) { if (l > r) return; if (node->l >= l && node->r <= r) { node->v = (node->v + (node->r - node->l + 1) * inc) % MOD; node->add = (node->add + inc) % MOD; return; } pushdown(node); if (l <= node->mid) modifyAdd(l, r, inc, node->left); if (r > node->mid) modifyAdd(l, r, inc, node->right); pushup(node); } void modifyMul(int l, int r, int m) { modifyMul(l, r, m, root); } void modifyMul(int l, int r, int m, Node* node) { if (l > r) return; if (node->l >= l && node->r <= r) { node->v = (node->v * m) % MOD; node->add = (node->add * m) % MOD; node->mul = (node->mul * m) % MOD; return; } pushdown(node); if (l <= node->mid) modifyMul(l, r, m, node->left); if (r > node->mid) modifyMul(l, r, m, node->right); pushup(node); } int query(int l, int r) { return query(l, r, root); } int query(int l, int r, Node* node) { if (l > r) return 0; if (node->l >= l && node->r <= r) return node->v; pushdown(node); int v = 0; if (l <= node->mid) v = (v + query(l, r, node->left)) % MOD; if (r > node->mid) v = (v + query(l, r, node->right)) % MOD; return v; } void pushup(Node* node) { node->v = (node->left->v + node->right->v) % MOD; } void pushdown(Node* node) { if (!node->left) node->left = new Node(node->l, node->mid); if (!node->right) node->right = new Node(node->mid + 1, node->r); if (node->add || node->mul != 1) { long add = node->add, mul = node->mul; Node* left = node->left; Node* right = node->right; left->v = (left->v * mul + (left->r - left->l + 1) * add) % MOD; right->v = (right->v * mul + (right->r - right->l + 1) * add) % MOD; left->add = (left->add * mul + add) % MOD; right->add = (right->add * mul + add) % MOD; left->mul = (left->mul * mul) % MOD; right->mul = (right->mul * mul) % MOD; node->add = 0; node->mul = 1; } } }; class Fancy { public: int n; SegmentTree* tree; Fancy() { n = 0; tree = new SegmentTree(); } void append(int val) { ++n; tree->modifyAdd(n, n, val); } void addAll(int inc) { tree->modifyAdd(1, n, inc); } void multAll(int m) { tree->modifyMul(1, n, m); } int getIndex(int idx) { return idx >= n ? -1 : tree->query(idx + 1, idx + 1); } }; /** * Your Fancy object will be instantiated and called as such: * Fancy* obj = new Fancy(); * obj->append(val); * obj->addAll(inc); * obj->multAll(m); * int param_4 = obj->getIndex(idx); */
-
MOD = int(1e9 + 7) class Node: def __init__(self, l, r): self.left = None self.right = None self.l = l self.r = r self.mid = (l + r) >> 1 self.v = 0 self.add = 0 self.mul = 1 class SegmentTree: def __init__(self): self.root = Node(1, int(1e5 + 1)) def modifyAdd(self, l, r, inc, node=None): if l > r: return if node is None: node = self.root if node.l >= l and node.r <= r: node.v = (node.v + (node.r - node.l + 1) * inc) % MOD node.add += inc return self.pushdown(node) if l <= node.mid: self.modifyAdd(l, r, inc, node.left) if r > node.mid: self.modifyAdd(l, r, inc, node.right) self.pushup(node) def modifyMul(self, l, r, m, node=None): if l > r: return if node is None: node = self.root if node.l >= l and node.r <= r: node.v = (node.v * m) % MOD node.add = (node.add * m) % MOD node.mul = (node.mul * m) % MOD return self.pushdown(node) if l <= node.mid: self.modifyMul(l, r, m, node.left) if r > node.mid: self.modifyMul(l, r, m, node.right) self.pushup(node) def query(self, l, r, node=None): if l > r: return 0 if node is None: node = self.root if node.l >= l and node.r <= r: return node.v self.pushdown(node) v = 0 if l <= node.mid: v = (v + self.query(l, r, node.left)) % MOD if r > node.mid: v = (v + self.query(l, r, node.right)) % MOD return v def pushup(self, node): node.v = (node.left.v + node.right.v) % MOD def pushdown(self, node): if node.left is None: node.left = Node(node.l, node.mid) if node.right is None: node.right = Node(node.mid + 1, node.r) left, right = node.left, node.right if node.add != 0 or node.mul != 1: left.v = (left.v * node.mul + (left.r - left.l + 1) * node.add) % MOD right.v = (right.v * node.mul + (right.r - right.l + 1) * node.add) % MOD left.add = (left.add * node.mul + node.add) % MOD right.add = (right.add * node.mul + node.add) % MOD left.mul = (left.mul * node.mul) % MOD right.mul = (right.mul * node.mul) % MOD node.add = 0 node.mul = 1 class Fancy: def __init__(self): self.n = 0 self.tree = SegmentTree() def append(self, val: int) -> None: self.n += 1 self.tree.modifyAdd(self.n, self.n, val) def addAll(self, inc: int) -> None: self.tree.modifyAdd(1, self.n, inc) def multAll(self, m: int) -> None: self.tree.modifyMul(1, self.n, m) def getIndex(self, idx: int) -> int: return -1 if idx >= self.n else self.tree.query(idx + 1, idx + 1) # Your Fancy object will be instantiated and called as such: # obj = Fancy() # obj.append(val) # obj.addAll(inc) # obj.multAll(m) # param_4 = obj.getIndex(idx)