Fast color quantization in OpenCV

后端 未结 2 903
隐瞒了意图╮
隐瞒了意图╮ 2020-11-28 11:50

How can I reduce the number of distinct colors in images using OpenCV (+ C++) the fastest way possible? I don\'t want the complete code. I\'m already doing it using kmeans b

相关标签:
2条回答
  • 2020-11-28 12:03

    Fast pairwise nearest neighbor based algorithm with 8 colors
    High quality and fast

    Efficient, Edge-Aware, Combined Color Quantization and Dithering with 8 colors
    Higher quality for 32 or less colors but slower

    Spatial color quantization with 8 colors
    Higher quality for 32 or less colors but the slowest

    Sample c++ code
    For the speed, it might be depending on GPU parallel programming C/C++.

    0 讨论(0)
  • 2020-11-28 12:17

    There are many ways to quantize colors. Here I describe four.

    Uniform quantization

    Here we are using a color map with uniformly distributed colors, whether they exist in the image or not. In MATLAB-speak you would write

    qimg = round(img*(N/255))*(255/N);
    

    to quantize each channel into N levels (assuming the input is in the range [0,255]. You can also use floor, which is more suitable in some cases. This leads to N^3 different colors. For example with N=8 you get 512 unique RGB colors.

    K-means clustering

    This is the "classical" method to generate an adaptive palette. Obviously it is going to be the most expensive. The OP is applying k-means on the collection of all pixels. Instead, k-means can be applied to the color histogram. The process is identical, but instead of 10 million data points (a typical image nowadays), you have only maybe 32^3 = 33 thousand. The quantization caused by the histogram with reduced number of bins has little effect here when dealing with natural photographs. If you are quantizing a graph, which has a limited set of colors, you don't need to do k-means clustering.

    You do a single pass through all pixels to create the histogram. Next, you run the regular k-means clustering, but using the histogram bins. Each data point has a weight now also (the number of pixels within that bin), that you need to take into account. The step in the algorithm that determines the cluster centers is affected. You need to compute the weighted mean of the data points, instead of the regular mean.

    The result is affected by the initialization.

    Octree quantization

    An octree is a data structure for spatial indexing, where the volume is recursively divided into 8 sub-volumes by cutting each axis in half. The tree thus is formed of nodes with 8 children each. For color quantization, the RGB cube is represented by an octree, and the number of pixels per node is counted (this is equivalent to building a color histogram, and constructing an octree on top of that). Next, leaf nodes are removed until the desired number of them is left. Removing leaf nodes happens 8 at a time, such that a node one level up becomes a leaf. There are different strategies to pick which nodes to prune, but they typically revolve around pruning nodes with low pixel counts.

    This is the method that Gimp uses.

    Because the octree always splits nodes down the middle, it is not as flexible as k-means clustering or the next method.

    Minimum variance quantization

    MATLAB's rgb2ind, which the OP mentions, does uniform quantization and something they call "minimum variance quantization":

    Minimum variance quantization cuts the RGB color cube into smaller boxes (not necessarily cubes) of different sizes, depending on how the colors are distributed in the image.

    I'm not sure what this means. This page doesn't give away anything more, but it has a figure that looks like a k-d tree partitioning of the RGB cube. K-d trees are spatial indexing structures that divide spatial data in half recursively. At each level, you pick the dimension where there is most separation, and split along that dimension, leading to one additional leaf node. In contrast to octrees, the splitting can happen at an optimal location, it is not down the middle of the node.

    The advantage of using a spatial indexing structure (either k-d trees or octrees) is that the color lookup is really fast. You start at the root, and make a binary decision based on either R, G or B value, until you reach a leaf node. There is no need to compute distances to each prototype cluster, as is the case of k-means.

    [Edit two weeks later] I have been thinking about a possible implementation, and came up with one. This is the algorithm:

    • The full color histogram is considered a partition. This will be the root for a k-d tree, which right now is also the leaf node because there are yet no other nodes.
    • A priority queue is created. It contains all the leaf nodes of the k-d tree. The priority is given by the variance of the partition along one axis, minus the variances of the two halves if we were to split the partition along that axis. The split location is picked such that the variances of the two halves are minimal (using Otsu's algorithm). That is, the larger the priority, the more total variance we reduce by making the split. For each leaf node, we compute this value for each axis, and use the largest result.
    • We process partitions on the queue until we have the desired number of partitions:
      • We split the partition with highest priority along the axis and at the location computed when determining the priority.
      • We compute the priority for each of the two halves, and put them on the queue.

    This is a relatively simple algorithm when described this way, the code is somewhat more complex, because I tried to make it efficient but generic.

    Comparison

    On a 256x256x256 RGB histogram I got these timings comparing k-means clustering and this new algorithm:

    # clusters    kmeans (s)    minvar (s)
         5          3.98         0.34
        20         17.9          0.48
        50        220.8          0.59
    

    Note that k-means needs more iterations as the number of clusters increases, hence the exponential time increase. Normally one would not use such a big histogram, I wanted to have large data to make the timings more robust.

    Here is an example of these three methods applied to a test image:

    Input:

    Uniform with N=4 leading to up to 64 different colors [with N=2 to get 8 different colors and comparable to the other methods, the result is very ugly]:

    K-means with 8 colors:

    New "minimum variance" with 8 colors:

    I like this last result better than the K-means result, though they are fairly similar.

    0 讨论(0)
提交回复
热议问题