.h:
@interface XXXViewController : UITViewController {
int direction;
BOOL isBlue;
}
.m:
头部导入: #import <QuartzCore/QuartzCore.h>
- (void)viewDidLoad {
[super viewDidLoad];
direction = 0;
isBlue = YES;
UISegmentedControl *sc = [[UISegmentedControl alloc] initWithItems:[@"Ripple Curl Uncurl Suck"componentsSeparatedByString:@" "]];
sc.segmentedControlStyle = UISegmentedControlStyleBar;
sc. selectedSegmentIndex = 0;
self.navigationItem.titleView = [sc autorelease];
UIView *bgView = [[UIView alloc] initWithFrame:CGRectMake(10, 10, 300, 240)];
bgView.backgroundColor = [UIColor lightGrayColor];
bgView.tag = 150;
UIImageView *imgView = [[UIImageView alloc] initWithImage:[UIImageimageNamed:@"BFlyCircle.png"]];
[bgView addSubview:imgView];
imgView.tag = 151;
imgView.center = CGPointMake(150, 120);
UIImageView *imgView2 = [[UIImageView alloc] initWithImage:[UIImageimageNamed:@"BFlyCircleMaroon.png"]];
[bgView addSubview:imgView2];
imgView2.tag = 152;
imgView2.center = CGPointMake(150, 120);
UIButton *startBtn = [UIButton buttonWithType:UIButtonTypeRoundedRect];
[startBtn setTitle:@"开始" forState:UIControlStateNormal];
startBtn.frame = CGRectMake(120, 260, 80, 25);
[startBtn addTarget:self action:@selector(startAction:)forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:bgView];
[self.view addSubview:startBtn];
}
- (void) startAction:(id) sender{
// 定义动画
CATransition *animation = [CATransition animation];
animation.delegate = self;
animation.duration =1.0f;
animation.timingFunction = UIViewAnimationCurveEaseInOut;
//可根据需要,设置type和subtype属性产生不同的动画效果
//animation.type = kCATransitionPush;
//animation.type = kCATransitionMoveIn;
//animation.type = kCATransitionReveal;
//animation.type = kCATransitionFade;
//animation.subtype = kCATransitionPush;
//animation.subtype = kCATransitionMoveIn;
//animation.subtype = kCATransitionReveal;
//animation.subtype = kCATransitionFade;
switch ([(UISegmentedControl *)self.navigationItem.titleView selectedSegmentIndex]) {
case 0:
animation.type = @"rippleEffect";
//animation.type = @"cube"; //立方体翻转
//animation.type = @"oglFlip"; //层翻转
//animation.type = @"cameraIrisHollowClose";
//animation.type = @"cameraIrisHollowOpen";
break;
case 1:
animation.type = @"pageCurl";
break;
case 2:
animation.type = @"pageUnCurl";
break;
case 3:
animation.type = @"suckEffect";
break;
default:
break;
}
switch (direction) { //方向
case 0:
animation.subtype = kCATransitionFromRight;
break;
case 1:
animation.subtype = kCATransitionFromTop;
break;
case 2:
animation.subtype = kCATransitionFromLeft;
break;
case 3:
animation.subtype = kCATransitionFromBottom;
break;
default:
break;
}
//执行动画
UIView *bgView = [self.view viewWithTag:150];
NSInteger front = [[bgView subviews] indexOfObject:[bgView viewWithTag:151]];
NSInteger back = [[bgView subviews] indexOfObject:[bgView viewWithTag:152]];
if (isBlue) {
[bgView exchangeSubviewAtIndex:front withSubviewAtIndex:back]; //图形变换
}else {
[bgView exchangeSubviewAtIndex:back withSubviewAtIndex:front];
}
[[bgView layer] addAnimation:animation forKey:@"animation"]; //bgView层执行动画
//[[self.view layer] addAnimation:animation forKey:@"animation"]; //self.view层执行动画
if(++direction > 3) direction -= 4;
isBlue = !isBlue;
}
原创博客,转载请注明出处:http://blog.csdn.net/renshengrumenglibing?viewmode=contents
机器人定位的目的是为了知道“自己在什么地方”,目前,机器人定位的方法可以分为非自主定位与自
主定位两大类。所谓非自主定位是在定位的过程中机器人需要借助机器人本身以外的装置如:全球定位
系统(GPS)、全局视觉系统等进行定位;自主定位是机器人仅依靠机器人本身携带的传感器进行定位。
由于在室内环境中,不能使用GPS,而安装其它的辅助定位系统比较麻烦。因此机器人一般采用自主
定位的方法。
按照初始位姿是否已知,可把机器人自主定位分为初始位姿已知的位姿跟踪(Pose tracking)和初始位
姿未知的全局定位(Global localization)。
位姿跟踪是在已知机器人的初始位姿的条件下,在机器人的运动过程中通过将观测到的特征与地图中
的特征进行匹配,求取它们之间的差别,进而更新机器人的位姿的机器人定位方法。位姿跟踪通常采用扩
展卡尔曼滤波器(Extended Kalman Filter,EKF)来实现。该方法采用高斯分布来近似地表示机器人位姿
的后验概率分布,其计算过程主要包括三步:首先是根据机器人的运模型预测机器人的位姿,然后将观测
信息与地图进行匹配,最后根据预测后的机器人位姿以及匹配的特征计算机器人应该观测到的信息,并利
用应该观测到的信息与实际观测到的信息之间的差距来更新机器人的位姿。
全局定位是在机器人的初始位姿不确定的条件下,利用局部的、不完全的观测信息估计机器人的当前
位姿。能否解决最典型而又最富挑战性的“绑架恢复”问题在一定程度上反应了机器人全局定位方法的鲁棒
性与可靠性。
一、移动机器人 SLAM 技术
可靠的定位性能是自主移动系统的关键要素。传统的定位方法是基于里程计估计的,存在不可避免的
定位误差。自从移动机器人诞生以来,对定位问题的研究就和地图创建问题密切关联,已知环境地图的定
位问题和已知定位的地图创建问题已经被广泛研究,提出了多种有效的解决途径。当地图和机器人的位置
都事先未知时,问题就变得更加复杂,出现了许多独有的新特征。在这种情况下,要求机器人在一个完全
未知的环境中从一个未知的位置出发,在递增地建立环境的导航地图同时,利用已建立的地图来同步刷新
自身的位置。该问题被称作同步定位和构图,简称 SLAM。在 SLAM 问题中,机器人位置和地图两者的估
算是高度相关的,任何一方都无法独立获取,这样形成了一种相辅相生、不断迭代的过程,因此有些学者
将其比作“鸡与蛋”问题。
近年来,移动机器人 SLAM 技术获得显著进步,被认为是解决环境未知和传感器信息不确定条件下的
移动机器人自主导航的最有效的技术之一。SLAM 基本思想是利用已创建地图修正基于运动模型的机器人
位姿估计误差;同时根据可靠的机器人位姿,创建出精度更高的地图。
关于传感器的不确定,以最常见的里程计为例,其典型的误差积累如图 所示,其中,左图是独立
利用里程计定位、独立利用激光传感器感知环境所创建的地图,由于没有进行里程计误差补偿,几次创
建的地图差异很大,与实际环境也不符;右图是采用 SLAM 创建的地图,基于 SLAM 可以利用已创建的
地图修正里程计的误差,这样机器人的位姿误差就不会随着机器人的运动距离的增大而无限制增长,因
此可以创建精度更高的地图,也同时解决了未知环境中的机器人定位问题。
SLAM中,系统的状态由机器人的位姿和地图信息(特征的位置信息)组成。假设机器人在t时刻观测到
特征m1,如图2所示。根据观测信息只能获得特征m1在机器人坐标系R中的坐标。机器人需要估计机器
人自己本身在世界坐标系W中的位姿,然后通过坐标变换才能计算特征的世界坐标。可见在地图创建的过
程中,必须计算机器人的位姿,也就是进行机器人的定位。然而,根据里程计获得的机器人位置信息很不
准确,显然错误的位置信息将会导致地图的不准确。
//SLAM示意图
在初始时刻,激光雷达创建的地图中并没有任何的特征。当机器人观测到某特征m时,可以根据
机器人的位姿,以及特征在机器人坐标系下的位姿,计算出特征在世界坐标系下的位姿,此时将特征
加入到地图中(更新地图);当机器人位姿改变,再次观测到特征m,可以根据特征在世界坐标系下
的位姿和特征在机器人坐标系下的位姿,解算出当前机器人的位姿(机器人定位)。
当机器人继续运动时,它将观测到更多的特征,根据同样的方法。机器人会把它们加入到地图中,
并且根据观测到的信息更新机器人的位姿以及它们的世界坐标。简单的说,SLAM利用观测到的特征计
算它们的世界坐标以实现地图创建,同时更新机器人的位姿以实现机器人的定位。
SLAM方法有很多种,主要包括基于扩展卡尔曼滤波的SLAM技术,基于传统粒子滤波的SLAM技术,
快速SLAM技术,基于扫描匹配的SLAM技术等等。
1.1基于扫描匹配的 SLAM 技术
基于扫描匹配的 SLAM是基于最近邻扫描匹配来估计两次扫描间机器人的平移和旋转的算法。扫描
匹配算法主要源自迭代最近点(Iterative Closest Point, ICP)算法及其改进算法。该算法通过迭代细调由
机器人里程计给出的初始位姿,限定了搜索空间。然而,该算法假定机器人的初始位姿和机器人的真实
位姿之间的偏差足够小,以便于达到全局最优匹配。
1.2 范例 :
已知两条直线在激光雷达坐标系下和全局坐标系下的直线方程,求激光雷达在全局坐标下
的位置(x,y)和姿态theta
实际上已知两条直线求解是多解的,当theta是真实解,那么theta+pi同样是方程组的解,此时
可以引入新的约束,激光雷达实际上看到的两条直线,只能是直线交叉点一侧的部分,那么求解之
后可以进行验证,进而排除一个解,此时解唯一。
假设直线L1 全局下的坐标方程分别为y = a1*x + b,在雷达坐标系的方程y = a2*x+b2;
倾斜角分别为thetaW和thetaR,那么由转角alpha + thetaR = thetaW => alpha = thetaW - thetaR;
[Xw] [cos(alpha) -sin(alpha)] Xr Tx
= * +
[Yw] [sin(alpha) cos(alpha)] Yr Ty
由两条直线解出格子坐标系下的交点,代入上式可以解出Tx Ty,上式中都是矩阵计算,由于没有word那么
强大,各位只能勉强看了。
此时实际上解释存在两个的,解出之后需要进行验证,记下线段的端点,看知否在交点的同一侧,如果在同
一侧,那么结果就是对的,否则就要再转180度。
直线的拟合参照上一篇笔记,http://blog.csdn.net/renshengrumenglibing/article/details/8604245
为了提高进度,我们可以对数据进行一次中值滤波,抑制噪声同时尽量保留数据的边沿。
//中值滤波 只能对初始的连续数据滤波 //滤波基本不丢弃数据,两端会各自扔掉几个数据 void OpenRadar::MedFilter(vector<int>& RadarRho, vector<double>& RadarTheta){ vector<int>rho; vector<double>theta; int halfWindowSize = 2; int *neighbor = new int[2*halfWindowSize+1]; int temp; for (int i = halfWindowSize; i< (int)RadarRho.size() - halfWindowSize;i++) { for (int j = -halfWindowSize;j <= halfWindowSize;j++) { neighbor[j + halfWindowSize] = RadarRho.at(i + j); } //排序 for (int m = 0; m < 2*halfWindowSize +1;m++) { for (int n = m + 1;n < 2*halfWindowSize + 1;n++) { if (neighbor[m]> neighbor[n]) { temp = neighbor[m]; neighbor[m] = neighbor[n]; neighbor[n] = temp; } } } rho.push_back(neighbor[halfWindowSize]); theta.push_back(RadarTheta.at(i)); } RadarRho.clear(); RadarTheta.clear(); for (int i = 0; i < (int)(rho.size());i++) { RadarRho.push_back(rho.at(i)); RadarTheta.push_back(theta.at(i)); } }
其他处理跟之前相同,此时可以根据已知的两条直线进行位姿解算。
//已知四条直线如何计算变换参数 void Coordinate::CalCoorTransPara(CoorTransPara &transPara, LinePara W1, LinePara W2, LinePara R1, LinePara R2) { double theta = ( W1.Rho - R1.Rho + W2.Rho - R2.Rho )/2; //double theta = ( W1.Rho - R1.Rho); //求解出Xw Yw Xr Yr double Xw = (double)(W1.b - W2.b)/(W2.a - W1.a); double Yw = W1.a*Xw + W1.b; double Xr = (double)(R1.b - R2.b)/(R2.a - R1.a); double Yr = R1.a*Xr + R1.b; int Tx = (int)(Xw - cos(theta)*Xr + sin(theta)*Yr); int Ty = (int)(Yw - sin(theta)*Xr - cos(theta)*Yr); //交点判定,场地上的几条直线都是有角点的 iPoint crossPoint;//交点 iPoint vectorW1,vectorR1;//向量 //iPoint vectorR2,vectorW2; if (W1.startPoint.x == W2.startPoint.x && W1.startPoint.y == W2.startPoint.y) { crossPoint = ipoint(W1.startPoint.x,W1.startPoint.y); vectorW1 = ipoint(W1.endPoint.x - W1.startPoint.x, W1.endPoint.y - W1.startPoint.y); //vectorW2 = ipoint(W2.endPoint.x - W2.startPoint.x, W2.endPoint.y - W2.startPoint.y); }else if (W1.endPoint.x == W2.startPoint.x && W1.endPoint.y == W2.startPoint.y) { crossPoint = ipoint(W1.endPoint.x,W1.endPoint.y); vectorW1 = ipoint(W1.startPoint.x - W1.endPoint.x, W1.startPoint.y - W1.endPoint.y); //vectorW2 = ipoint(W2.endPoint.x - W2.startPoint.x, W2.endPoint.y - W2.startPoint.y); }else if (W1.startPoint.x == W2.endPoint.x && W1.startPoint.y == W2.endPoint.y) { crossPoint = ipoint(W1.startPoint.x,W1.startPoint.y); vectorW1 = ipoint(W1.endPoint.x - W1.startPoint.x, W1.endPoint.y - W1.startPoint.y); //vectorW2 = ipoint(W2.startPoint.x - W2.endPoint.x, W2.startPoint.y - W2.endPoint.y); }else if (W1.endPoint.x == W2.endPoint.x && W1.endPoint.y == W2.endPoint.y) { crossPoint = ipoint(W1.endPoint.x,W1.endPoint.y); vectorW1 = ipoint(W1.startPoint.x - W1.endPoint.x, W1.startPoint.y - W1.endPoint.y); //vectorW2 = ipoint(W2.startPoint.x - W2.endPoint.x, W2.startPoint.y - W2.endPoint.y); } //将激光雷达下的两个点旋转到W系下 transPara.theta = theta; transPara.Tx = Tx; transPara.Ty = Ty; iPoint R1ToW; //iPoint R2ToW; TransformCoord(transPara,R1.startPoint,R1ToW); //TransformCoord(transPara,R2.startPoint,R2ToW); vectorR1.x = R1ToW.x - crossPoint.x; vectorR1.y = R1ToW.y - crossPoint.y; //判断是否在同一侧? if (vectorW1.x * vectorR1.x + vectorW1.y*vectorR1.y < 0) { //旋转角度差了180度,需要调转180度 transPara.theta = theta + PI; transPara.Tx = (int)(Xw - cos(transPara.theta)*Xr + sin(transPara.theta)*Yr); transPara.Ty = (int)(Yw - sin(transPara.theta)*Xr - cos(transPara.theta)*Yr); }else{ } //数据测试 /* TransformCoord(transPara,R1.startPoint,R1ToW); cout<<"R1ToW.x "<<R1ToW.x<<" R1ToW.y "<<R1ToW.y<<endl; TransformCoord(transPara,R1.endPoint,R1ToW); cout<<"R1ToW.x "<<R1ToW.x<<" R1ToW.y "<<R1ToW.y<<endl; TransformCoord(transPara,R2.startPoint,R2ToW); cout<<"R2ToW.x "<<R2ToW.x<<" R2ToW.y "<<R2ToW.y<<endl; TransformCoord(transPara,R2.endPoint,R2ToW); cout<<"R2ToW.x "<<R2ToW.x<<" R2ToW.y "<<R2ToW.y<<endl;*/ //进行一次验证,看看交点进行坐标变换之后是否接近匹配的点 /* iPoint R = ipoint(Xr,Yr); iPoint R2W; TransformCoord(transPara,R,R2W); cout<<"R2W.x "<<R2W.x<<" R2W.y "<<R2W.y<<endl;*/ }
//完整的Coordinate.h
#pragma once #include "WeightedFit.h" #include <iostream> using namespace std; //场地中的关键点 static iPoint FieldPointA = ipoint(9192,0); static iPoint FieldPointB = ipoint(0,9192); static iPoint FieldPointC = ipoint(-9192,0); static iPoint FieldPointD = ipoint(0,-9192); //场地中的直线变量 static LinePara FieldLine1 = linePara(-1.0,9192.3881554,FieldPointA,FieldPointB); static LinePara FieldLine2 = linePara(1.0,9192.3881554,FieldPointB,FieldPointC); static LinePara FieldLine3 = linePara(-1.0,-9192.3881554,FieldPointC,FieldPointD); static LinePara FieldLine4 = linePara(1.0,-9192.3881554,FieldPointD,FieldPointA); static LinePara FieldLine5 = linePara(100000.0,0.0,FieldPointB,FieldPointD); //场地中的圆 static CirclePara FieldCircle1 = circlePara(-3000,1301,400,350); static CirclePara FieldCircle2 = circlePara(-1951,880,400,350); static CirclePara FieldCircle3 = circlePara(-651,815,400,350); static CirclePara FieldCircle4 = circlePara(-495,2416,400,350); static CirclePara FieldCircle5 = circlePara(-3347,-997,400,350); static CirclePara FieldCircle6 = circlePara(-2400,-2848,400,350); static CirclePara FieldCircle7 = circlePara(-1499,-2499,400,350); static CirclePara FieldCircle8 = circlePara(3000,1301,400,350); static CirclePara FieldCircle9 = circlePara(1951,880,400,350); static CirclePara FieldCircle10 = circlePara(651,815,400,350); static CirclePara FieldCircle11= circlePara(495,2416,400,350); static CirclePara FieldCircle12 = circlePara(3347,-997,400,350); static CirclePara FieldCircle13 = circlePara(2400,-2848,400,350); static CirclePara FieldCircle14 = circlePara(1499,-2499,400,350); //坐标系类,进行坐标系相关的计算 typedef struct{ int Tx; int Ty; double theta;//旋转角 }CoorTransPara; //坐标变换参数 class Coordinate { public: Coordinate(void); ~Coordinate(void); //已知四条直线如何计算变换参数 void CalCoorTransPara(CoorTransPara &transPara, LinePara W1, LinePara W2, LinePara R1, LinePara R2); void CoortransTest(); void CalRadarCoord(); CoorTransPara RadarCoordTransPara;//全局坐标系和雷达坐标系之间的转换参数 void printRadarCoordtransPara(CoorTransPara coordtrans); void TransformCoord(CoorTransPara transPara,iPoint R,iPoint& W); };
#include "Coordinate.h" Coordinate::Coordinate(void) { } Coordinate::~Coordinate(void) { } //已知四条直线如何计算变换参数 void Coordinate::CalCoorTransPara(CoorTransPara &transPara, LinePara W1, LinePara W2, LinePara R1, LinePara R2) { double theta = ( W1.Rho - R1.Rho + W2.Rho - R2.Rho )/2; //double theta = ( W1.Rho - R1.Rho); //求解出Xw Yw Xr Yr double Xw = (double)(W1.b - W2.b)/(W2.a - W1.a); double Yw = W1.a*Xw + W1.b; double Xr = (double)(R1.b - R2.b)/(R2.a - R1.a); double Yr = R1.a*Xr + R1.b; int Tx = (int)(Xw - cos(theta)*Xr + sin(theta)*Yr); int Ty = (int)(Yw - sin(theta)*Xr - cos(theta)*Yr); //交点判定,场地上的几条直线都是有角点的 iPoint crossPoint;//交点 iPoint vectorW1,vectorR1;//向量 //iPoint vectorR2,vectorW2; if (W1.startPoint.x == W2.startPoint.x && W1.startPoint.y == W2.startPoint.y) { crossPoint = ipoint(W1.startPoint.x,W1.startPoint.y); vectorW1 = ipoint(W1.endPoint.x - W1.startPoint.x, W1.endPoint.y - W1.startPoint.y); //vectorW2 = ipoint(W2.endPoint.x - W2.startPoint.x, W2.endPoint.y - W2.startPoint.y); }else if (W1.endPoint.x == W2.startPoint.x && W1.endPoint.y == W2.startPoint.y) { crossPoint = ipoint(W1.endPoint.x,W1.endPoint.y); vectorW1 = ipoint(W1.startPoint.x - W1.endPoint.x, W1.startPoint.y - W1.endPoint.y); //vectorW2 = ipoint(W2.endPoint.x - W2.startPoint.x, W2.endPoint.y - W2.startPoint.y); }else if (W1.startPoint.x == W2.endPoint.x && W1.startPoint.y == W2.endPoint.y) { crossPoint = ipoint(W1.startPoint.x,W1.startPoint.y); vectorW1 = ipoint(W1.endPoint.x - W1.startPoint.x, W1.endPoint.y - W1.startPoint.y); //vectorW2 = ipoint(W2.startPoint.x - W2.endPoint.x, W2.startPoint.y - W2.endPoint.y); }else if (W1.endPoint.x == W2.endPoint.x && W1.endPoint.y == W2.endPoint.y) { crossPoint = ipoint(W1.endPoint.x,W1.endPoint.y); vectorW1 = ipoint(W1.startPoint.x - W1.endPoint.x, W1.startPoint.y - W1.endPoint.y); //vectorW2 = ipoint(W2.startPoint.x - W2.endPoint.x, W2.startPoint.y - W2.endPoint.y); } //将激光雷达下的两个点旋转到W系下 transPara.theta = theta; transPara.Tx = Tx; transPara.Ty = Ty; iPoint R1ToW; //iPoint R2ToW; TransformCoord(transPara,R1.startPoint,R1ToW); //TransformCoord(transPara,R2.startPoint,R2ToW); vectorR1.x = R1ToW.x - crossPoint.x; vectorR1.y = R1ToW.y - crossPoint.y; //判断是否在同一侧? if (vectorW1.x * vectorR1.x + vectorW1.y*vectorR1.y < 0) { //旋转角度差了180度,需要调转180度 transPara.theta = theta + PI; transPara.Tx = (int)(Xw - cos(transPara.theta)*Xr + sin(transPara.theta)*Yr); transPara.Ty = (int)(Yw - sin(transPara.theta)*Xr - cos(transPara.theta)*Yr); }else{ } //数据测试 /* TransformCoord(transPara,R1.startPoint,R1ToW); cout<<"R1ToW.x "<<R1ToW.x<<" R1ToW.y "<<R1ToW.y<<endl; TransformCoord(transPara,R1.endPoint,R1ToW); cout<<"R1ToW.x "<<R1ToW.x<<" R1ToW.y "<<R1ToW.y<<endl; TransformCoord(transPara,R2.startPoint,R2ToW); cout<<"R2ToW.x "<<R2ToW.x<<" R2ToW.y "<<R2ToW.y<<endl; TransformCoord(transPara,R2.endPoint,R2ToW); cout<<"R2ToW.x "<<R2ToW.x<<" R2ToW.y "<<R2ToW.y<<endl;*/ //进行一次验证,看看交点进行坐标变换之后是否接近匹配的点 /* iPoint R = ipoint(Xr,Yr); iPoint R2W; TransformCoord(transPara,R,R2W); cout<<"R2W.x "<<R2W.x<<" R2W.y "<<R2W.y<<endl;*/ } void Coordinate::CoortransTest(){ Coordinate coord; CoorTransPara coordtrans; coord.CalCoorTransPara(coordtrans,FieldLine1,FieldLine5,FieldLine2,FieldLine5); cout<<"theta : "<<coordtrans.theta*180/PI<<" Tx: "<<coordtrans.Tx<<" Ty: "<<coordtrans.Ty<<endl; } void Coordinate::printRadarCoordtransPara(CoorTransPara coordtrans){ cout<<"theta : "<<coordtrans.theta*180/PI<<" Tx: "<<coordtrans.Tx<<" Ty: "<<coordtrans.Ty<<endl; } void Coordinate::TransformCoord(CoorTransPara transPara,iPoint R,iPoint& W){ W.x = (int)(R.x*cos(transPara.theta) - R.y*sin(transPara.theta) ); W.y = (int)(R.x*sin(transPara.theta) + R.y*cos(transPara.theta) ); W.x = W.x + transPara.Tx ; W.y = W.y + transPara.Ty ; }
前边两篇文章,主要介绍了Objective-C的runtime system以及message相关的一些概念上的东西,都是为今天要说的东西做铺垫。
五、消息调用流程
一切还是从消息表达式[receiver message]开始,在被转换成objc_msgSend(receiver, SEL)后,在运行时,runtime system会做以下事情:
1、检查忽略的Selector,比如当我们运行在有垃圾回收机制的环境中,将会忽略retain和release消息。
2、检查receiver是否为nil。不像其他语言,nil在objective-C中是完全合法的,并且这里有很多原因你也愿意这样,比如,至少我们省去了给一个对象发送消息前检查对象是否为空的操作。如果receiver为空,则会将 selector也设置为空,并且直接返回到消息调用的地方。如果对象非空,就继续下一步。
3、接下来会根据SEL到当前类中查找对应的IMP,首先会在cache中检索它,如果找到了就根据函数指针跳转到这个函数执行,否则进行下一步。
4、检索当前类对象中的方法表(method list),如果找到了,加入cache中,并且就跳转到这个函数之行,否则进行下一步。
5、从父类中寻找,直到根类:NSObject类。找到了就将方法加入对应类的cache表中,如果仍为找到,则要进入第七项介绍的内容:动态方法决议。
6、如果动态方法决议仍不能解决问题,只能进行最后一次尝试,进入消息转发流程。
7、如果还不行,去死吧。
下面的图部分展示了这个调用过程:
写到这大家肯定会发出这样的疑问:我仅仅想调用一个方法而已,却不得不经历那么多步骤,效率上怎么保证??苹果也做了一些优化上的工作。
六、函数检索优化措施
主要从下面两个方面着手:
1、通过SEL进行IMP匹配
先来看看类对象中保存的方法列表和方法的数据结构:
typedef struct method_list_t { uint32_t entsize_NEVER_USE; uint32_t count; struct method_t first; } method_list_t; typedef struct method_t { SEL name; const char *types;//参数类型和返回值类型 IMP imp; } method_t;在前一篇文章介绍SEL的时候,我们已经说过了苹果在通过SEL检索IMP时做的努力,这里不再累述。
2、cache缓存
cache的原则就是缓存那些可能要执行的函数地址,那么下次调用的时候,速度就可以快速很多。这个和CPU的各种缓存原理相通。好吧,说了这么多了,再来认识几个名词:
struct objc_cache { uintptr_t mask; uintptr_t occupied; cache_entry *buckets[1]; }; typedef struct { SEL name; void *unused; IMP imp; } cache_entry;看这个结构,有没有搞错又是hash table。
objc_msgSend 首先在cache list 中找SEL,没有找到就在class method中找,super class method中找(当然super class 也有cache list)。而cache的机制则非常复杂了,由于Objective-C是动态语言。所以,这里面还有很多的多线程同步问题,而这些锁又是效率的大敌,相关的内容已经远远超过本文讨论的范围。
如果在缓存中已经有了需要的方法选标,则消息仅仅比函数调用慢一点点。如果程序运行了足够长的时间,几乎每个消息都能在缓存中找到方法实现。程序运行时,缓存也将随着新的消息的增加而增加。据牛人说(没有亲测过),苹果通过这些优化,使消息传递和直接的函数调用效率上的差距已经相当的小。
外面的风吹的太猛了点,北京的天气又是雾霾又是大风,真受不鸟
-------------未完待续-------------