Eigen库实现简单的旋转、平移操作
本来课程要求用GUI界面来实现Eigen的旋转、平移操作的,但是接触GUI编程时间太短,虽然要求很简单,但是做了几天还是没有完成。就把命令行下面的简单的贴一下吧。
main.cpp
#include <iostream>
#include <string>
#include <map>
#include <algorithm>
#include "func.h"
#include "Point.h"
#include "Shape.h"
enum Motion {
Rotate = 1,
Move,
Zoom,
};
std::map<std::string, Motion> motion = {
{"rotate", Motion::Rotate},
{"move", Motion::Move},
{"zoom", Motion::Zoom},
};
Motion resolveCommand(std::string command) {
return motion[command];
}
int main() {
std::string name, pointNum;
printf("请输入图形的名字:");
getline(std::cin, name);
if (name.empty()) {
name = "T";
printf("你的输入为空,因此为你的图形命名(T)\n");
}
printf("请输入图形的端点数量:");
getline(std::cin, pointNum);
if (is_number(pointNum)) {
std::cout << "你输入的是整数\n";
} else {
std::cout << "你输入的不是整数\n";
}
int num = atoi(pointNum.c_str());
Point *shapePoints = new Point[num];
for (int i = 0; i < num; ++i) {
double x, y;
printf("请输入第%d个点横坐标x = ", i + 1);
scanf("%lf", &x);
printf("请输入第%d个点纵坐标y = ", i + 1);
scanf("%lf", &y);
shapePoints[i].update(x, y);
}
std::cout << "你输入图形的名字为:" << name << " " << "你输入的端点数量为:" << pointNum << std::endl;
Shape *shape = new Shape(shapePoints, num, name.c_str());
shape->printShape();
std::string command;
bool flag = true;
while (flag) {
printf("\n请输入对图形的操作指令:");
std::cin.ignore();
getline(std::cin, command);
transform(command.begin(),command.end(),command.begin(), ::tolower); //大小写转换
switch (resolveCommand(command)) {
case Motion::Rotate:
double angle;
printf("请输入旋转角度,逆时针为正:");
scanf("%lf", &angle);
shape->rotateShape(angle);
break;
case Motion::Move:
double mx, my;
printf("输入X方向移动距离:");
scanf("%lf", &mx);
printf("输入Y方向移动距离:");
scanf("%lf", &my);
shape->moveShape(mx, my);
break;
case Motion::Zoom:
double zx, zy;
printf("输入X方向缩放大小:");
scanf("%lf", &zx);
printf("输入Y方向缩放大小:");
scanf("%lf", &zy);
shape->zoomShape(zx, zy);
break;
default:
flag = false;
}
shape->printShape();
}
printf("Error!!!你输入的命令有误,已经退出指令模式退出\n");
delete shape;
delete[] shapePoints;
shapePoints = NULL;
定义了一个Point点类
point.h
#include "Eigen/Dense"
using namespace Eigen;
class Point {
private:
double x, y;
public:
Point();
Point(double x, double y);
void update(double x, double y);
void print();
void move(double x, double y);
void rotate(double angle);
void zoom(double zx, double zy);
Vector2d pointByMatrix();
void setByMatrix(Vector2d m);
~Point();
};
point.cpp
#include "Point.h"
#include <iostream>
Point::Point() {
this->x = 0;
this->y = 0;
}
Point::Point(double x, double y) {
this->x = x;
this->y = y;
}
void Point::update(double x, double y) {
this->x = x;
this->y = y;
}
void Point::print() {
std::cout << "(" << this->x << ", " << this->y << ")";
}
Point::~Point() {
std::cout << "调用Point析构函数" << std::endl;
}
/**
* 平移
*/
void Point::move(double x, double y) {
Vector2d a = pointByMatrix();
Vector2d b(x, y);
a = a + b;
setByMatrix(a);
}
/**
* 旋转变换,angle为角度, 逆时针为正, 顺时针为负
*/
void Point::rotate(double angle) {
MatrixXd T(2, 2);
angle = angle / 180 * M_PI;
T(0, 0) = cos(angle);
T(0, 1) = sin(angle);
T(1, 0) = -sin(angle);
T(1, 1) = cos(angle);
std::cout << "旋转矩阵T = " << std::endl << T << std::endl;
Vector2d m = this->pointByMatrix();
m = T * m; //貌似只能T * m
this->setByMatrix(m);
}
/**
* 缩放(比例变换), zx分别表示x轴缩放比例,zy表示y轴缩放比例
*/
void Point::zoom(double zx, double zy) {
Matrix2d T;
T << zx, 0, 0, zy;
Vector2d result = T * this->pointByMatrix();
this->setByMatrix(result);
}
Vector2d Point::pointByMatrix() {
Vector2d a(this->x, this->y);
return a;
}
void Point::setByMatrix(Vector2d m) {
this->x = m(0);
this->y = m(1);
}
定义了一个简单的Shape图形类。
shape.h
#include "Point.h"
class Shape {
private:
const char *shapeName;
int pointNum;
Point *pointArr;
public:
Shape(Point *arr, int num, const char *name);
void printShape();
void rotateShape(double angle);
void moveShape(double x, double y);
void zoomShape(double zx, double zy);
~Shape();
};
shape.cpp
#include "Shape.h"
Shape::Shape(Point *arr, int num, const char *name) {
this->pointArr = arr;
this->pointNum = num;
this->shapeName = name;
}
void Shape::moveShape(double x, double y) {
printf("你输入的指令为 move, 移动距离: (%lf, %lf)\n", x, y);
for (int i = 0; i < this->pointNum; ++i) {
this->pointArr[i].move(x, y);
}
}
void Shape::rotateShape(double angle) {
printf("你输入的指令为 rotate, 旋转角度: %lf\n", angle);
for (int i = 0; i < this->pointNum; ++i) {
this->pointArr[i].rotate(angle);
}
}
void Shape::zoomShape(double zx, double zy) {
printf("你输入的指令为 zoom, 缩放比例为: (%lf, %lf)\n", zx, zy);
for (int i = 0; i < this->pointNum; ++i) {
this->pointArr[i].zoom(zx, zy);
}
}
void Shape::printShape() {
printf("图形%s的%d个点坐标分别为", this->shapeName, this->pointNum);
for (int i = 0; i < this->pointNum - 1; ++i) {
this->pointArr[i].print();
printf(",");
}
this->pointArr[(this->pointNum - 1)].print();
printf("\n");
}
Shape::~Shape() {
printf("调用Helper析构函数\n");
}
运行结果为:

源码提交Github
Eigen库实现简单的旋转、平移操作的更多相关文章
- Eigen库矩阵运算使用方法
Eigen库矩阵运算使用方法 Eigen这个类库,存的东西好多的,来看一下主要的几个头文件吧: ——Core 有关矩阵和数组的类,有基本的线性代数(包含 三角形 和 自伴乘积 相关),还有相应对数组的 ...
- 我的Android进阶之旅】GitHub 上排名前 100 的 Android 开源库进行简单的介绍
GitHub Android Libraries Top 100 简介 本文转载于:https://github.com/Freelander/Android_Data/blob/master/And ...
- d3.js封装文本实现自动换行和旋转平移等功能
我们下面话不多说,本文主要介绍的是利用D3.js封装文本实现自动换行功能的步骤,下面来一起看看吧. 一.引用 multext.js 文件 multext.js function appendMulti ...
- C++标准库<string>简单总结
C++标准库<string>简单总结 在C++中,如果需要对字符串进行处理,那么它自带的标准库<string>无疑是最好的选择,它实现了很多常用的字符处理函数. 要想使用标准C ...
- Linq to SQL 简单的增删改操作
Linq to SQL 简单的增删改操作. 新建数据库表tbGuestBook.结构如下: 新建web项目,完成相应的dbml文件.留言页面布局如下 <body> <form id= ...
- NDK 开发实例二(添加 Eigen库)
上一篇,我已经阐述了如何创建一个简单的NDK实例: NDK 开发实例一(Android.mk环境配置下) 在上一篇的基础上,我们来添加Eigen库,然后做一个简单实例. Eigen是一个高层次的C + ...
- C++标准库vector类型的使用和操作总结
vector是一种类型对象的集合,它是一种顺序容器,容器中的所有对象必须都是同一种类型.想了解顺序容器的更多内容:C++顺序容器知识总结.vector的对象是可以动态生长的,这说明它在初始化时可以不用 ...
- app 下载更新 file-downloader 文件下载库的简单介绍和使用
app 下载更新 file-downloader 文件下载库的简单介绍和使用 今天介绍一个下载库:file-downloader 文件下载库 说明: * 本文内容来自原 file-downloader ...
- Eigen库和STL容器冲突问题
博客参考:https://blog.csdn.net/huajun998/article/details/54311561 在程序中想使用类似于如下的容器 std::vector<Eigne:: ...
随机推荐
- wpf依赖属性、绑定实现原理、附加属性学习
依赖属性和普通属性相比节省内存的原因:对于普通属性,每个对象有需要存储一个普通属性的值,即便是默认值.而依赖属性的默认值是静态的存储在类中的,所有对象都使用同一默认值,所以对于拥有大量属性的控件来说这 ...
- swift皮筋弹动发射飞机
今天在那个ios教程网上看到了一个不错的ios游戏源码,这是一个款采用swift实现的皮筋弹动发射飞机游戏源码,游戏源码比较详细,大家可以研究学习一下吧. <ignore_js_op> & ...
- C puzzles详解【51-57题】
第五十一题 Write a C function which does the addition of two integers without using the '+' operator. You ...
- Ubuntu安装wps for linux
1.WPS For Linux 2013 还是只提供了32位版本,我用的是 64位 Ubuntu,如果您也是64位系统,还需要提前安装一些32位的库文件. sudo apt-get install i ...
- 【easyui】--combobox--赋值和获取选中的值
//初始化下拉选框 $('#communityIdDiv').combobox({ url:basepath+"pushController/queryCommonityName" ...
- C# 添加系统计划任务方案
你可以在Windows 计划任务管理器里面添加计划任务,调试OK后导出XML文件,然后在要部署的计算机上执行CMD导入命令,把计划任务信息配置导入到服务器上的计划任务列表中,命令如下: SCHTASK ...
- DM9000网卡的基本工作原理
MAC:主要负责数据帧的创建,数据差错,检查,传送控制等. PHY:物理接口收发器,当收到MAC过来的数据时,它会加上校验码,然后按照物理层的规则进行数据编码,再发送到传输介质上,接收过程则相反. M ...
- 18.python的异常处理
python中至少包括两种错误:语法错误(syntax errors)和异常(exceptions). 1.语法错误 语法错误,也被称作解析错误
- EMVTag系列4《5A 应用主账号》
L:var.最大10 -M(必备):此数据应存在并提供给终端,终端在读应用数据过程中,如果没有读到必备数据,终端中止交易:等同磁条上的应用主帐户. 银行卡号一般是16位或者19位.由如下三部分构成: ...
- SSIS包配置动态配置数据库连接
动态连接数据库便于维护 用SSIS包配置实现 1.控制流页签 - 右键 - 包配置 2.配置xml文件 3.指定连接属性:ServerName.UserName.Password 测试: 1.配置错误 ...