根据外部点对CGPoints数组进行排序

时间:2013-06-24 16:29:32

标签: objective-c

我有一个CGPoints数组,并希望根据与外部点的最近距离对其进行排序。我已完成以下操作,但返回的结果不正确。

NSArray *sortedArray = [self.availableRenderPoints sortedArrayUsingComparator:^NSComparisonResult(NSValue *obj1, NSValue *obj2)
{
    CGPoint p1 = [obj1 CGPointValue];
    CGPoint p2 = [obj2 CGPointValue];

    CGFloat distance1 = ccpDistance(startPoint, p1);
    CGFloat distance2 = ccpDistance(startPoint, p2);

    if (distance1 <= distance2) return p1.y < p2.y;
    if (distance2 <= distance1) return p2.y < p1.y;

    return p1.x < p2.x;
}];

如何根据距startPoint的最近距离对其进行排序?

2 个答案:

答案 0 :(得分:5)

不应该只是

NSArray *sortedArray = [self.availableRenderPoints sortedArrayUsingComparator:^NSComparisonResult(NSValue *obj1, NSValue *obj2)
{
    CGPoint p1 = [obj1 CGPointValue];
    CGPoint p2 = [obj2 CGPointValue];

    CGFloat distance1 = ccpDistance(startPoint, p1);
    CGFloat distance2 = ccpDistance(startPoint, p2);

    if (distance1 < distance2) return NSOrderedAscending;
    if (distance2 < distance1) return NSOrderedDescending;

    return NSOrderedSame;
}];

答案 1 :(得分:0)

Martin R的回答对我来说很好,但使用ccpDistance意味着采用平方根,这是不必要的。通过比较平方距离可以获得更好的性能:

NSArray *sortedArray = [self.availableRenderPoints sortedArrayUsingComparator:^NSComparisonResult(NSValue *obj1, NSValue *obj2) {
    CGPoint p1 = [obj1 CGPointValue];
    CGPoint p2 = [obj2 CGPointValue];

    CGFloat squaredDistance1 = ccpLengthSQ(ccpSub(p1, startPoint));
    CGFloat squaredDistance2 = ccpLengthSQ(ccpSub(p2, startPoint));

    // Sort by nearness to startPoint.
    // If equally near, sort by Y coordinate.
    // If equal Y coordinate, sort by X coordinate.
    return
        distance1 < distance2 ? NSOrderedAscending
        : distance2 > distance1 ? NSOrderedDescending
        : p1.y < p2.y ? NSOrderedAscending
        : p2.y < p1.y ? NSOrderedDescending
        : p1.x < p2.x ? NSOrderedAscending
        : p2.x < p1.x ? NSOrderedDescending
        : NSOrderedSame;
}];

另一方面,如果您的阵列很小或平方根足够快,避免平方根可能不会产生明显的差异。