模拟PLC 的圆弧插补方式在VC中绘制圆弧
最近同事想让要做一个绘图的控件。VC里面的画弧函数Arc需要提供外接矩形的坐标。同事觉得不好用,他更习惯圆弧插补的那种方式。于是看了看圆弧插补的东西。其实这种画弧方式就是提供圆弧的起点、终点和半径来画弧。
首先来简单介绍下圆弧插补:
有两种圆弧插补:
G02 顺时针圆弧插补
G03 逆时针圆弧插补
圆弧插补编程(半径编程):
圆弧用编程功能G02 或G03 和其后圆弧终点坐标和半径值定义。

圆弧半径用字母“R”表示。如果圆弧小于180 度,半径用正数符号,如果大于180 度用负数符号。这样基于所选圆弧插补(G02 或G03),可定义所选圆弧。

结合圆弧插补,设计绘制圆弧的函数:函数可分为两种,顺时针绘制和逆时针绘制(分布对应G02 和G03)。函数的参数为圆弧起点,终点,半径。其中的半径若为正数,则绘制的圆弧为弧度小于180 的弧,这里称为小圆弧。若半径为负数,则绘制的弧为大雨180度的弧,这里成之为大圆弧。
圆弧的绘制最终还是要使用C++ 提供的画弧函数Arc 。 因此我们需要找出来圆所在的外接矩形(这里是正方形)。因为我们已知半径,所以找到圆心就可以推导出圆所在的矩形。
圆心的推导过程参考文章已知圆上两点坐标和半径,求圆心 已知两点坐标和半径,求圆心 。圆心解出来有两个(x01,y01)(x02,y02)。如图所示,过相同的点并且半径相同的圆也确实有两个。那么到底哪一个是符合条件的圆呢。

首先来讨论逆时针画弧的函数。如上图,从起点A到终点B,小圆弧就指的红色部分的弧,大圆弧是指的蓝色部分的弧。小圆弧的圆心是O2,大圆弧的圆心是O1;
那么由什么条件能判断出所得的两个圆心(x01,y01)(x02,y02)哪一个是逆时针里的大圆弧的圆心O1,哪一个是逆时针里的小圆弧圆心O2呢? 这里我采用的是向量叉乘的方式判断的。

也就是起点到终点组成的向量,与起点与大弧圆心组成的向量叉乘结果是小于0 的。(这个从图上使用右手法则可以判断出来,由AB 向AO1 弯曲,拇指垂直屏幕向里)。
(有关向量知识参考 C语言-向量基本概念 向量叉乘判断点的位置)。
所以在上一步所得的两个圆心坐标,与起点坐标组成向量。
设A(x1,y1) B(x2,y2)
向量AB={x2-x1,y2-y1}
向量a={x01-x1,y01-y1}
向量b={x02-x1,y02-y1}
若
则(x01,y01)为大圆弧圆心 (x02,y02)为小圆弧圆心
否则 反之。
代码如下:
//已知圆弧上两点 和半径,求圆心
void CircleCenter(double x1,double y1,double x2,double y2,double R,double &x01,double &y01,double &x02,double &y02)
{
//x1 == x2
if (abs(x1-x2)<0.0000001)
{
//(x1,y1)(x2,y2)之间的距离 /2
double dis = abs(y1-y2)/2; double dx = sqrt(R*R-dis*dis); double dy = (y1+y2)/2; x01 = x1-dx;
y01 = dy; x02 = x1+dx;
y02 = dy;
return ;
} double c1 = (x2*x2 - x1*x1 + y2*y2 - y1*y1) / (2 *(x2 - x1));
double c2 = (y2 - y1) / (x2 - x1); //斜率
double A = (c2*c2 + 1);
double B = (2 * x1*c2 - 2 * c1*c2 - 2 * y1);
double C = x1*x1 - 2 * x1*c1 + c1*c1 + y1*y1 - R*R;
y01 = (-B + sqrt(B*B - 4 * A*C)) / (2 * A);
x01 = c1 - c2 * y01; y02 = (-B - sqrt(B*B - 4 * A*C)) / (2 * A);
x02 = c1 - c2*y02;
}
//逆时针画弧
void CDrawShapeCtrl::Arc_AntiClock(DOUBLE StartX, DOUBLE StartY, DOUBLE EndX, DOUBLE EndY, DOUBLE R)
{
AFX_MANAGE_STATE(AfxGetStaticModuleState()); // TODO: Add your dispatch handler code here
//圆心坐标
double x01,y01,x02,y02;
double x_big,y_big;//大弧圆心
double x_small,y_small;//小弧圆心 LONG nLeftRect, nTopRect,nRightRect,nBottomRect; CircleCenter(StartX,StartY,EndX,EndY,R,x01,y01,x02,y02); //向量
double ax = EndX- StartX;
double ay = EndY - StartY; double bx = x01 - StartX;
double by = y01 - StartY; //利用向量的叉乘判断圆心位置
//叉乘<0 则为大弧圆心;否则为小弧圆心
double mulRt = ax*by-bx*ay; if (mulRt<0)
{
x_big = x01;
y_big = y01;
x_small = x02;
y_small = y02;
}
else
{
x_big = x02;
y_big = y02;
x_small = x01;
y_small = y01;
} CClientDC dc(this); CRect rc;
GetClientRect(rc);
dc.SetMapMode(MM_ISOTROPIC);//MM_ISOTROPIC //逻辑坐标原点
dc.SetViewportOrg(rc.right/2,rc.bottom/2);
//设置映射比例为1,逻辑坐标Y轴方向与设备坐标相反
dc.SetWindowExt(100,100);
dc.SetViewportExt(100,-100); //R>0 弧<180度; R<0 弧>180度
if (R<0) //大弧
{
nLeftRect = x_big-R;
nTopRect = y_big + R;
nRightRect = x_big+R;
nBottomRect = y_big -R; dc.Arc(nLeftRect,nTopRect,nRightRect,nBottomRect,StartX,StartY,EndX,EndY);
}
else //小弧
{
nLeftRect = x_small-R;
nTopRect = y_small+R;
nRightRect = x_small+R;
nBottomRect = y_small - R;
dc.Arc(nLeftRect,nTopRect,nRightRect,nBottomRect,StartX,StartY,EndX,EndY); } }
顺时针函数,只要将起点终点坐标对换,直接调用逆时针函数即可。
arc 函数参考:
https://blog.csdn.net/u012513234/article/details/45460783
模拟PLC 的圆弧插补方式在VC中绘制圆弧的更多相关文章
- [转]VC中调用外部exe程序方式
本文转自:http://blog.sina.com.cn/s/blog_486285690100ljwu.html 目前知道三种方式:WinExec,ShellExecute ,CreateProce ...
- VC中GetLastError()获取错误信息的使用,以及错误代码的含义
转载:http://www.seacha.com/article.php/knowledge/windows/mfc/2011/0423/335.html VC中GetLastError()获取错误信 ...
- VC++中开发汇编语言(转)
汇编程序结构 一个显示字符串的汇编程序 程序格式 一.模式定义 二.includelib语句 三.函数声明语句 四.数据和代码部分 Visual C/C++环境 建立工程 汇编程序的调试 一.设置断点 ...
- VC中基于 Windows 的精确定时[转]
在工业生产控制系统中,有许多需要定时完成的操作,如定时显示当前时间,定时刷新屏幕上的进度条,上位 机定时向下位机发送命令和传送数据等.特别是在对控制性能要求较高的实时控制系统和数据采集系统中,就更需要 ...
- VC中LINK 2001 和 LINK 2009 的错误的解决
最近将两个开源C++项目编译成windows版本的时候遇到很多问题,关键是两个项目经过同事的修改之后,一个项目引用了另一个项目,两个项目的头文件中都有一些跨平台的关于数据类型,以及一些通用函数的定义, ...
- VC中Source Files, Header Files, Resource Files,External Dependencies的区别
VC中Source Files, Header Files, Resource Files,External Dependencies的区别 区别: Source Files 放源文件(.c..cpp ...
- 如何在VC中查询中文,及QT5的中文处理
1,VC中查询所有含中文字符串 使用正则表达式,查询内容为: (".*[\u4E00-\u9FA5]+)|([\u4E00-\u9FA5]+.*") 这样,就可以把代码中的中文都查 ...
- VC中不同类型DLL及区别
1. DLL的概念可以向程序提供一些函数.变量或类. 静态链接库与动态链接库的区别:(1)静态链接库与动态链接库都是共享代码的方式.静态链接库把最后的指令都包含在最终生成的EXE文件中了:动态链接库不 ...
- VC中如何获取当前时间(精度达到毫秒级)
标 题: VC中如何获取当前时间(精度达到毫秒级)作 者: 0xFFFFCCCC时 间: 2013-06-24链 接: http://www.cnblogs.com/Y4ng/p/Millisecon ...
随机推荐
- Oracle使用外部表批量创建用户
整体思路:通过使用外部表将用户名导入Oracle的表中,然后通过PL/SQL遍历数据表,批量创建用户. 具体步骤如下: 1.在安装数据库的服务器的C盘根目录创建一个User List.txt文件,内容 ...
- SQL Server Service Broker 示例(转)
1.定义数据类型.协议和服务(发送服务和接收服务) USE master; GO ALTER DATABASE 目标数据库 SET ENABLE_BROKER; GO -- 如果上面的操作执行后,长时 ...
- C#编程时应注意的性能处理
GC堆回收 那么除了通过new对象而达到代的阈(临界)值时,还有什么能够导致垃圾堆进行垃圾回收呢? 还可能windows报告内存不足.CLR卸载AppDomain.CLR关闭等其它特殊情况. 或者,我 ...
- sql语句实例练习
1.最晚入职员工查询 select * from employees where hire_date = (select max(hire_date) from employees) 2.倒数第三 ...
- 嵌入式GCC笔记
GNU C Complier 查看 gcc 版本 :gcc -v 文件后缀名 .c的后缀 为C语言的文件源码 gcc -o output gcc -o 输出文件名字 输入文件名 gcc -v -o ...
- 自己DIY出来一个JSON结构化展示器
说来也巧,这个玩意,一直都想亲手写一个,因为一直用着各种网上提供的工具,觉得这个还是有些用途,毕竟,后面的实现思路和原理不是太复杂,就是对json的遍历,然后给予不同节点类型以不同的展现风格. 我这次 ...
- 数据分组分析—-groupby
数据分组分析—-groupby 代码功能: 对于综合表格data,基于title进行分组处理,并统计每一组的size,得到的是一个series序列,此序列可以放入索引中使用,index() impor ...
- 黑电平校正BLC
参考:https://www.cnblogs.com/zhangAlin/p/10661763.html
- 浏览器端使用javascript调用腾讯翻译api
最近在学习的小玩意,发现腾讯的文档十分坑爹,里面有很多错误的指示. 不过腾讯的机器翻译还是很牛的,我觉得翻译水准比谷歌好很多. 腾讯的机器翻译貌似在试用阶段,不收费,用QQ或微信登录即可申请使用. 首 ...
- mi家前端面经
已经好久没想写面经了……菜鸟面到生无可恋. 1.用CSS实现下面圆形 答案: <!DOCTYPE html> <html> <head> <style typ ...