Skip to content

Machine Learning project about badly scanned digits recognition

Notifications You must be signed in to change notification settings

dario-marvin/MachineLearningDigits

Repository files navigation

MachineLearningDigits

Introduction

The idea of this project is to try to solve a simulated real life problem with the help of Machine Learning (ML) techniques and a dataset created ad hoc: suppose you have to digitalise and extract information from some documents, and a high quality scanner is not available, or high quality scans are too expensive, in terms of scanning time or storage memory. Is it then still possible to recover the information contained on the (bad) scans?

In short, we want to build a homemade Optical Character Recognition (OCR) machine using techniques from supervised ML. For the moment, we only try to recover single digits, but an upgrade using a similar approach for recognizing letters and words is easily generalizable and will be implemented in the near future.
To verify whether our idea is feasible, we create a dataset using a low quality scan of a sheet of paper containing a long sequence of digits. Each number on the sheet is individually extracted using techniques from image processing and the list of its pixel values is saved. The data are then analyzed with different ML classifiers and the results are plotted and commented.

Methodology

Document creation

To create our dataset we generated a random sequence of 4290 single digits, which can be observed in the file sequence.dat. A PDF file containing this sequence was then compiled using LaTeX and the resulting file was printed in draft quality, scanned at lowest quality setting (75 dpi) and finally saved as a PNG image in the file page1.png.
In the next figures you can see the upper part of the resulting PNG, and a magnified detail of the upper left corner.

Dataset extraction

Of course, it is unthinkable to manually save the pixels of every single digit in the sequence, so we used image processing instead. As a first step, we compute the mean of the pixel values for every row of pixels composing the image. If there are only clear pixels, i.e. it is not a row containing numbers, the mean of the row will be relatively high (remember white RBG value is 255 and black is 0). If instead we cross some darker pixels, the mean will be lower.
Thus we select all rows with mean pixel value higher than a fixed threshold and to illustrate how the process works, we color them in white on the previous image.

At this point, the remaining stripes of pixels separated by white lines should contain all our numbers. In a perfect world, all stripes would have the same height, but sadly it's not our case, as it ranges from 7 to 9 pixels. So we decided to set universal height 7 pixels for every number image, and in case a stripe was 8 or 9 pixels tall we ignored the uppermost or lowermost row or even both depending on which one has the highest pixel mean value (i.e. the least information).

To capture the digit images on the stripes, the idea is similar to the proceeding method: for each column in the stripe we compute the mean of the pixel values. We start with the leftmost column and search until we obtain a mean value smaller than a fixed threshold, meaning that the column contains some dark pixels, hence a number. From there we select the previous column (even if it has only clear pixels) and the following 5, as that's the width in pixel of the average number. From there we start the search again until we find another number or we reach the end of the stripe.

In the end, each image of a digit will be composed of 6 pixels in width and 7 in height. We show here the images extracted for the first 8 digits in the sequence and their real value.

Data analysis

Since it is well known there is not a perfect general learning algorithm for every problem, we explore many different approaches and select the best one for this problem according to the results. We try some of the most common methods for ML classification, including:

  • Decision tree learning (DT)
  • k-nearest neighbors algorithm (KNN)
  • Linear discriminant analysis (LDA)
  • Gaussian naive Bayes (GNB)
  • Support Vector Machines (SVM)

For each of these approaches we train a classifier using the first 3290 images, which will compose our Training Set, then we ask the model to make a prediction on the value of the remaining 1000 images, which will be our Test Set. We finally compare the predictions with the real values.

Results

Accuracy of Decision Tree classifier on training set: 1.0
Accuracy of Decision Tree classifier on test set: 0.929

Accuracy of K-NN classifier on training set: 0.997872340426
Accuracy of K-NN classifier on test set: 0.997

/usr/local/lib/python3.5/dist-packages/sklearn/discriminant_analysis.py:442: UserWarning: The priors do not sum to 1. Renormalizing
  UserWarning)
Accuracy of LDA classifier on training set: 0.982978723404
Accuracy of LDA classifier on test set: 0.979

Accuracy of GNB classifier on training set: 0.953191489362
Accuracy of GNB classifier on test set: 0.934

Accuracy of SVM classifier on training set: 0.998784194529
Accuracy of SVM classifier on test set: 0.98

The results of the classification show some relatively good values, around 95% of success, although it should be noted that in the LDA analysis we receive a warning saying the priors do not sum to one and thus will be renormalized.

The method that performs best in the Test Set is the k-nearest neighbors algorithm, with a score of 0.997, which means only 3 images were misclassified over the 1000 analyzed.
For this classifier we print classification report and confusion matrix.

             precision    recall  f1-score   support

          0       1.00      1.00      1.00        85
          1       0.98      1.00      0.99        98
          2       0.99      1.00      0.99        99
          3       1.00      1.00      1.00        95
          4       1.00      0.98      0.99       115
          5       1.00      1.00      1.00       101
          6       1.00      1.00      1.00        95
          7       1.00      0.99      1.00       113
          8       1.00      1.00      1.00        90
          9       1.00      1.00      1.00       109

avg / total       1.00      1.00      1.00      1000


[[ 85   0   0   0   0   0   0   0   0   0]
 [  0  98   0   0   0   0   0   0   0   0]
 [  0   0  99   0   0   0   0   0   0   0]
 [  0   0   0  95   0   0   0   0   0   0]
 [  0   2   0   0 113   0   0   0   0   0]
 [  0   0   0   0   0 101   0   0   0   0]
 [  0   0   0   0   0   0  95   0   0   0]
 [  0   0   1   0   0   0   0 112   0   0]
 [  0   0   0   0   0   0   0   0  90   0]
 [  0   0   0   0   0   0   0   0   0 109]]

From the confusion matrix we understand that two images whose real value was 4 were wrongly classified as 1 instead, and one 7 was classified as 2. To understand how the system got confused, we plot the images in question, together with their real and predicted values.

Because the image are (really) badly scanned, the central part in both the two 4 is shifted one pixel left and thus confused as the straight part of a 1. Similarly, the horizontal line of the considered 7 is shifted one pixel down, hence the confusion with the upper part of a 2. Luckily these seem to be isolated cases of particularly poor scanning performance.

Further experiments

Curse of dimensionality

In the previous section we used 3290 images to train the predictors, but what would have happened if we had had a much smaller set to train with? What is a "good" Training Set size? To answer these questions, we feed the classifier a gradually larger amount of images and report the scores each time. To be consistent, we test the predictors always on the same Test Set, the last 1000 observations in the sequence.
We remind that as a general rule of thumb, the "curse of dimensionality" says the minimum number of observations should be the 5 times the dimensionality of the images, in our case 210 observations, so we also plot a black vertical line corresponding to that value.

We can see from the graph that the predictors indeed start the regolarize somewhere around the indicated threshold of 210 observations, however they keep improving until around a Training Set containing at least 500 observations, after which the score is basically constant.

Conclusion and future works

We showed it is possible to retrieve the correct values of some badly scanned digits with a high precision of 99.7%. The classifier that seems to work best for this problem is the k-nearest neighbors algorithm.
In the near future we plan to continue using this approach on more and more complicated data, such as single badly scanned letters of the alphabet, and later complete words extracted from a real scanned book.

How to compile and run the codes

Make sure to have downloaded the files page1.png and sequence.dat in the same folder along with the two python source code files.
Open a terminal and navigate to your folder with the command cd, then run the commands

python3 digits_recognizer.py

and

python3 experiments.py

About

Machine Learning project about badly scanned digits recognition

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages