开发者

Algorithm to loop over an array from the middle outwards?

开发者 https://www.devze.com 2023-03-23 16:28 出处:网络
I\'m working on a divide-and-conquer algorithm (in fact, one 开发者_StackOverflowthat does curve fitting to a number of input points). For the \'divide\' part, I need to calculate an error term for ea

I'm working on a divide-and-conquer algorithm (in fact, one 开发者_StackOverflowthat does curve fitting to a number of input points). For the 'divide' part, I need to calculate an error term for each point, and if the error exceeds a given threshold, I wish to split the curve at that point and process the left and right sections of the input separately. A simple loop does the trick; but it would be advantageous for me to start at the middle of the current section and work outwards. (To clarify: if I do find a point whose error is too large, I recursively call and generate separate curves for the left and right sections - if all the points are within the threshold, then my curve fits and I return).

After a bit of head-scratching, I came up with this (the points are in an array, and the current section is from startIndex to endIndex inclusive):

int steps = (endIndex+1-startIndex);
int i = (startIndex+endIndex)>>1;
int stepdir = 1;
for(int q=0; q<steps; q++, i+=stepdir*q, stepdir=-stepdir)
{
   // test point i here and return early if error exceeds threshold
}

In other words, beginning near the middle, go one index forward, two back, three forward, four back... It works, and I'm sure it's efficient, but it strikes me that there should be a cleaner way to do it, in particular, I ended up having to check the Java language spec to make sure that the statements in the for update expression did evaluate in sequence (even though , is not a sequence operator as it is in C/C++).

Any ideas gratefully appreciated. Is there a cleaner way?


This would be more readable imho

for (int q=0; q < steps; q++) {
   int index = i + ( q% 2 == 0 ? q/2 : -(q/2+1)); //index lookup here
}


If your excess-error checker is simple (e.g. a function call), the clearest thing is to write :

int mid = npoints / 2;
for (int i = 0; i <= mid; i++) {
     if( excess_error(mid + i + 1) ) {
          // divide at mid + i + 1
     } else if excess_error(mid - i) {
          // divide at mid - i
     }
}

Again the "divide at xyz" code should be a function call, or you get cut & pasted code.

(I haven't thought carefully about corner cases, and off-by-one errors, so be careful when i==mid, but you get the picture.)


Here's a more general solution for anyone that needs to search outwards from an arbitrary point (in this example cell [6] in an array of length 7).

int arraySize = 7;
int start = 6;

for (int i=0; i < arraySize; i++) {
   int index = (start+((i%2==0)?i/2:arraySize-(i+1)/2))%arraySize;
   print(index+",");
}
exit();

Prints 6,5,0,4,1,3,2,

0

精彩评论

暂无评论...
验证码 换一张
取 消