A reasonably fast and accurate missing value imputation by iterative PCA.
pip install pca-impute
>>> import pca_impute
>>> X = np.array([[1, 2, 3], [2, 3, None], [3, None, 6], [None, 6, None]], dtype=float)
>>> pca_impute.impute(X, n_components=1)
array([[1. , 2. , 3. ],
[2. , 3. , 4.25710236],
[3. , 5.77464251, 6. ],
[2.9853071 , 6. , 5.99766182]])
>>>
The PCA-based missing value imputer is known to be reasonably fast and accurate, but scikit-learn has only slower and/or less advanced imputers (in my personal opinion). So here is an implementation of the PCA-based imputer!
- Initialize the estimate of the missing values by the mean. Then iterate 2-5 to refine the estimate.
- Fill the missing values of original data with the estimate.
- Fit a PCA to the filled data.
- Reconstruct the filled data by the PCA. (i.e. project the data to the subspace found by the PCA.)
- Update the estimate of the missing values by taking the corresponding entries from the reconstruction.
If I'm not mistaken, this algorithm solves
This is because the above algorithm corresponds to the (block) coordinate descent of the following equivalent problem:
According to my benchmark with synthetic data from a probabilistic PCA model (with missing at random assumption), pca-impute had better mean squared error (MSE) for predicting artificially dropped values, compared to other imputation algorithms available in scikit-learn.
Additionally, the execution time of the algorithm was about 3 times faster than a similar implementation of the iterative PCA (iterative SVD) algorithm in fancyimpute and our implementation had slightly better MSE.
The benchmark script is available at example/benchmark.py
, and the following is the result I got on my laptop:
$ python example/benchmark.py
Mean imputation: 0.027649641036987305 sec
MSE: 4.660934114515747
pca_impute.impute: 0.4510965347290039 sec
MSE: 0.4987599067216591
fancyimpute.IterativeSVD: 1.6951684951782227 sec
MSE: 0.7397661208001316
sklearn.impute.IterativeImputer: 87.45235776901245 sec
MSE: 0.557537819762688
sklearn.impute.KNNImputer: 51.810137033462524 sec
MSE: 1.280416815792845