对于平面直角坐标系上的坐标 ( x , y ) (x,y) (x,y),定义如下两种操作:
本题要求将平面坐标 ( x , y ) (x, y) (x,y),经过 n n n个操作 ( t 1 , t 2 , ⋯ , t n ) (t_1, t_2, \cdots, t_n) (t1,t2,⋯,tn)后,对于给定的操作序列,计算 m m m个如下查询:
在考场上,笔者发现此题为区间查询问题,因而首先想到使用树状数组。但是树状数组的建树的时间复杂度虽然为 O ( n ) O(n) O(n),但是每次查询的时间复杂度为 O ( l o g n ) O(log\ n) O(log n)。而此题不涉及到对区间值的修改,因而无需使用树状数组,只需要记录 k k k的前缀和和 θ \theta θ的前缀积即可。前缀和向量和前缀积向量的建立的时间复杂度为 O ( n ) O(n) O(n),每次区间查询的时间复杂度为 O ( 1 ) O(1) O(1)。
具体而言,由于拉伸和旋转2种行为相互独立,我们只需分别求出经过 n n n个操作 ( t 1 , t 2 , ⋯ , t n ) (t_1, t_2, \cdots, t_n) (t1,t2,⋯,tn)后,总共旋转的角度和拉伸的倍数。我们仅需维护2个向量:
被查询坐标 ( x , y ) (x, y) (x,y)经过 t i , ⋯ , t j t_i, \cdots, t_j ti,⋯,tj( 1 ≤ i ≤ j ≤ n 1 \le i \le j \le n 1≤i≤j≤n)后,拉伸倍数为 k j / k i − 1 k_j / k_{i-1} kj/ki−1,角度为 θ j − θ i − 1 \theta_j-\theta_{i-1} θj−θi−1。
#include
using namespace std;
int n, m;
vector<double> xita(100005);
vector<double> k(100005, 1);
int main() {
cin >> n >> m;
for (int i = 1; i <= n; ++i) {
int type;
double value;
cin >> type >> value;
if (type == 1) {
k[i] = k[i - 1] * value;
xita[i] = xita[i - 1];
} else {
k[i] = k[i - 1];
xita[i] = xita[i - 1] + value;
}
}
for (int i = 0; i < m; ++i) {
int l, r;
double x, y;
cin >> l >> r >> x >> y;
double sum_xita = xita[r] - xita[l - 1];
double pro_k = k[r] / k[l - 1];
cout << fixed << setprecision(3) << (x * cos(sum_xita) - y * sin(sum_xita)) * pro_k << " "
<< (x * sin(sum_xita) + y * cos(sum_xita)) * pro_k << endl;
}
return 0;
}