My previous batch EM approaches can be considered as maximizing the auxiliary function F(α,β,γ,q)=EZ∼q[logL(D|α,β,γ,Z)]+logP(α,β,γ)+EZ∼q[logP(Z)]+H(q), where α are worker-indexed parameters, β are item-indexed parameters, γ are global parameters, q is the joint distribution over all unobserved labels, Z is the set of all unobserved labels, D is the data set of item-worker-label triples, logL(D|α,β,γ,Z) is the log-likelihood of the data set, P(α,β,γ) is the prior distribution over the generative model parameters, P(Z) is the prior unobserved label distribution, and H(q) is the entropy of the unobserved label distribution.
The unobserved label distribution is assumed to factor over items, q(Z)=∏iqi(Zi), as is the prior distribution P(Z)=∏iPi(Zi). Alternatively only a constrained maximum of the auxiliary function is found, subject to this constraint. The data likelihood is assumed independent conditioned upon (α,β,Z), leading to F(α,β,γ,q)=∑iEZi∼qi[logL(Di|α,βi,γ,Zi)]+logP(α,β,γ)+∑iEZi∼qi[logPi(Zi)]+∑qiH(qi), where i indexes items and Di is the set of data associated with item i. Further assuming the prior distribution is of the form P(α,β,γ)=P(α,γ)∏iP(βi) and rearranging yields F(α,β,γ,q)=∑iFi(α,βi,γ,qi),Fi(α,βi,γ,qi)=EZi∼qi[logL(Di|α,βi,γ,Zi)]+1|I|logP(α,γ)+logP(βi)+EZi∼qi[logPi(Zi)]+H(qi), where |I| is the total number of items. Now the objective function looks like a sum of terms where βi and qi only appear once. This indicates that, if the data were streamed in blocks corresponding to the same item and the optimal α and γ were already known, the βi and qi could be individually maximized and discarded. Of course, the optimal α and γ are not known, but hopefully over time as more data is encountered the estimates get increasingly good. That suggests the following procedure:
- Receive a block Di of item-worker-label triples corresponding to a single item.
- Maximize Fi(α,βi,γ,qi) with respect to βi and qi.
- Basically I run EM on this block of data with fixed α and γ.
- Set α←α+ηt∇αFi|α,β∗i,γ,q∗i and γ←γ+ηt∇γFi|α,β∗i,γ,q∗i.
- ηt is a learning which decays over time, e.g., ηt=η0(τ0+t)−ρ.
- η0≥0, τ0≥0 and ρ≥0 are tuning parameters for the learning algorithm.
- Effectively |I| is also a tuning parameter which sets the relative importance of the prior.
- If desired (e.g., ``inference mode''), output β∗i and q∗i.
- Discard β∗i and q∗i.
- Return to (1).
Scalability with respect to the number of workers is a potential issue. This is because α is maintained as state, and it is indexed by worker (e.g., in nominallabelextract, αw is the confusion matrix for worker w). To overcome this I use the hashing trick: I have a fixed finite number of α parameters and I hash the worker id to get the α for that worker. When I get a hash collision this means I treat two (or more) workers as equivalent, but it allows me to bound the space usage of the algorithm up front. In practice doing hashing tricks like this always seems to work out fabulously. In this particular context, in the limit of a very large number of workers I will model every worker with the population confusion matrix. This is a graceful way to degrade as the sample complexity overwhelms the (fixed) model complexity. (I don't actually anticipate having a large number of workers; the way crowdsourcing seems to go is, one does some small tasks to identify high quality workers and then a larger version of the task restricted to those workers).
Here's an example run involving 40 passes over a small test dataset.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 | % time ~ /src/nincompoop/nominalonlineextract/src/nominalonlineextract --initial_t 10000 --n_items 9859 --n_labels 5 --priorz 1,1,1,1,1 --model flass --data <(. /multicat 40 =( sort -R ethnicity4.noe. in )) --eta 1 --rho 0.5 initial_t = 10000 eta = 1.000000 rho = 0.500000 n_items = 9859 n_labels = 5 n_workers = 65536 symmetric = false test_only = false prediction file = (no output) priorz = 0.199987,0.199987,0.199987,0.199987,0.199987 cumul since example current current current avg q last counter label predict ratings -1.183628 -1.183628 2 -1 0 5 -1.125888 -1.092893 5 -1 0 5 -1.145204 -1.162910 10 -1 0 5 -1.081261 -1.009520 19 0 0 5 -1.124367 -1.173712 36 -1 3 3 -1.083097 -1.039129 69 -1 0 4 -1.037481 -0.988452 134 -1 1 2 -0.929367 -0.820539 263 -1 1 5 -0.820125 -0.709057 520 -1 4 5 -0.738361 -0.653392 1033 -1 1 4 -0.658806 -0.579719 2058 -1 1 5 -0.610473 -0.562028 4107 -1 4 5 -0.566530 -0.522431 8204 -1 0 3 -0.522385 -0.478110 16397 -1 2 4 -0.487094 -0.451771 32782 -1 0 3 -0.460216 -0.433323 65551 -1 4 5 -0.441042 -0.421860 131088 -1 2 5 -0.427205 -0.413365 262161 -1 0 5 -0.420944 -0.408528 394360 -1 1 -1 ~ /src/nincompoop/nominalonlineextract/src/nominalonlineextract --initial_t 85.77s user 0.22s system 99% cpu 1:26.41 total |
nominallabelextract, the implementation of the batch EM analog to the above, converges in about 90 seconds on this dataset and so the run times are a dead heat. For larger datasets, there is less need to do so many passes over the dataset so I would expect the online version to become increasingly advantageous. Furthermore I've been improving the performance of nominallabelextract for several months whereas I just wrote nominalonlineextract so there might be additional speed improvements in the latter. Nonetheless it appears for datasets that fit into memory batch EM is competitive.
nominalonlineextract is available from the nincompoop code repository on Google code. I'll be putting together online versions of the other algorithms in the near-term (the basic approach holds for all of them, but there are different tricks for each specific likelihood).