Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

RFC: Per-variable relative tolerance criteria for x #183

Merged
merged 21 commits into from
Apr 25, 2019
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
7e5f41a
xtol_rel: L1 norm for whole-vector stopping criterion
aitap Aug 4, 2018
a92e4f5
vector_norm: fix uninitialized ret, move branch outside for loop, mak…
aitap Aug 18, 2018
02a2ab0
diff_norm: like vector_norm, but for x-oldx
aitap Aug 18, 2018
66f0c11
xtol_abs: test directly when diff_norm tests for weighted xtol_rel
aitap Nov 8, 2018
2108efb
add double * x_weights to nlopt and nlopt_stopping structures
aitap Nov 9, 2018
5ff10d3
don't forget to free the weights; actually declare the weight-related…
aitap Nov 9, 2018
6c6430e
provide scale support for vector_norm and diff_norm
aitap Nov 9, 2018
a0e0040
add argument checks to nlopt_get_x_weights
aitap Apr 23, 2019
038affa
do not allocate x_weights unless requested by
aitap Apr 24, 2019
caf1554
start documenting the new functions
aitap Apr 24, 2019
a65b52a
fixes, add equations
stevengj Apr 24, 2019
c31d1e7
Update NLopt_Reference.md
stevengj Apr 24, 2019
00da0ee
param names
stevengj Apr 24, 2019
91c262b
check for w < 0 in set_x_weights
aitap Apr 25, 2019
98ef03c
add x_weights to C++ interface and document them
aitap Apr 25, 2019
d45c581
add x_weights to F77 interface and document them
aitap Apr 25, 2019
f8141bc
add & document x_weights to SWIG-based interfaces
aitap Apr 25, 2019
64eedcc
add and document x_weights to Octave&Matlab
aitap Apr 25, 2019
6f99d9e
set_x_weights: ISO C90 compatibility
aitap Apr 25, 2019
42ce760
set_x_weights: provide informative error on w <= 0
aitap Apr 25, 2019
5484d29
tweaks
stevengj Apr 25, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 9 additions & 9 deletions doc/docs/NLopt_Reference.md
Original file line number Diff line number Diff line change
Expand Up @@ -239,25 +239,25 @@ nlopt_result nlopt_set_xtol_rel(nlopt_opt opt, double tol);
double nlopt_get_xtol_rel(const nlopt_opt opt);
```


Set relative tolerance on optimization parameters: stop when an optimization step (or an estimate of the optimum) changes every parameter by less than `tol` multiplied by the absolute value of the parameter. (If there is any chance that an optimal parameter is close to zero, you might want to set an absolute tolerance with `nlopt_set_xtol_abs` as well.) Criterion is disabled if `tol` is non-positive.
Set relative tolerance on optimization parameters: stop when an optimization step (or an estimate of the optimum) causes a relative change the parameters $x$ by less than `tol`, i.e. $\Vert \Delta x \Vert_w < \mbox{tol}\cdot\Vert x \Vert_w$ as measured by a weighted L₁ norm $\Vert x \Vert_w = \sum_i w_i |x_i|$, where the weights $w_i$ default to 1.
(If there is any chance that the optimal $\Vert x \Vert$ is close to zero, you might want to set an absolute tolerance with `nlopt_set_xtol_abs` as well.) Criterion is disabled if `tol` is non-positive.

```
nlopt_result nlopt_set_xtol_abs(nlopt_opt opt, const double* tol);
nlopt_result nlopt_get_xtol_abs(const nlopt_opt opt, double *tol);
nlopt_result nlopt_set_x_weights(nlopt_opt opt, const double *w);
nlopt_result nlopt_set_x_weights1(nlopt_opt opt, const double w);
nlopt_result nlopt_get_x_weights(const nlopt_opt opt, double *w);
```


Set absolute tolerances on optimization parameters. `tol` is a pointer to an array of length `n` (the dimension from `nlopt_create`) giving the tolerances: stop when an optimization step (or an estimate of the optimum) changes every parameter `x[i]` by less than `tol[i]`. (Note that this function makes a copy of the `tol` array, so subsequent changes to the caller's `tol` have no effect on `opt`.) In `nlopt_get_xtol_abs`, `tol` must be an array of length `n`, which upon successful return contains a copy of the current tolerances.

For convenience, the following function may be used to set the absolute tolerances in all `n` optimization parameters to the same value:
Set/get the weights used when the computing L₁ norm for the `xtol_rel` stopping criterion above, where `*w` must point to an array of length equal to the number of optimization parameters in `opt`. `nlopt_set_x_weights1` can be used to set all of the weights to the same value `w`. The weights default to `1`, but non-constant weights can be used to handle situations where the different parameters `x` have different units or importance, for example.

```
nlopt_result nlopt_set_xtol_abs(nlopt_opt opt, const double *tol);
nlopt_result nlopt_set_xtol_abs1(nlopt_opt opt, double tol);
nlopt_result nlopt_get_xtol_abs(const nlopt_opt opt, double *tol);
```


Criterion is disabled if `tol` is non-positive.
Set absolute tolerances on optimization parameters. `tol` is a pointer to an array of length `n` (the dimension from `nlopt_create`) giving the tolerances: stop when an optimization step (or an estimate of the optimum) changes every parameter `x[i]` by less than `tol[i]`. (Note that `nlopt_set_xtol_abs` makes a copy of the `tol` array, so subsequent changes to the caller's `tol` have no effect on `opt`.) In `nlopt_get_xtol_abs`, `tol` must be an array of length `n`, which upon successful return contains a copy of the current tolerances. For convenience, the `nlopt_set_xtol_abs1` may be used to set the absolute tolerances in all `n` optimization parameters to the same value. Criterion is disabled if `tol` is non-positive.

```
nlopt_result nlopt_set_maxeval(nlopt_opt opt, int maxeval);
Expand Down
1 change: 1 addition & 0 deletions src/api/nlopt-internal.h
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@ extern "C" {
double stopval; /* stop when f reaches stopval or better */
double ftol_rel, ftol_abs; /* relative/absolute f tolerances */
double xtol_rel, *xtol_abs; /* rel/abs x tolerances */
double *x_weights; /* weights for relative x tolerance */
int maxeval; /* max # evaluations */
int numevals; /* number of evaluations */
double maxtime; /* max time (seconds) */
Expand Down
3 changes: 3 additions & 0 deletions src/api/nlopt.h
Original file line number Diff line number Diff line change
Expand Up @@ -243,6 +243,9 @@ NLOPT_EXTERN(double) nlopt_get_xtol_rel(const nlopt_opt opt);
NLOPT_EXTERN(nlopt_result) nlopt_set_xtol_abs1(nlopt_opt opt, double tol);
NLOPT_EXTERN(nlopt_result) nlopt_set_xtol_abs(nlopt_opt opt, const double *tol);
NLOPT_EXTERN(nlopt_result) nlopt_get_xtol_abs(const nlopt_opt opt, double *tol);
NLOPT_EXTERN(nlopt_result) nlopt_set_x_weights1(nlopt_opt opt, double tol);
NLOPT_EXTERN(nlopt_result) nlopt_set_x_weights(nlopt_opt opt, const double *tol);
NLOPT_EXTERN(nlopt_result) nlopt_get_x_weights(const nlopt_opt opt, double *tol);

NLOPT_EXTERN(nlopt_result) nlopt_set_maxeval(nlopt_opt opt, int maxeval);
NLOPT_EXTERN(int) nlopt_get_maxeval(const nlopt_opt opt);
Expand Down
1 change: 1 addition & 0 deletions src/api/optimize.c
Original file line number Diff line number Diff line change
Expand Up @@ -426,6 +426,7 @@ static nlopt_result nlopt_optimize_(nlopt_opt opt, double *x, double *minf)
stop.ftol_abs = opt->ftol_abs;
stop.xtol_rel = opt->xtol_rel;
stop.xtol_abs = opt->xtol_abs;
stop.x_weights = opt->x_weights;
opt->numevals = 0;
stop.nevals_p = &(opt->numevals);
stop.maxeval = opt->maxeval;
Expand Down
60 changes: 59 additions & 1 deletion src/api/options.c
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ void NLOPT_STDCALL nlopt_destroy(nlopt_opt opt)
free(opt->lb);
free(opt->ub);
free(opt->xtol_abs);
free(opt->x_weights);
free(opt->fc);
free(opt->h);
nlopt_destroy(opt->local_opt);
Expand Down Expand Up @@ -88,6 +89,7 @@ nlopt_opt NLOPT_STDCALL nlopt_create(nlopt_algorithm algorithm, unsigned n)
opt->stopval = -HUGE_VAL;
opt->ftol_rel = opt->ftol_abs = 0;
opt->xtol_rel = 0;
opt->x_weights = NULL;
opt->xtol_abs = NULL;
opt->maxeval = 0;
opt->numevals = 0;
Expand Down Expand Up @@ -133,7 +135,7 @@ nlopt_opt NLOPT_STDCALL nlopt_copy(const nlopt_opt opt)
nlopt_munge munge;
nopt = (nlopt_opt) malloc(sizeof(struct nlopt_opt_s));
*nopt = *opt;
nopt->lb = nopt->ub = nopt->xtol_abs = NULL;
nopt->lb = nopt->ub = nopt->xtol_abs = nopt->x_weights = NULL;
nopt->fc = nopt->h = NULL;
nopt->m_alloc = nopt->p_alloc = 0;
nopt->local_opt = NULL;
Expand All @@ -157,6 +159,12 @@ nlopt_opt NLOPT_STDCALL nlopt_copy(const nlopt_opt opt)
nopt->xtol_abs = (double *) malloc(sizeof(double) * (opt->n));
if (!opt->xtol_abs)
goto oom;
if (opt->x_weights) {
nopt->x_weights = (double *) malloc(sizeof(double) * (opt->n));
if (!opt->x_weights)
goto oom;
memcpy(nopt->x_weights, opt->x_weights, sizeof(double) * (opt->n));
}

memcpy(nopt->lb, opt->lb, sizeof(double) * (opt->n));
memcpy(nopt->ub, opt->ub, sizeof(double) * (opt->n));
Expand Down Expand Up @@ -628,6 +636,56 @@ nlopt_result NLOPT_STDCALL nlopt_get_xtol_abs(const nlopt_opt opt, double *xtol_
return NLOPT_SUCCESS;
}

nlopt_result NLOPT_STDCALL nlopt_set_x_weights(nlopt_opt opt, const double *x_weights)
{
if (opt) {
nlopt_unset_errmsg(opt);
if (!opt->x_weights) {
opt->x_weights = (double *) calloc(opt->n, sizeof(double));
if (!opt->x_weights) {
return NLOPT_OUT_OF_MEMORY;
}
}
memcpy(opt->x_weights, x_weights, opt->n * sizeof(double));
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It should probably check that all of the weights are > 0 or return an INVALID_ARGS

return NLOPT_SUCCESS;
}
return NLOPT_INVALID_ARGS;
}

nlopt_result NLOPT_STDCALL nlopt_set_x_weights1(nlopt_opt opt, double x_weights)
{
if (opt) {
unsigned i;
nlopt_unset_errmsg(opt);
if (!opt->x_weights) {
opt->x_weights = (double *) calloc(opt->n, sizeof(double));
if (!opt->x_weights) {
return NLOPT_OUT_OF_MEMORY;
}
}
for (i = 0; i < opt->n; ++i)
opt->x_weights[i] = x_weights;
return NLOPT_SUCCESS;
}
return NLOPT_INVALID_ARGS;
}

nlopt_result NLOPT_STDCALL nlopt_get_x_weights(const nlopt_opt opt, double *x_weights)
{
nlopt_unset_errmsg(opt);
if (opt && (opt->n == 0 || x_weights)) {
if (opt->x_weights) {
memcpy(x_weights, opt->x_weights, sizeof(double) * (opt->n));
} else {
unsigned i;
for (i = 0; i < opt->n; ++i)
x_weights[i] = 1;
}
return NLOPT_SUCCESS;
}
return NLOPT_INVALID_ARGS;
}

GETSET(maxeval, int, maxeval)

GET(numevals, int, numevals)
Expand Down
1 change: 1 addition & 0 deletions src/util/nlopt-util.h
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@ extern "C" {
double ftol_abs;
double xtol_rel;
const double *xtol_abs;
const double *x_weights;
int *nevals_p, maxeval;
double maxtime, start;
int *force_stop;
Expand Down
66 changes: 58 additions & 8 deletions src/util/stop.c
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,55 @@

/* utility routines to implement the various stopping criteria */

static double sc(double x, double smin, double smax)
{
return smin + x * (smax - smin);
}

static double vector_norm(unsigned n, const double *vec, const double *w, const double *scale_min, const double *scale_max)
{
unsigned i;
double ret = 0;
if (scale_min && scale_max) {
if (w)
for (i = 0; i < n; i++)
ret += w[i] * fabs(sc(vec[i], scale_min[i], scale_max[i]));
else
for (i = 0; i < n; i++)
ret += fabs(sc(vec[i], scale_min[i], scale_max[i]));
} else {
if (w)
for (i = 0; i < n; i++)
ret += w[i] * fabs(vec[i]);
else
for (i = 0; i < n; i++)
ret += fabs(vec[i]);
}
return ret;
}

static double diff_norm(unsigned n, const double *x, const double *oldx, const double *w, const double *scale_min, const double *scale_max)
{
unsigned i;
double ret = 0;
if (scale_min && scale_max) {
if (w)
for (i = 0; i < n; i++)
ret += w[i] * fabs(sc(x[i], scale_min[i], scale_max[i]) - sc(oldx[i], scale_min[i], scale_max[i]));
else
for (i = 0; i < n; i++)
ret += fabs(sc(x[i], scale_min[i], scale_max[i]) - sc(oldx[i], scale_min[i], scale_max[i]));
} else {
if (w)
for (i = 0; i < n; i++)
ret += w[i] * fabs(x[i] - oldx[i]);
else
for (i = 0; i < n; i++)
ret += fabs(x[i] - oldx[i]);
}
return ret;
}

static int relstop(double vold, double vnew, double reltol, double abstol)
{
if (nlopt_isinf(vold))
Expand All @@ -49,33 +98,34 @@ int nlopt_stop_f(const nlopt_stopping * s, double f, double oldf)
int nlopt_stop_x(const nlopt_stopping * s, const double *x, const double *oldx)
{
unsigned i;
if (diff_norm(s->n, x, oldx, s->x_weights, NULL, NULL) <= s->xtol_rel * vector_norm(s->n, x, s->x_weights, NULL, NULL))
return 1;
for (i = 0; i < s->n; ++i)
if (!relstop(oldx[i], x[i], s->xtol_rel, s->xtol_abs[i]))
if (fabs(x[i] - oldx[i]) > s->xtol_abs[i])
return 0;
stevengj marked this conversation as resolved.
Show resolved Hide resolved
return 1;
}

int nlopt_stop_dx(const nlopt_stopping * s, const double *x, const double *dx)
{
unsigned i;
if (vector_norm(s->n, dx, s->x_weights, NULL, NULL) <= s->xtol_rel * vector_norm(s->n, x, s->x_weights, NULL, NULL))
return 1;
for (i = 0; i < s->n; ++i)
if (!relstop(x[i] - dx[i], x[i], s->xtol_rel, s->xtol_abs[i]))
if (fabs(dx[i]) > s->xtol_abs[i])
return 0;
return 1;
}

static double sc(double x, double smin, double smax)
{
return smin + x * (smax - smin);
}

/* some of the algorithms rescale x to a unit hypercube, so we need to
scale back before we can compare to the tolerances */
int nlopt_stop_xs(const nlopt_stopping * s, const double *xs, const double *oldxs, const double *scale_min, const double *scale_max)
{
unsigned i;
if (diff_norm(s->n, xs, oldxs, s->x_weights, scale_min, scale_max) <= s->xtol_rel * vector_norm(s->n, xs, s->x_weights, scale_min, scale_max))
return 1;
for (i = 0; i < s->n; ++i)
if (!relstop(sc(oldxs[i], scale_min[i], scale_max[i]), sc(xs[i], scale_min[i], scale_max[i]), s->xtol_rel, s->xtol_abs[i]))
if (fabs(sc(xs[i], scale_min[i], scale_max[i]) - sc(oldxs[i], scale_min[i], scale_max[i])) > s->xtol_abs[i])
return 0;
return 1;
}
Expand Down