NYOJ 298-点的变换(经典矩阵解决点平移、缩放、翻转和旋转)

题目地址:NYOJ 298

思路:该题如果用对每个点模拟的操作,时间复杂度为O(n+m),结果肯定超时。然而利用矩阵乘法可以在O(m)的时间内把所有的操作合并为一个矩阵,然后每个点与该矩阵相乘可以得出最终的位置。

PS:十个利用矩阵乘法解决的经典题目 超级详细。

#include <stdio.h>
#include <math.h>
#include <string.h>
#include <stdlib.h>
#include <iostream>
#include <sstream>
#include <algorithm>
#include <set>
#include <queue>
#include <stack>
#include <map>
using namespace std;
typedef long long LL;
const int inf=0x3f3f3f3f;
const double pi= acos(-1.0);
const double esp=1e-6;
const int maxn=1e5+10;
struct node {
    double mp[5][5];
} q[maxn],res,init,ans;
struct node Mul(struct node x,struct node y) {
    struct node tmp;
    int i,j,k;
    for(i=0; i<3; i++) {
        for(j=0; j<3; j++) {
            tmp.mp[i][j]=0;
            for(k=0; k<3; k++)
                tmp.mp[i][j]=tmp.mp[i][j]+x.mp[i][k]*y.mp[k][j];
        }
    }
    return tmp;
}
int main() {
    int n,m,i,j;
    double a,b,bs,jd,hd;
    while(~scanf("%d %d",&n,&m)) {
        for(i=0; i<n; i++) {
            scanf("%lf %lf",&q[i].mp[0][0],&q[i].mp[1][0]);
            q[i].mp[2][0]=1;
        }
        for(i=0; i<3; i++) {
            ans.mp[i][i]=1;
        }
        char str[10];
        while(m--) {
            memset(str,0,sizeof(str));
            memset(res.mp,0,sizeof(res.mp));
            scanf("%s",str);
            for(i=0; i<3; i++)
                res.mp[i][i]=1;
            if(str[0]=='M') {
                scanf("%lf %lf",&a,&b);
                res.mp[0][2]=a;
                res.mp[1][2]=b;
            } else if(str[0]=='X') {
                res.mp[1][1]=-1;
            } else if(str[0]=='Y') {
                res.mp[0][0]=-1;
            } else if(str[0]=='S') {
                scanf("%lf",&bs);
                res.mp[0][0]=bs;
                res.mp[1][1]=bs;
            } else if(str[0]=='R') {
                scanf("%lf",&jd);
                hd=jd/180*pi;
                res.mp[0][0]=cos(hd);
                res.mp[0][1]=-sin(hd);
                res.mp[1][0]=sin(hd);
                res.mp[1][1]=cos(hd);
            }
            ans=Mul(res,ans);
        }
        for(i=0; i<n; i++) {
            init=Mul(ans,q[i]);
            printf("%.1lf %.1lf\n",init.mp[0][0],init.mp[1][0]);
        }

    }
    return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-07 23:29:53

NYOJ 298-点的变换(经典矩阵解决点平移、缩放、翻转和旋转)的相关文章

NYOJ 298 点的变换(矩阵快速幂)

点的变换 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 平面上有不超过10000个点,坐标都是已知的,现在可能对所有的点做以下几种操作: 平移一定距离(M),相对X轴上下翻转(X),相对Y轴左右翻转(Y),坐标缩小或放大一定的倍数(S),所有点对坐标原点逆时针旋转一定角度(R). 操作的次数不超过1000000次,求最终所有点的坐标. 提示:如果程序中用到PI的值,可以用acos(-1.0)获得. 输入 只有一组测试数据 测试数据的第一行是两个整数N,M,分别表示

NYOJ 298 点的变换

题目链接:298 点的变换 这题放在矩阵快速幂里,我一开始想不透它是怎么和矩阵搭上边的,然后写了个暴力的果然超时,上网看了题解后,发现竟然能够构造一些精巧的矩阵来处理,不得不说实在太强大了! http://blog.csdn.net/lyhvoyage/article/details/39755595 然后我的代码是: 1 #include<cstdio> 2 #include<cstring> 3 #include<cmath> 4 #include<cstdl

解决三维建模软件中摄像机旋转的问题

我们看到很多3D建模软件都可以很方便地对摄像机进行旋转,比如说Blender,通过按下鼠标中键,就可以很方便地对摄像机进行旋转.在学习了几年OpenGL之后,我也想要模仿这一点做出类似的功能.但是当研究的时候,遇到了一些困难. 蒋彩阳原创文章,首发地址:http://blog.csdn.net/gamesdev/article/details/44828475.欢迎同行前来探讨. 此前了解到,这些建模软件对于三维建模中的摄像机的定义是:拥有位置(position).观察点(look at)以及向

NYOJ298 点的变换 【矩阵乘法经典】

任意门:http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=298 点的变换 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 平面上有不超过10000个点,坐标都是已知的,现在可能对所有的点做以下几种操作: 平移一定距离(M),相对X轴上下翻转(X),相对Y轴左右翻转(Y),坐标缩小或放大一定的倍数(S),所有点对坐标原点逆时针旋转一定角度(R). 操作的次数不超过1000000次,求最终所有点的坐标. 提示:如果

NYOJ 298

利用矩阵来做变换,参考Max大神的思想的,虽然不是同一道题. ----------- 给定n个点,m个操作,构造O(m+n)的算法输出m个操作后各点的位置.操作有平移.缩放.翻转和旋转    这里的操作是对所有点同时进行的.其中翻转是以坐标轴为对称轴进行翻转(两种情况),旋转则以原点为中心.如果对每个点分别进行模拟,那么m个操作总共耗时O(mn).利用矩阵乘法可以在O(m)的时间里把所有操作合并为一个矩阵,然后每个点与该矩阵相乘即可直接得出最终该点的位置,总共耗时O(m+n).假设初始时某个点的

经典矩阵问题

//经典矩阵问题是利用数字生成一个矩阵,而该矩阵刚好是一个正方形,该矩阵内的数字是有 //规律的排序而形成矩阵.比较常见有以下形式 //1. //  1 2 9 10 //  4 3 8 11 //  5 6 7 12 //  16 15 14 13 //2. //  1 2 6 7 //  3 5 8 13 //  4 9 12 14 //  10 11 15 16 //3. //  1 2 3 4 //  12 13 14 5 //  11 16 15 6 //  10 9 8 7 //4.

Python入门经典. 以解决计算问题为导向的Python编程实践(高清版)PDF

Python入门经典. 以解决计算问题为导向的Python编程实践(高清版)PDF百度网盘链接:https://pan.baidu.com/s/1juLsew8UiOErRheQPOuTaw 提取码:fssd 复制这段内容后打开百度网盘手机App,操作更方便哦内容简介 · · · · · · <Python入门经典:以解决计算问题为导向的Python编程实践>是一本系统而科学的Python入门教程,美国密歇根州立大学等多所美国知名高校采用其作为编程语言的入门教材,被奉为经典.它不仅从计算机教学

WPF/Silverlight中图形的平移,缩放,旋转,倾斜变换演示

原文:WPF/Silverlight中图形的平移,缩放,旋转,倾斜变换演示 为方便描述, 这里仅以正方形来做演示, 其他图形从略. 运行时效果图: XAML代码:// Transform.XAML <Canvas Width="700" Height="700" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"  xmlns:x="http://sc

opencv2 矩阵方式 resize图像缩放代码(转载)

http://blog.sina.com.cn/s/blog_74a459380101r0yx.html opencv2 矩阵方式 resize图像缩放代码(转载) (2014-05-16 09:55:35) 转载▼   分类: Opencv_Function 最近学习opencv的时候遇到的一些技术问题,拿出来分享一下.opencv1和opencv2最大的区别就是c++支持,这使得网上有些资料是opencv1的c语言写的,而有些人喜欢c++,当然接口函数也就不同了.下面是一个c++的openc