Commit 3e9f7adb authored by Davis King's avatar Davis King

Added a parameter to control the search radius to the

find_min_single_variable() and find_max_single_variable() routines.  Also made
these functions a little more robust to unusual objective functions.
parent 0bffe511
...@@ -562,19 +562,24 @@ namespace dlib ...@@ -562,19 +562,24 @@ namespace dlib
const double begin = -1e200, const double begin = -1e200,
const double end = 1e200, const double end = 1e200,
const double eps = 1e-3, const double eps = 1e-3,
const long max_iter = 100 const long max_iter = 100,
const double initial_search_radius = 1
) )
{ {
DLIB_CASSERT( eps > 0 && DLIB_CASSERT( eps > 0 &&
max_iter > 1 && max_iter > 1 &&
begin <= starting_point && starting_point <= end, begin <= starting_point && starting_point <= end &&
initial_search_radius > 0,
"eps: " << eps "eps: " << eps
<< "\n max_iter: "<< max_iter << "\n max_iter: "<< max_iter
<< "\n begin: "<< begin << "\n begin: "<< begin
<< "\n end: "<< end << "\n end: "<< end
<< "\n starting_point: "<< starting_point << "\n starting_point: "<< starting_point
<< "\n initial_search_radius: "<< initial_search_radius
); );
double search_radius = initial_search_radius;
double p1=0, p2=0, p3=0, f1=0, f2=0, f3=0; double p1=0, p2=0, p3=0, f1=0, f2=0, f3=0;
long f_evals = 1; long f_evals = 1;
...@@ -614,7 +619,6 @@ namespace dlib ...@@ -614,7 +619,6 @@ namespace dlib
// Now we have 3 points on the function. Start looking for a bracketing set such that // Now we have 3 points on the function. Start looking for a bracketing set such that
// f1 > f2 < f3 is the case. // f1 > f2 < f3 is the case.
double jump_size = 1;
while ( !(f1 > f2 && f2 < f3)) while ( !(f1 > f2 && f2 < f3))
{ {
// check for hitting max_iter or if the interval is now too small // check for hitting max_iter or if the interval is now too small
...@@ -641,6 +645,24 @@ namespace dlib ...@@ -641,6 +645,24 @@ namespace dlib
starting_point = p3; starting_point = p3;
return f3; return f3;
} }
// If the left most points are identical in function value then expand out the
// left a bit, unless it's already at bound.
if (f1==f2 && f1!=begin)
{
p1 = max(p1 - search_radius, begin);
f1 = f(p1);
search_radius *= 2;
continue;
}
if (f2==f3 && f3!=end)
{
p3 = min(p3 + search_radius, end);
f3 = f(p3);
search_radius *= 2;
continue;
}
// if f1 is small then take a step to the left // if f1 is small then take a step to the left
if (f1 < f3) if (f1 < f3)
...@@ -649,7 +671,7 @@ namespace dlib ...@@ -649,7 +671,7 @@ namespace dlib
// a point between p1 and p2 in the hopes that shrinking the interval will // a point between p1 and p2 in the hopes that shrinking the interval will
// be a good thing to do. Or if p1 and p2 aren't differentiated then try and // be a good thing to do. Or if p1 and p2 aren't differentiated then try and
// get them to obtain different values. // get them to obtain different values.
if (p1 == begin || (f1 == f2 && (end-begin) < jump_size )) if (p1 == begin || (f1 == f2 && (end-begin) < search_radius ))
{ {
p3 = p2; p3 = p2;
f3 = f2; f3 = f2;
...@@ -666,10 +688,10 @@ namespace dlib ...@@ -666,10 +688,10 @@ namespace dlib
p2 = p1; p2 = p1;
f2 = f1; f2 = f1;
p1 = max(p1 - jump_size, begin); p1 = max(p1 - search_radius, begin);
f1 = f(p1); f1 = f(p1);
jump_size *= 2; search_radius *= 2;
} }
} }
...@@ -680,7 +702,7 @@ namespace dlib ...@@ -680,7 +702,7 @@ namespace dlib
// a point between p2 and p3 in the hopes that shrinking the interval will // a point between p2 and p3 in the hopes that shrinking the interval will
// be a good thing to do. Or if p2 and p3 aren't differentiated then try and // be a good thing to do. Or if p2 and p3 aren't differentiated then try and
// get them to obtain different values. // get them to obtain different values.
if (p3 == end || (f2 == f3 && (end-begin) < jump_size)) if (p3 == end || (f2 == f3 && (end-begin) < search_radius))
{ {
p1 = p2; p1 = p2;
f1 = f2; f1 = f2;
...@@ -697,10 +719,10 @@ namespace dlib ...@@ -697,10 +719,10 @@ namespace dlib
p2 = p3; p2 = p3;
f2 = f3; f2 = f3;
p3 = min(p3 + jump_size, end); p3 = min(p3 + search_radius, end);
f3 = f(p3); f3 = f(p3);
jump_size *= 2; search_radius *= 2;
} }
} }
...@@ -837,10 +859,11 @@ namespace dlib ...@@ -837,10 +859,11 @@ namespace dlib
const double begin = -1e200, const double begin = -1e200,
const double end = 1e200, const double end = 1e200,
const double eps = 1e-3, const double eps = 1e-3,
const long max_iter = 100 const long max_iter = 100,
const double initial_search_radius = 1
) )
{ {
return -find_min_single_variable(negate_function(f), starting_point, begin, end, eps, max_iter); return -find_min_single_variable(negate_function(f), starting_point, begin, end, eps, max_iter, initial_search_radius);
} }
// ---------------------------------------------------------------------------------------- // ----------------------------------------------------------------------------------------
......
...@@ -283,7 +283,8 @@ namespace dlib ...@@ -283,7 +283,8 @@ namespace dlib
const double begin = -1e200, const double begin = -1e200,
const double end = 1e200, const double end = 1e200,
const double eps = 1e-3, const double eps = 1e-3,
const long max_iter = 100 const long max_iter = 100,
const double initial_search_radius = 1
) )
/*! /*!
requires requires
...@@ -292,13 +293,17 @@ namespace dlib ...@@ -292,13 +293,17 @@ namespace dlib
- begin <= starting_point <= end - begin <= starting_point <= end
- f must be a function of a double that returns a double - f must be a function of a double that returns a double
(e.g. f(starting_point) should be a valid expression that evaluates to a double) (e.g. f(starting_point) should be a valid expression that evaluates to a double)
- initial_search_radius > 0
ensures ensures
- Finds a point P such that: - Finds a point P such that:
- P is a local minimum of the function f(). - P is a local minimum of the function f().
- begin <= P <= end - begin <= P <= end
- Evaluates f() no more than max_iter times - Evaluates f() no more than max_iter times
- Stops searching when the window around the minimum point is smaller than eps. - Stops searching when the window around the minimum point is smaller than eps.
The search will begin with the given starting_point. The search will begin with the given starting_point and expand out to the
left and right by initial_search_radius sized steps. So if you think the
minimum is likely to be found within X distance from the starting_point then
set initial_search_radius to X.
- #starting_point == P - #starting_point == P
- returns f(P) - returns f(P)
throws throws
...@@ -318,7 +323,8 @@ namespace dlib ...@@ -318,7 +323,8 @@ namespace dlib
const double begin = -1e200, const double begin = -1e200,
const double end = 1e200, const double end = 1e200,
const double eps = 1e-3, const double eps = 1e-3,
const long max_iter = 100 const long max_iter = 100,
const double initial_search_radius = 1
) )
/*! /*!
requires requires
...@@ -327,13 +333,17 @@ namespace dlib ...@@ -327,13 +333,17 @@ namespace dlib
- begin <= starting_point <= end - begin <= starting_point <= end
- f must be a function of a double that returns a double - f must be a function of a double that returns a double
(e.g. f(starting_point) should be a valid expression that evaluates to a double) (e.g. f(starting_point) should be a valid expression that evaluates to a double)
- initial_search_radius > 0
ensures ensures
- Finds a point P such that: - Finds a point P such that:
- P is a local maximum of the function f(). - P is a local maximum of the function f().
- begin <= P <= end - begin <= P <= end
- Evaluates f() no more than max_iter times - Evaluates f() no more than max_iter times
- Stops searching when the window around the minimum point is smaller than eps. - Stops searching when the window around the maximum point is smaller than eps.
The search will begin with the given starting_point. The search will begin with the given starting_point and expand out to the
left and right by initial_search_radius sized steps. So if you think the
maximum is likely to be found within X distance from the starting_point then
set initial_search_radius to X.
- #starting_point == P - #starting_point == P
- returns f(P) - returns f(P)
throws throws
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment