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

DOC improve documentation of nearmiss #1028

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
89 changes: 55 additions & 34 deletions doc/under_sampling.rst
Original file line number Diff line number Diff line change
Expand Up @@ -125,75 +125,96 @@ It would also work with pandas dataframe::
>>> df_resampled, y_resampled = rus.fit_resample(df_adult, y_adult)
>>> df_resampled.head() # doctest: +SKIP

:class:`NearMiss` adds some heuristic rules to select samples
:cite:`mani2003knn`. :class:`NearMiss` implements 3 different types of
heuristic which can be selected with the parameter ``version``::
NearMiss
^^^^^^^^

:class:`NearMiss` is another controlled under-sampling technique. It aims to balance
the class distribution by eliminating samples from the targeted classes. But these
samples are not removed at random. Instead, :class:`NearMiss` removes instances of the
target class(es) that increase the "space" or separation between the target class and
the minority class. In other words, :class:`NearMiss` removes observations from the
target class that are closer to the boundary they form with the minority class samples.

To find out which samples are closer to the boundary with the minority class,
:class:`NearMiss` uses the K-Nearest Neighbours algorithm. :class:`NearMiss` implements
3 different heuristics, which we can select with the parameter ``version`` and we
will explain in the coming paragraphs. We can perform this undersampling as follows::

>>> from imblearn.under_sampling import NearMiss
>>> nm1 = NearMiss(version=1)
>>> X_resampled_nm1, y_resampled = nm1.fit_resample(X, y)
>>> print(sorted(Counter(y_resampled).items()))
[(0, 64), (1, 64), (2, 64)]

As later stated in the next section, :class:`NearMiss` heuristic rules are
Copy link
Contributor Author

Choose a reason for hiding this comment

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

I think that describing parameters without having described the logic breaks the flow and doesn't really help the user understand what the parameters do.

I suggest removing this paragraph altogether and expanding on the meaning of the parameters (if necessary) on the docstrings.

based on nearest neighbors algorithm. Therefore, the parameters ``n_neighbors``
and ``n_neighbors_ver3`` accept classifier derived from ``KNeighborsMixin``
from scikit-learn. The former parameter is used to compute the average distance
to the neighbors while the latter is used for the pre-selection of the samples
of interest.

Mathematical formulation
^^^^^^^^^^^^^^^^^^^^^^^^
~~~~~~~~~~~~~~~~~~~~~~~~

:class:`NearMiss` uses the K-Nearest Neighbours algorithm to identify the samples of the
target class(es) that are closer to the minority class, as well as the distance that
separates them.

Let *positive samples* be the samples belonging to the targeted class to be
under-sampled. *Negative sample* refers to the samples from the minority class
(i.e., the most under-represented class).
Let *positive samples* be the samples from the class to be under-sampled, and
*negative sample* the samples from the minority class (i.e., the most
under-represented class).

NearMiss-1 selects the positive samples for which the average distance
to the :math:`N` closest samples of the negative class is the smallest.
**NearMiss-1** selects the positive samples whose average distance to the :math:`K`
closest samples of the negative class is the smallest (:math:`K` is the number of
neighbours in the K-Nearest Neighbour algorithm). The following image illustrates the
logic:

.. image:: ./auto_examples/under-sampling/images/sphx_glr_plot_illustration_nearmiss_001.png
:target: ./auto_examples/under-sampling/plot_illustration_nearmiss.html
:scale: 60
:align: center

NearMiss-2 selects the positive samples for which the average distance to the
:math:`N` farthest samples of the negative class is the smallest.
**NearMiss-2** selects the positive samples whose average distance to the
:math:`K` farthest samples of the negative class is the smallest. The following image
illustrates the logic:

.. image:: ./auto_examples/under-sampling/images/sphx_glr_plot_illustration_nearmiss_002.png
:target: ./auto_examples/under-sampling/plot_illustration_nearmiss.html
:scale: 60
:align: center

NearMiss-3 is a 2-steps algorithm. First, for each negative sample, their
:math:`M` nearest-neighbors will be kept. Then, the positive samples selected
are the one for which the average distance to the :math:`N` nearest-neighbors
is the largest.
**NearMiss-3** is a 2-steps algorithm:

First, for each negative sample, that is, for each observation of the minority class,
Copy link
Contributor Author

Choose a reason for hiding this comment

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

I refer back to target class and minority because positive and negative are a bit confusing, and at this stage, the reader may have forgotten which is which.

it selects :math:`M` nearest-neighbors from the postivie class (target class). This
ensures that all observations from the minority class have at least some neighbours
from the target class.

Next, it selects positive samples whose average distance to the :math:`K`
nearest-neighbors of the minority class is the largest.

The following image illustrates the logic:

.. image:: ./auto_examples/under-sampling/images/sphx_glr_plot_illustration_nearmiss_003.png
:target: ./auto_examples/under-sampling/plot_illustration_nearmiss.html
:scale: 60
:align: center

In the next example, the different :class:`NearMiss` variant are applied on the
previous toy example. It can be seen that the decision functions obtained in
each case are different.

When under-sampling a specific class, NearMiss-1 can be altered by the presence
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is the paragraph that I don't understand.

Are we saying that Nearmiss-1 is sensitive to noise, because the samples from the majority closer to the minority are noise?

That is what I understood and rephrased it. Correct me if I am wrong.

But besides that, is it always the case that observations from the majority closer to the minority are noise? I get it that they will be the hardest to classify, but that does not necessarily make them noise. Noise, as I understand it, is some sort of random variation. Or in this case, it would be samples from the majority class that are not necessarily representative of the majority, or in other words sort of outliers. But all of these, is not in the text. The text as it stands right now just says that if they are close, they are noise, and I don't think that is correct.

Thoughts?

Copy link
Member

Choose a reason for hiding this comment

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

A noisy observation would be something outside of the sample distribution. So I agree that it does not have anything to do with the decision boundary. Those are just harder to classify if the data overlap but they are not noise.

of noise. In fact, it will implied that samples of the targeted class will be
selected around these samples as it is the case in the illustration below for
the yellow class. However, in the normal case, samples next to the boundaries
will be selected. NearMiss-2 will not have this effect since it does not focus
on the nearest samples but rather on the farthest samples. We can imagine that
the presence of noise can also altered the sampling mainly in the presence of
marginal outliers. NearMiss-3 is probably the version which will be less
affected by noise due to the first step sample selection.
In the following example, we apply the different :class:`NearMiss` variants to a toy
dataset. Note how the decision functions obtained in each case are different (left
plots):

.. image:: ./auto_examples/under-sampling/images/sphx_glr_plot_comparison_under_sampling_003.png
:target: ./auto_examples/under-sampling/plot_comparison_under_sampling.html
:scale: 60
:align: center

NearMiss-1 is sensitive to noise. In fact, we could think that those observations from
the target class that are closer to samples from the minority class are indeed noise.
NearMiss-1 will select however, those observations, as shown in the first row of the
previous illustration (check the yellow class).

NearMiss-2 will be less sensitive to noise since it does not select the nearest, but
rather on the farthest samples of the target class.

NearMiss-3 is probably the least sensitive version to noise due to the first sample
selection step.

Copy link
Contributor Author

@solegalli solegalli Jul 11, 2023

Choose a reason for hiding this comment

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

More thoughts on NearMiss:

These is just my opinion, I would be happy to be challenged on this one.

To me, it feels that the authors almost used some random logic to select the samples: those with the smallest average distance, those with the largest and then something else in version 3. In their article, they don't really describe how they came to this conclusion, so it seems arbitrary to me. Given this, I am not sure it makes sense to discuss anything further on this user guide. I would remove all the discussion about noise in fact.

In addition, this method was designed and tested on text data, something similar to bag of words / tokens, which is a very particular format. And I am not aware that it has been tested / used somewhere else? So overall, I am not a great fan of this method, but that aside, should we not add a disclaimer somewhere saying that it was designed for text, so caution when implementing in more traditional datasets?

Copy link
Member

Choose a reason for hiding this comment

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

Actually we don't even implement the right things here :)
#980


Cleaning under-sampling techniques
----------------------------------

Expand Down
15 changes: 6 additions & 9 deletions imblearn/under_sampling/_prototype_selection/_nearmiss.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,28 +35,25 @@ class NearMiss(BaseUnderSampler):

n_neighbors : int or estimator object, default=3
If ``int``, size of the neighbourhood to consider to compute the
average distance to the minority point samples. If object, an
average distance to the minority samples. If object, an
estimator that inherits from
:class:`~sklearn.neighbors.base.KNeighborsMixin` that will be used to
find the k_neighbors.
By default, it will be a 3-NN.

n_neighbors_ver3 : int or estimator object, default=3
If ``int``, NearMiss-3 algorithm start by a phase of re-sampling. This
parameter correspond to the number of neighbours selected create the
subset in which the selection will be performed. If object, an
estimator that inherits from
Only used if `version=3`. If ``int``, the number of target class samples that
are closest to a minority sample that will be retained in the first subsampling
step. If object, an estimator that inherits from
:class:`~sklearn.neighbors.base.KNeighborsMixin` that will be used to
find the k_neighbors.
By default, it will be a 3-NN.

{n_jobs}

Attributes
----------
sampling_strategy_ : dict
Dictionary containing the information to sample the dataset. The keys
corresponds to the class labels from which to sample and the values
correspond to the class labels from which to sample and the values
are the number of samples to sample.

nn_ : estimator object
Expand Down Expand Up @@ -144,7 +141,7 @@ def __init__(
def _selection_dist_based(
self, X, y, dist_vec, num_samples, key, sel_strategy="nearest"
):
"""Select the appropriate samples depending of the strategy selected.
"""Select the appropriate samples depending on the selected strategy.

Parameters
----------
Expand Down