mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 16:18:28 -05:00
Simplified geoAppendIfWithinShape() and removed spurious calls do sdsdup and sdsfree (#11522)
In scenarios in which we have large datasets and the elements are not contained within the range we do spurious calls do sdsdup and sdsfree. I.e. instead of pre-creating an sds before we know if we're gonna use it or not, change the role of geoAppendIfWithinShape to just do geoWithinShape, and let the caller create the string only when needed. Co-authored-by: Oran Agra <oran@redislabs.com>
This commit is contained in:
parent
cb7447b387
commit
376b689b03
68
src/geo.c
68
src/geo.c
@ -60,14 +60,20 @@ geoArray *geoArrayCreate(void) {
|
||||
return ga;
|
||||
}
|
||||
|
||||
/* Add a new entry and return its pointer so that the caller can populate
|
||||
* it with data. */
|
||||
geoPoint *geoArrayAppend(geoArray *ga) {
|
||||
/* Add and populate with data a new entry to the geoArray. */
|
||||
geoPoint *geoArrayAppend(geoArray *ga, double *xy, double dist,
|
||||
double score, char *member)
|
||||
{
|
||||
if (ga->used == ga->buckets) {
|
||||
ga->buckets = (ga->buckets == 0) ? 8 : ga->buckets*2;
|
||||
ga->array = zrealloc(ga->array,sizeof(geoPoint)*ga->buckets);
|
||||
}
|
||||
geoPoint *gp = ga->array+ga->used;
|
||||
gp->longitude = xy[0];
|
||||
gp->latitude = xy[1];
|
||||
gp->dist = dist;
|
||||
gp->member = member;
|
||||
gp->score = score;
|
||||
ga->used++;
|
||||
return gp;
|
||||
}
|
||||
@ -210,33 +216,33 @@ void addReplyDoubleDistance(client *c, double d) {
|
||||
}
|
||||
|
||||
/* Helper function for geoGetPointsInRange(): given a sorted set score
|
||||
* representing a point, and a GeoShape, appends this entry as a geoPoint
|
||||
* into the specified geoArray only if the point is within the search area.
|
||||
* representing a point, and a GeoShape, checks if the point is within the search area.
|
||||
*
|
||||
* returns C_OK if the point is included, or C_ERR if it is outside. */
|
||||
int geoAppendIfWithinShape(geoArray *ga, GeoShape *shape, double score, sds member) {
|
||||
double distance = 0, xy[2];
|
||||
|
||||
* shape: the rectangle
|
||||
* score: the encoded version of lat,long
|
||||
* xy: output variable, the decoded lat,long
|
||||
* distance: output variable, the distance between the center of the shape and the point
|
||||
*
|
||||
* Return values:
|
||||
*
|
||||
* The return value is C_OK if the point is within search area, or C_ERR if it is outside.
|
||||
* "*xy" is populated with the decoded lat,long.
|
||||
* "*distance" is populated with the distance between the center of the shape and the point.
|
||||
*/
|
||||
int geoWithinShape(GeoShape *shape, double score, double *xy, double *distance) {
|
||||
if (!decodeGeohash(score,xy)) return C_ERR; /* Can't decode. */
|
||||
/* Note that geohashGetDistanceIfInRadiusWGS84() takes arguments in
|
||||
* reverse order: longitude first, latitude later. */
|
||||
if (shape->type == CIRCULAR_TYPE) {
|
||||
if (!geohashGetDistanceIfInRadiusWGS84(shape->xy[0], shape->xy[1], xy[0], xy[1],
|
||||
shape->t.radius*shape->conversion, &distance)) return C_ERR;
|
||||
shape->t.radius*shape->conversion, distance))
|
||||
return C_ERR;
|
||||
} else if (shape->type == RECTANGLE_TYPE) {
|
||||
if (!geohashGetDistanceIfInRectangle(shape->t.r.width * shape->conversion,
|
||||
shape->t.r.height * shape->conversion,
|
||||
shape->xy[0], shape->xy[1], xy[0], xy[1], &distance))
|
||||
shape->xy[0], shape->xy[1], xy[0], xy[1], distance))
|
||||
return C_ERR;
|
||||
}
|
||||
|
||||
/* Append the new element. */
|
||||
geoPoint *gp = geoArrayAppend(ga);
|
||||
gp->longitude = xy[0];
|
||||
gp->latitude = xy[1];
|
||||
gp->dist = distance;
|
||||
gp->member = member;
|
||||
gp->score = score;
|
||||
return C_OK;
|
||||
}
|
||||
|
||||
@ -257,8 +263,6 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
|
||||
/* That's: min <= val < max */
|
||||
zrangespec range = { .min = min, .max = max, .minex = 0, .maxex = 1 };
|
||||
size_t origincount = ga->used;
|
||||
sds member;
|
||||
|
||||
if (zobj->encoding == OBJ_ENCODING_LISTPACK) {
|
||||
unsigned char *zl = zobj->ptr;
|
||||
unsigned char *eptr, *sptr;
|
||||
@ -274,6 +278,8 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
|
||||
|
||||
sptr = lpNext(zl, eptr);
|
||||
while (eptr) {
|
||||
double xy[2];
|
||||
double distance = 0;
|
||||
score = zzlGetScore(sptr);
|
||||
|
||||
/* If we fell out of range, break. */
|
||||
@ -281,10 +287,11 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
|
||||
break;
|
||||
|
||||
vstr = lpGetValue(eptr, &vlen, &vlong);
|
||||
member = (vstr == NULL) ? sdsfromlonglong(vlong) :
|
||||
sdsnewlen(vstr,vlen);
|
||||
if (geoAppendIfWithinShape(ga,shape,score,member)
|
||||
== C_ERR) sdsfree(member);
|
||||
if (geoWithinShape(shape, score, xy, &distance) == C_OK) {
|
||||
/* Append the new element. */
|
||||
char *member = (vstr == NULL) ? sdsfromlonglong(vlong) : sdsnewlen(vstr, vlen);
|
||||
geoArrayAppend(ga, xy, distance, score, member);
|
||||
}
|
||||
if (ga->used && limit && ga->used >= limit) break;
|
||||
zzlNext(zl, &eptr, &sptr);
|
||||
}
|
||||
@ -299,14 +306,15 @@ int geoGetPointsInRange(robj *zobj, double min, double max, GeoShape *shape, geo
|
||||
}
|
||||
|
||||
while (ln) {
|
||||
sds ele = ln->ele;
|
||||
double xy[2];
|
||||
double distance = 0;
|
||||
/* Abort when the node is no longer in range. */
|
||||
if (!zslValueLteMax(ln->score, &range))
|
||||
break;
|
||||
|
||||
ele = sdsdup(ele);
|
||||
if (geoAppendIfWithinShape(ga,shape,ln->score,ele)
|
||||
== C_ERR) sdsfree(ele);
|
||||
if (geoWithinShape(shape, ln->score, xy, &distance) == C_OK) {
|
||||
/* Append the new element. */
|
||||
geoArrayAppend(ga, xy, distance, ln->score, sdsdup(ln->ele));
|
||||
}
|
||||
if (ga->used && limit && ga->used >= limit) break;
|
||||
ln = ln->level[0].forward;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user