最新ICS工厂有一项incam脚本新需求,这里介绍5种解决方法解决

需求如下图所示:绿色所圈处是是需求出的中心点(图形间距一致归为一类并计算中心点坐标)

前题条件:
1.一个SET里面可能有多个CAM,存在CAM1,CAM2,CAM3
2.每个CAM与CAM这最小间距不是固定值

对方法求解前;对此数据存储结构列出来:

    /// <summary>
/// Mod_step 坐标data类型 存放PNL中的子板排放坐标位置
/// </summary>
public class Mod_Sr_data
{
public string step_name { get; set; }
public gPoint ps;
public int angle { get; set; }
public bool mirror { get; set; }
public gPoint min;
public gPoint max;
public gPoint size
{
get { return new gPoint(Math.Abs(this.min.x - this.max.x), Math.Abs(this.min.y - this.max.y)); }
}
public gPoint center
{
get { return new gPoint((this.max.x + this.min.x) / , (this.max.y + this.min.y) / ); }
}
}

方法一:矩阵排序分组法求解

第一步:分别进行X与Y排列,如下图所示

 

第二步,求出X最近距离,与Y最近距离

第三步, 通过X与Y最近距离求出,间距分组ID号

第四步,通过间距分组ID号,遍历分组

第五步,通过每个分组求出中心点

缺点:只能支持矩阵排列(X数*Y数=PCS总数),X或Y间距全部需保持一致,最小左下角相连PCS最少2个,不支持PCS旋转

代码实现:

        private static  void SetCenterAddPad1()
{
step gstep = new step(g.JOB);
gProfile profile = g.getProfile(g.STEP, g.JOB);
List<Mod_Sr_data> sr_dataList = gstep.get_step_Sr_data(g.STEP);
List<Mod_Sr_data> sr_dataList_y_Order = sr_dataList.OrderBy(tt => tt.min.y).ThenBy(tt => tt.min.x).ToList();
List<Mod_Sr_data> sr_dataList_x_Order = sr_dataList.OrderBy(tt => tt.min.x).ThenBy(tt => tt.min.y).ToList();
List<int> x_CountList = new List<int>();
x_CountList.Add();
List<int> y_CountList = new List<int>();
y_CountList.Add();
double tempDi = ;
double minDi = ;
bool isMinDi = false;
int tempCount = ;
for (int i = ; i < sr_dataList_y_Order.Count - ; i++)
{
if (Math.Abs(sr_dataList_y_Order[i + ].min.y - sr_dataList_y_Order[i].min.y) > 0.1)
break;
tempDi = Math.Abs(sr_dataList_y_Order[i + ].min.x - sr_dataList_y_Order[i].min.x);
if (!isMinDi)
{
minDi = tempDi;
isMinDi = true;
x_CountList[tempCount] = i + ;
}
else
{
if ((Math.Abs(minDi - tempDi) < 0.001))
{
x_CountList[tempCount] = i + ;
}
else
{
tempCount++;
x_CountList.Add(i + );
}
}
}
isMinDi = false;
tempCount = ;
for (int i = ; i < sr_dataList_x_Order.Count - ; i++)
{
if (Math.Abs(sr_dataList_x_Order[i + ].min.x - sr_dataList_x_Order[i].min.x) > 0.1)
break;
tempDi = Math.Abs(sr_dataList_x_Order[i + ].min.y - sr_dataList_x_Order[i].min.y);
if (!isMinDi)
{
minDi = tempDi;
isMinDi = true;
y_CountList[tempCount] = i + ;
}
else
{
if ((Math.Abs(minDi - tempDi) < 0.001))
{
y_CountList[tempCount] = i + ;
}
else
{
tempCount++;
y_CountList.Add(i + );
}
}
} int x_Count = x_CountList.Max(tt => tt) + ;
int y_Count = y_CountList.Max(tt => tt) + ;
List<Mod_Sr_data>[] sr_dataListGroup = new List<Mod_Sr_data>[(x_CountList.Count * y_CountList.Count)];
for (int i = ; i < sr_dataListGroup.Count(); i++)
{
sr_dataListGroup[i] = new List<Mod_Sr_data>();
}
for (int i = ; i < sr_dataList_y_Order.Count; i++)
{
int x_index = i % x_Count;
int y_index = i / x_Count;
for (int j = ; j < x_CountList.Count; j++)
{
if (x_index <= x_CountList[j])
{
x_index = j;
break;
}
}
for (int j = ; j < y_CountList.Count; j++)
{
if (y_index <= y_CountList[j])
{
y_index = j;
break;
}
}
int index = y_index * x_CountList.Count + x_index;
sr_dataListGroup[index].Add(sr_dataList_y_Order[i]);
}
List<gPoint> gpointList = new List<gPoint>();
foreach (var item in sr_dataListGroup)
{
double xVal = item.Sum(tt => tt.center.x) / item.Count;
double yVal = item.Sum(tt => tt.center.y) / item.Count;
gpointList.Add(new gPoint(xVal, yVal));
}
add add_ = new add();
add_.pad(gpointList.ToArray(), );
}

方法二:坐标对号入坑法(类拟桶排序算法思想上改进)

第一步:分别进行X与Y排列

第二步,求出X最近距离,与Y最近距离

第三步, 建二维数组准备挖坑了(X与Y尺寸依据旋转时X与Y互换)

第四步,遍历数据填入到对应的坑位

第五步,通过二维的坑位依次对比最近距离X与Y进行划分数据分组

第五步,通过每个分组求出中心点

缺点:只能支持矩阵排列(中心可以缺少PCS),X或Y间距全部需保持一致,

代码实现,未完待完善

        private void SetCenterAddPad2()
{ step gstep = new step(g.JOB);
gProfile profile = g.getProfile(g.STEP, g.JOB);
List<Mod_Sr_data> sr_dataList = gstep.get_step_Sr_data(g.STEP);
List<Mod_Sr_data> sr_dataList_y_Order = sr_dataList.OrderBy(tt => tt.min.y).ThenBy(tt => tt.min.x).ToList();
List<Mod_Sr_data> sr_dataList_x_Order = sr_dataList.OrderBy(tt => tt.min.x).ThenBy(tt => tt.min.y).ToList();
double tempDi = ;
double yDi = ;
double xDi = ;
for (int i = ; i < sr_dataList_y_Order.Count - ; i++)
{
tempDi = Math.Abs(sr_dataList_y_Order[i + ].min.y - sr_dataList_y_Order[i].min.y);
if (tempDi > 0.01 && yDi > tempDi)
yDi = tempDi;
}
for (int i = ; i < sr_dataList_x_Order.Count - ; i++)
{
tempDi = Math.Abs(sr_dataList_x_Order[i + ].min.x - sr_dataList_x_Order[i].min.x);
if (tempDi > 0.01 && xDi > tempDi)
xDi = tempDi;
}
int x_array, y_array;
double xWidth, yHeigth;
double PcsAng = Math.Abs(sr_dataList[].angle - );
if ( < PcsAng && PcsAng < )
{
x_array = (int)Math.Ceiling(profile.Prof.size.y / sr_dataList[].size.y);
y_array = (int)Math.Ceiling(profile.Prof.size.x / sr_dataList[].size.x);
xWidth = sr_dataList[].size.y;
yHeigth = sr_dataList[].size.x;
}
else
{
x_array = (int)Math.Ceiling(profile.Prof.size.x / sr_dataList[].size.x);
y_array = (int)Math.Ceiling(profile.Prof.size.y / sr_dataList[].size.y);
xWidth = sr_dataList[].size.x;
yHeigth = sr_dataList[].size.y;
}
Mod_Sr_data[,] sr_dataArray = new Mod_Sr_data[x_array, y_array];
for (int i = ; i < sr_dataList_y_Order.Count; i++)
{
int x_index = (int)Math.Floor((sr_dataList_y_Order[i].min.x - profile.Prof.min.x) / xWidth);
int y_index = (int)Math.Floor((sr_dataList_y_Order[i].min.y - profile.Prof.min.y) / yHeigth);
sr_dataArray[x_index, y_index] = sr_dataList_y_Order[i];
}
List<int> x_CountList = new List<int>();
x_CountList.Add();
List<int> y_CountList = new List<int>();
y_CountList.Add();
for (int i = ; i < x_array-; i++)
{
for (int j = ; j < y_array-; j++)
{
var aa = sr_dataArray[i, j]; }
}
}

方法三:最近邻聚类算法

方法四:递归最左下角坐标定原点,进行相等距离求

方法五:扩边求解

PCB genesis SET取中心点--算法实现的更多相关文章

  1. K中心点算法之PAM

    一.PAM聚类算法:         选用簇中位置最中心的对象,试图对n个对象给出k个划分:代表对象也被称为是中心点,其他对象则被称为非代表对象:最初随机选择k个对象作为中心点,该算法反复地用非代表对 ...

  2. PCB Genesis SET拼板(圆形板拼板) 实现效果(二)

    越来发现Genesis采用Surface多边形数据结构的重要性了,当撑握了多边形缩放,交集, 差集,并集等算法, 想实现PCB拼板简直轻而易举了;当然借助多边形算法可以开发出更多的PCB实用的工具出来 ...

  3. PCB genesis自制孔点 Font字体实现方法

    一.先看genesis原有Font字体 在PCB工程CAM加孔点字体要求时,通常我们直接用Geneis软件给我们提供了2种孔点字体canned_57与canned_67,但此字体可能不能满足各个工厂个 ...

  4. 【转】C语言快速幂取模算法小结

    (转自:http://www.jb51.net/article/54947.htm) 本文实例汇总了C语言实现的快速幂取模算法,是比较常见的算法.分享给大家供大家参考之用.具体如下: 首先,所谓的快速 ...

  5. Raising Modulo Numbers_快速幂取模算法

    Description People are different. Some secretly read magazines full of interesting girls' pictures, ...

  6. 位运算之——按位与(&)操作——(快速取模算法)

    学习redis 字典结构,hash找槽位 求槽位的索引值时,用到了 hash值 & sizemask操作, 其后的scan操作涉及扫描顺序逻辑,对同模的槽位 按一定规则扫描! 其中涉及位运算 ...

  7. 【Java基础】14、位运算之——按位与(&)操作——(快速取模算法)

    学习redis 字典结构,hash找槽位 求槽位的索引值时,用到了 hash值 & sizemask操作, 其后的scan操作涉及扫描顺序逻辑,对同模的槽位 按一定规则扫描! 其中涉及位运算 ...

  8. 09 Memcached 分布式之取模算法的缺陷

    一: Memcached 分布式之取模算法的缺陷(1)假设你有8台服务器,运行中突然down一台,则求余数的底数就7. 后果: key_0%8==0 ,key_0%7==0 =>hist(命中) ...

  9. [原创][开源] SunnyUI.Net 开发日志:UIBarChart 坐标轴刻度取值算法

    _ 在开发UIBarChart的过程中,需要绘制Y轴的刻度,数据作图时,纵横坐标轴刻度范围及刻度值的取法,很大程度上取决于数据的分布.对某一组数据,我们很容易就能知道如何选取这些值才能使图画得漂亮.但 ...

随机推荐

  1. Linux未来监控tracing框架——eBPF

    Linux未来监控tracing框架--eBPF eBPF源于早年间的成型于 BSD 之上的传统技术 BPF(Berkeley Packet Filter).BPF 的全称是 Berkeley Pac ...

  2. JPA 与 JDBC 的区别和基本用法

    JPA 概念 JPA(Java Persistence API)用于对象持久化的 API,是 Java EE 5.0 平台标准的 ORM 规范,使得应用程序以统一的方式访问持久层. 与 JDBC 的对 ...

  3. [Luogu] P3258 [JLOI2014]松鼠的新家

    题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在”树“上. 松鼠想邀请小熊维尼前 ...

  4. UVA - 247 Calling Circles(Floyd求传递闭包)

    题目: 思路: 利用Floyd求传递闭包(mp[i][j] = mp[i][j]||(mp[i][k]&&mp[k][j]);),当mp[i][j]=1&&mp[j][ ...

  5. ebay 如何获取用户token

    1. 首先 配置环境加载依赖的ebay SDK 下载地址 https://go.developer.ebay.com/ebay-sdks 需要在本地仓库安装下面的jar mvn install:ins ...

  6. Thinkphp 批量更新方法 saveALL

    批量更新只适用于一个字段的更新,原理是用自定义函数拼接sql语句,然后再执行sql语句. //数据 $data[] = array('id'=>1,'value'=>value1); $d ...

  7. 我安装android studio的过程与经历

    虽然android studio已经出来两年多了,但是我一直都没真正用过.之前用Eclipse还算用得挺好.我并不是一个专职的android开发者,我是个游戏开发者,打包的时候要用到android.不 ...

  8. [K/3Cloud] KSQL日期常量用法注意

    KSQL中用日期常量必须用{ts'" + dateTime.ToString("yyyy-M-d HH:mm:ss") + "'} 正确写法: INSERT I ...

  9. Oracle删除约束和主键的语句

    https://blog.csdn.net/xue_yanan/article/details/78210654?locationNum=8&fps=1

  10. FreeMarker-简单示例

    以下是简单的FreeMarker示例,直接采用模板 + 数据模型 = 输出的方式.示例中是Application的项目,主要用于展示模板输出HTML文件的功能. 示例: 1.引入POM依赖 <! ...