一:数学原理

K-Means算法的作者是MacQueen, 基本的数学原理很容易理解,假设有一个像素

数据集P。我们要根据值不同将它分为两个基本的数据集合Cluster1, Cluster2,使

用K-Means算法大致如下:

假设两个Cluster的RGB值分别为112,225,244和23,34,99则像素集合中的像素点

a(222,212,234), b(198,205,229), c(25,77,52),d(34,55,101)计算每个像素点与这

两个cluster中心点的欧几里德距离,则像素点a, b属于前面一个cluster, c,d属于

后面一个cluster。然后在根据(222+198)/2, (212+205)/2, (234+52)/2更新cluster

的RGB值,对后一个cluster做同样处理。然后再计算每个像素点到cluster中心点

的欧几里德距离。最终值没有变化则得到分类Cluster点集合。

二:算法基本流程

图像处理------K-Means算法演示-LMLPHP

三:算法关键代码解析

初始化cluster中心点代码如下:

  1. Random random = new Random();
  2. for (int i = 0; i < numOfCluster; i++)
  3. {
  4. int randomNumber1 = random.nextInt(width);
  5. int randomNumber2 = random.nextInt(height);
  6. index = randomNumber2 * width + randomNumber1;
  7. ClusterCenter cc = new ClusterCenter(randomNumber1, randomNumber2, inPixels[index]);
  8. cc.setcIndex(i);
  9. clusterCenterList.add(cc);
  10. }

初始化所有像素点代码如下:

  1. // create all cluster point
  2. for (int row = 0; row < height; ++row)
  3. {
  4. for (int col = 0; col < width; ++col)
  5. {
  6. index = row * width + col;
  7. int color = inPixels[index];
  8. pointList.add(new ClusterPoint(row, col, color));
  9. }
  10. }

计算两个像素点之间欧几里德距离的代码如下:

  1. // int pa = (p.getPixelColor() >> 24) & 0xff;
  2. int pr = (p.getPixelColor() >> 16) & 0xff;
  3. int pg = (p.getPixelColor() >> 8) & 0xff;
  4. int pb = p.getPixelColor() & 0xff;
  5. // int ca = (c.getPixelColor() >> 24) & 0xff;
  6. int cr = (c.getPixelColor() >> 16) & 0xff;
  7. int cg = (c.getPixelColor() >> 8) & 0xff;
  8. int cb = c.getPixelColor() & 0xff;
  9. return Math.sqrt(Math.pow((pr - cr), 2.0) + Math.pow((pg - cg), 2.0) + Math.pow((pb - cb), 2.0));

; i<clusterCenterList.size(); i++)

  • {
  • clusterCenterList.get(i).setNumOfPoints(0);
  • }
  • // recalculate the sum and total of points for each cluster
  • double[] redSums = new double[3];
  • double[] greenSum = new double[3];
  • double[] blueSum = new double[3];
  • for(int i=0; i<pointList.size(); i++)
  • {
  • int cIndex = (int)pointList.get(i).getClusterIndex();
  • clusterCenterList.get(cIndex).addPoints();
  • int ta = (pointList.get(i).getPixelColor() >> 24) & 0xff;
  • int tr = (pointList.get(i).getPixelColor() >> 16) & 0xff;
  • int tg = (pointList.get(i).getPixelColor() >> 8) & 0xff;
  • int tb = pointList.get(i).getPixelColor() & 0xff;
  • ta = 255;
  • redSums[cIndex] += tr;
  • greenSum[cIndex] += tg;
  • blueSum[cIndex] += tb;
  • }
  • double[] oldClusterCentersColors = new double[clusterCenterList.size()];
  • for(int i=0; i<clusterCenterList.size(); i++)
  • {
  • double sum  = clusterCenterList.get(i).getNumOfPoints();
  • int cIndex = clusterCenterList.get(i).getcIndex();
  • int red = (int)(greenSum[cIndex]/sum);
  • int green = (int)(greenSum[cIndex]/sum);
  • int blue = (int)(blueSum[cIndex]/sum);
  • System.out.println("red = " + red + " green = " + green + " blue = " + blue);
  • int clusterColor = (255 << 24) | (red << 16) | (green << 8) | blue;
  • clusterCenterList.get(i).setPixelColor(clusterColor);
  • oldClusterCentersColors[i] = clusterColor;
  • }
  • return oldClusterCentersColors;
  • }
  • 四:运行效果

    图像处理------K-Means算法演示-LMLPHP
    五:K-Means算法源代码

    1. package com.gloomyfish.segmentation.kmeans;
    2. import java.awt.image.BufferedImage;
    3. import java.util.ArrayList;
    4. import java.util.List;
    5. import java.util.Random;
    6. import com.gloomyfish.filter.study.AbstractBufferedImageOp;
    7. import com.gloomyfish.segmentation.fuzzycmeans.ClusterPoint;
    8. public class KMeansProcessor extends AbstractBufferedImageOp {
    9. private List<ClusterCenter> clusterCenterList;
    10. private List<ClusterPoint> pointList;
    11. private int numOfCluster;
    12. public KMeansProcessor(int clusters)
    13. {
    14. this.numOfCluster = clusters;
    15. pointList = new ArrayList<ClusterPoint>();
    16. this.clusterCenterList = new ArrayList<ClusterCenter>();
    17. }
    18. @Override
    19. public BufferedImage filter(BufferedImage src, BufferedImage dest) {
    20. // initialization the pixel data
    21. int width = src.getWidth();
    22. int height = src.getHeight();
    23. int[] inPixels = new int[width*height];
    24. src.getRGB( 0, 0, width, height, inPixels, 0, width );
    25. int index = 0;
    26. //Create random points to use a the cluster center
    27. Random random = new Random();
    28. for (int i = 0; i < numOfCluster; i++)
    29. {
    30. int randomNumber1 = random.nextInt(width);
    31. int randomNumber2 = random.nextInt(height);
    32. index = randomNumber2 * width + randomNumber1;
    33. ClusterCenter cc = new ClusterCenter(randomNumber1, randomNumber2, inPixels[index]);
    34. cc.setcIndex(i);
    35. clusterCenterList.add(cc);
    36. }
    37. // create all cluster point
    38. for (int row = 0; row < height; ++row)
    39. {
    40. for (int col = 0; col < width; ++col)
    41. {
    42. index = row * width + col;
    43. int color = inPixels[index];
    44. pointList.add(new ClusterPoint(row, col, color));
    45. }
    46. }
    47. // initialize the clusters for each point
    48. double[] clusterDisValues = new double[clusterCenterList.size()];
    49. for(int i=0; i<pointList.size(); i++)
    50. {
    51. for(int j=0; j<clusterCenterList.size(); j++)
    52. {
    53. clusterDisValues[j] = calculateEuclideanDistance(pointList.get(i), clusterCenterList.get(j));
    54. }
    55. pointList.get(i).setClusterIndex(getCloserCluster(clusterDisValues));
    56. }
    57. // calculate the old summary
    58. // assign the points to cluster center
    59. // calculate the new cluster center
    60. // computation the delta value
    61. // stop condition--
    62. double[] oldClusterCenterColors = reCalculateClusterCenters();
    63. while(true)
    64. {
    65. stepClusters();
    66. double[] newClusterCenterColors = reCalculateClusterCenters();
    67. if(isStop(oldClusterCenterColors, newClusterCenterColors))
    68. {
    69. break;
    70. }
    71. else
    72. {
    73. oldClusterCenterColors = newClusterCenterColors;
    74. }
    75. }
    76. //update the result image
    77. dest = createCompatibleDestImage(src, null );
    78. index = 0;
    79. int[] outPixels = new int[width*height];
    80. for (int j = 0; j < pointList.size(); j++)
    81. {
    82. for (int i = 0; i < clusterCenterList.size(); i++)
    83. {
    84. ClusterPoint p = this.pointList.get(j);
    85. if (clusterCenterList.get(i).getcIndex() == p.getClusterIndex())
    86. {
    87. int row = (int)p.getX(); // row
    88. int col = (int)p.getY(); // column
    89. index = row * width + col;
    90. outPixels[index] = clusterCenterList.get(i).getPixelColor();
    91. }
    92. }
    93. }
    94. // fill the pixel data
    95. setRGB( dest, 0, 0, width, height, outPixels );
    96. return dest;
    97. }
    98. private boolean isStop(double[] oldClusterCenterColors, double[] newClusterCenterColors) {
    99. for(int i=0; i<oldClusterCenterColors.length; i++)
    100. {
    101. System.out.println("cluster " + i + " old : " + oldClusterCenterColors[i] + ", new : " + newClusterCenterColors[i]);
    102. if(oldClusterCenterColors[i]  != newClusterCenterColors[i])
    103. {
    104. return false;
    105. }
    106. }
    107. System.out.println();
    108. return true;
    109. }
    110. /**
    111. * update the cluster index by distance value
    112. */
    113. private void stepClusters()
    114. {
    115. // initialize the clusters for each point
    116. double[] clusterDisValues = new double[clusterCenterList.size()];
    117. for(int i=0; i<pointList.size(); i++)
    118. {
    119. for(int j=0; j<clusterCenterList.size(); j++)
    120. {
    121. clusterDisValues[j] = calculateEuclideanDistance(pointList.get(i), clusterCenterList.get(j));
    122. }
    123. pointList.get(i).setClusterIndex(getCloserCluster(clusterDisValues));
    124. }
    125. }
    126. /**
    127. * using cluster color of each point to update cluster center color
    128. *
    129. * @return
    130. */
    131. private double[] reCalculateClusterCenters() {
    132. // clear the points now
    133. for(int i=0; i<clusterCenterList.size(); i++)
    134. {
    135. clusterCenterList.get(i).setNumOfPoints(0);
    136. }
    137. // recalculate the sum and total of points for each cluster
    138. double[] redSums = new double[3];
    139. double[] greenSum = new double[3];
    140. double[] blueSum = new double[3];
    141. for(int i=0; i<pointList.size(); i++)
    142. {
    143. int cIndex = (int)pointList.get(i).getClusterIndex();
    144. clusterCenterList.get(cIndex).addPoints();
    145. int ta = (pointList.get(i).getPixelColor() >> 24) & 0xff;
    146. int tr = (pointList.get(i).getPixelColor() >> 16) & 0xff;
    147. int tg = (pointList.get(i).getPixelColor() >> 8) & 0xff;
    148. int tb = pointList.get(i).getPixelColor() & 0xff;
    149. ta = 255;
    150. redSums[cIndex] += tr;
    151. greenSum[cIndex] += tg;
    152. blueSum[cIndex] += tb;
    153. }
    154. double[] oldClusterCentersColors = new double[clusterCenterList.size()];
    155. for(int i=0; i<clusterCenterList.size(); i++)
    156. {
    157. double sum  = clusterCenterList.get(i).getNumOfPoints();
    158. int cIndex = clusterCenterList.get(i).getcIndex();
    159. int red = (int)(greenSum[cIndex]/sum);
    160. int green = (int)(greenSum[cIndex]/sum);
    161. int blue = (int)(blueSum[cIndex]/sum);
    162. System.out.println("red = " + red + " green = " + green + " blue = " + blue);
    163. int clusterColor = (255 << 24) | (red << 16) | (green << 8) | blue;
    164. clusterCenterList.get(i).setPixelColor(clusterColor);
    165. oldClusterCentersColors[i] = clusterColor;
    166. }
    167. return oldClusterCentersColors;
    168. }
    169. /**
    170. *
    171. * @param clusterDisValues
    172. * @return
    173. */
    174. private double getCloserCluster(double[] clusterDisValues)
    175. {
    176. double min = clusterDisValues[0];
    177. int clusterIndex = 0;
    178. for(int i=0; i<clusterDisValues.length; i++)
    179. {
    180. if(min > clusterDisValues[i])
    181. {
    182. min = clusterDisValues[i];
    183. clusterIndex = i;
    184. }
    185. }
    186. return clusterIndex;
    187. }
    188. /**
    189. *
    190. * @param point
    191. * @param cluster
    192. * @return distance value
    193. */
    194. private double calculateEuclideanDistance(ClusterPoint p, ClusterCenter c)
    195. {
    196. // int pa = (p.getPixelColor() >> 24) & 0xff;
    197. int pr = (p.getPixelColor() >> 16) & 0xff;
    198. int pg = (p.getPixelColor() >> 8) & 0xff;
    199. int pb = p.getPixelColor() & 0xff;
    200. // int ca = (c.getPixelColor() >> 24) & 0xff;
    201. int cr = (c.getPixelColor() >> 16) & 0xff;
    202. int cg = (c.getPixelColor() >> 8) & 0xff;
    203. int cb = c.getPixelColor() & 0xff;
    204. return Math.sqrt(Math.pow((pr - cr), 2.0) + Math.pow((pg - cg), 2.0) + Math.pow((pb - cb), 2.0));
    205. }
    206. }
    05-13 07:57
    查看更多