1.8 Ordered logistic and probit regression
Ordered regression for an outcome yn∈{1,…,k} with predictors xn∈RD is determined by a single coefficient vector β∈RD along with a sequence of cutpoints c∈RK−1 sorted so that cd<cd+1. The discrete output is k if the linear predictor xnβ falls between ck−1 and ck, assuming c0=−∞ and cK=∞. The noise term is fixed by the form of regression, with examples for ordered logistic and ordered probit models.
Ordered logistic regression
The ordered logistic model can be coded in Stan using the
ordered
data type for the cutpoints and the built-in
ordered_logistic
distribution.
data {
int<lower=2> K;
int<lower=0> N;
int<lower=1> D;
array[N] int<lower=1, upper=K> y;
array[N] row_vector[D] x;
}parameters {
vector[D] beta;
ordered[K - 1] c;
}model {
for (n in 1:N) {
y[n] ~ ordered_logistic(x[n] * beta, c);
} }
The vector of cutpoints c
is declared as ordered[K - 1]
,
which guarantees that c[k]
is less than c[k + 1]
.
If the cutpoints were assigned independent priors, the constraint effectively truncates the joint prior to support over points that satisfy the ordering constraint. Luckily, Stan does not need to compute the effect of the constraint on the normalizing term because the probability is needed only up to a proportion.
Ordered probit
An ordered probit model could be coded in exactly the same way by
swapping the cumulative logistic (inv_logit
) for the cumulative
normal (Phi
).
data {
int<lower=2> K;
int<lower=0> N;
int<lower=1> D;
array[N] int<lower=1, upper=K> y;
array[N] row_vector[D] x;
}parameters {
vector[D] beta;
ordered[K - 1] c;
}model {
vector[K] theta;
for (n in 1:N) {
real eta;
eta = x[n] * beta;1] = 1 - Phi(eta - c[1]);
theta[for (k in 2:(K - 1)) {
1]) - Phi(eta - c[k]);
theta[k] = Phi(eta - c[k -
}1]);
theta[K] = Phi(eta - c[K -
y[n] ~ categorical(theta);
} }
The logistic model could also be coded this way by replacing
Phi
with inv_logit
, though the built-in encoding based
on the softmax transform is more efficient and more numerically
stable. A small efficiency gain could be achieved by computing the
values Phi(eta - c[k])
once and storing them for re-use.