E - Rotate and Flip 题解(矩阵)

题目链接

题目大意

给你n个点你对这n个点进行m次操作

操作有四种

第一种使得所有点nx=y ny=-x

第二种使得所有点nx=-y ny=x

第三种使得所有点nx=2p-x ny=y

第四种使得所有点nx=x ny=2p-y

有q次查询

每次查询为b位置经过a次操作后的坐标

题目思路

这个就是每个位置都和以前的位置有关就要想到矩阵操作

把它转化为矩阵的操作即可

第一种操作

\[\begin{pmatrix} 0& 1&0 \\ -1 & 0 &0\\ 0&0&1 \end{pmatrix} \]

第二种操作

\[\begin{pmatrix} 0& -1&0 \\ 1 & 0 &0\\ 0&0&1 \end{pmatrix} \]

第三种操作

\[\begin{pmatrix} -1& 0&2p \\ 0 & 1 &0 \\ 0&0&1 \end{pmatrix} \]

第四种操作

\[\begin{pmatrix} 1& 0&0 \\ 0 & -1 &2p\\ 0&0&1 \end{pmatrix} \]

注意相乘为\(pre[i]=mul(basei,pre[i-1])\) 注意顺序

代码

#include<bits/stdc++.h>
#define fi first
#define se second
#define debug cout<<"I AM HERE"<<endl;
using namespace std;
typedef long long ll;
const int maxn=2e5+5,inf=0x3f3f3f3f,mod=998244353;
const int eps=1e-6;
int n,m,q;
int x[maxn],y[maxn];
struct matrix{
    ll a[4][4];
}base1,base2,base3,base4,pre[maxn];
void init(){
    pre[0].a[1][1]=pre[0].a[2][2]=pre[0].a[3][3]=1;
    base1.a[1][2]=base1.a[3][3]=1;
    base1.a[2][1]=-1;
    base2.a[1][2]=-1;
    base2.a[2][1]=base2.a[3][3]=1;
    base3.a[1][1]=-1;
    base3.a[2][2]=base3.a[3][3]=1; //a[1][3] =2p
    base4.a[1][1]=base4.a[3][3]=1;
    base4.a[2][2]=-1; //a[2][3]=2p
}
matrix mul(matrix a,matrix b){
    matrix ans;
    memset(ans.a,0,sizeof(ans.a));
    for(int i=1;i<=3;i++){
        for(int j=1;j<=3;j++){
            for(int k=1;k<=3;k++){
                ans.a[i][k]+=a.a[i][j]*b.a[j][k];
            }
        }
    }
    return ans;
}
signed main(){
    init();
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        scanf("%lld%lld",&x[i],&y[i]);
    }
    scanf("%d",&m);
    for(int i=1,opt,p;i<=m;i++){
        scanf("%d",&opt);
        if(opt==1){
            pre[i]=mul(base1,pre[i-1]);
        }else if(opt==2){
            pre[i]=mul(base2,pre[i-1]);
        }else if(opt==3){
            scanf("%d",&p);
            base3.a[1][3]=2*p;
            pre[i]=mul(base3,pre[i-1]);
        }else if(opt==4){
            scanf("%d",&p);
            base4.a[2][3]=2*p;
            pre[i]=mul(base4,pre[i-1]);
        }
    }
    scanf("%d",&q);
    for(int i=1,a,b;i<=q;i++){
        scanf("%d%d",&a,&b);
        ll prx=pre[a].a[1][1]*x[b]+pre[a].a[1][2]*y[b]+pre[a].a[1][3]*1;
        ll pry=pre[a].a[2][1]*x[b]+pre[a].a[2][2]*y[b]+pre[a].a[2][3]*1;
        printf("%lld %lld\n",prx,pry);
    }
    return 0;
}

posted @ 2021-01-28 00:29  hunxuewangzi  阅读(71)  评论(0编辑  收藏  举报