第一種:==r > 95 and g > 40 and g < 100 and b > 20 and max([r, g, b]) - min([r, g, b]) > 15 and abs(r - g) > 15 and r > g and r > b==
第二種:==nr = r / (r + g + b), ng = g / (r + g + b), nb = b / (r +g + b) ,nr / ng > 1.185 and r * b / (r + g + b) ** 2 > 0.107 and r * g / (r + g + b) ** 2 > 0.112==
for y in range(self.height): for x in range(self.width): # 得到像素的 RGB 三個(gè)通道的值 # [x, y] 是 [(x,y)] 的簡(jiǎn)便寫(xiě)法 r = pixels[x, y][0] # red g = pixels[x, y][1] # green b = pixels[x, y][2] # blue # 判斷當(dāng)前像素是否為膚色像素 isSkin = True if self._classify_skin(r, g, b) else False # 給每個(gè)像素分配唯一 id 值(1, 2, 3...height*width) # 注意 x, y 的值從零開(kāi)始 _id = x + y * self.width + 1 # 為每個(gè)像素創(chuàng)建一個(gè)對(duì)應(yīng)的 Skin 對(duì)象,并添加到 self.skin_map 中 self.skin_map.append(self.Skin(_id, isSkin, None, x, y))
若當(dāng)前像素并不是膚色,那么跳過(guò)本次循環(huán),繼續(xù)遍歷:
# 若當(dāng)前像素不為膚色像素,跳過(guò)此次循環(huán) if not isSkin: continue
# 用來(lái)記錄相鄰像素中膚色像素所在的區(qū)域號(hào),初始化為 -1 region = -1 # 遍歷每一個(gè)相鄰像素的索引 for index in check_indexes: # 嘗試索引相鄰像素的 Skin 對(duì)象,沒(méi)有則跳出循環(huán) try: self.skin_map[index] except IndexError: break # 相鄰像素若為膚色像素: if self.skin_map[index].skin: # 若相鄰像素與當(dāng)前像素的 region 均為有效值,且二者不同,且尚未添加相同的合并任務(wù) if (self.skin_map[index].region != None and region != None and region != -1 and self.skin_map[index].region != region and self.last_from != region and self.last_to != self.skin_map[index].region) : # 那么這添加這兩個(gè)區(qū)域的合并任務(wù) self._add_merge(region, self.skin_map[index].region) # 記錄此相鄰像素所在的區(qū)域號(hào) region = self.skin_map[index].region
# 基于像素的膚色檢測(cè)技術(shù) def _classify_skin(self, r, g, b): # 根據(jù)RGB值判定 rgb_classifier = r > 95 and \ g > 40 and g < 100 and \ b > 20 and \ max([r, g, b]) - min([r, g, b]) > 15 and \ abs(r - g) > 15 and \ r > g and \ r > b # 根據(jù)處理后的 RGB 值判定 nr, ng, nb = self._to_normalized(r, g, b) norm_rgb_classifier = nr / ng > 1.185 and \ float(r * b) / ((r + g + b) ** 2) > 0.107 and \ float(r * g) / ((r + g + b) ** 2) > 0.112
# HSV 顏色模式下的判定 h, s, v = self._to_hsv(r, g, b) hsv_classifier = h > 0 and \ h < 35 and \ s > 0.23 and \ s < 0.68
# YCbCr 顏色模式下的判定 y, cb, cr = self._to_ycbcr(r, g, b) ycbcr_classifier = 97.5 <= cb <= 142.5 and 134 <= cr <= 176
# 效果不是很好,還需改公式 # return rgb_classifier or norm_rgb_classifier or hsv_classifier or ycbcr_classifier return ycbcr_classifier
def _to_normalized(self, r, g, b): if r == 0: r = 0.0001 if g == 0: g = 0.0001 if b == 0: b = 0.0001 _sum = float(r + g + b) return [r / _sum, g / _sum, b / _sum]
self._add_merge()方法主要是對(duì)self.merge_regions操作,而self.merge_regions 的元素都是包含一些 int 對(duì)象(區(qū)域號(hào))的列表,列表中的區(qū)域號(hào)代表的區(qū)域都是待合并的區(qū)。self._add_merge()方法接收兩個(gè)區(qū)域號(hào),將之添加到self.merge_regions中。
# 遍歷每個(gè) self.merge_regions 的元素 for index, region in enumerate(self.merge_regions): # 遍歷元素中的每個(gè)區(qū)域號(hào) for r_index in region: if r_index == _from: from_index = index if r_index == _to: to_index = index
# 若兩個(gè)區(qū)域號(hào)都存在于 self.merge_regions 中 if from_index != -1 and to_index != -1: # 如果這兩個(gè)區(qū)域號(hào)分別存在于兩個(gè)列表中 # 那么合并這兩個(gè)列表 if from_index != to_index: self.merge_regions[from_index].extend(self.merge_regions[to_index]) del(self.merge_regions[to_index]) return
# 若兩個(gè)區(qū)域號(hào)都不存在于 self.merge_regions 中 if from_index == -1 and to_index == -1: # 創(chuàng)建新的區(qū)域號(hào)列表 self.merge_regions.append([_from, _to]) return # 若兩個(gè)區(qū)域號(hào)中有一個(gè)存在于 self.merge_regions 中 if from_index != -1 and to_index == -1: # 將不存在于 self.merge_regions 中的那個(gè)區(qū)域號(hào) # 添加到另一個(gè)區(qū)域號(hào)所在的列表 self.merge_regions[from_index].append(_to) return # 若兩個(gè)待合并的區(qū)域號(hào)中有一個(gè)存在于 self.merge_regions 中 if from_index == -1 and to_index != -1: # 將不存在于 self.merge_regions 中的那個(gè)區(qū)域號(hào) # 添加到另一個(gè)區(qū)域號(hào)所在的列表 self.merge_regions[to_index].append(_from) return
# 將 merge_regions 中的元素中的區(qū)域號(hào)代表的所有區(qū)域合并 for index, region in enumerate(merge_regions): try: new_detected_regions[index] except IndexError: new_detected_regions.append([]) for r_index in region: new_detected_regions[index].extend(detected_regions[r_index]) detected_regions[r_index] = []
# 添加剩下的其余皮膚區(qū)域到 new_detected_regions for region in detected_regions: if len(region) > 0: new_detected_regions.append(region)
# 皮膚區(qū)域清理函數(shù) # 只保存像素?cái)?shù)大于指定數(shù)量的皮膚區(qū)域 def _clear_regions(self, detected_regions): for region in detected_regions: if len(region) > 30: self.skin_regions.append(region)
parser = argparse.ArgumentParser(description='Detect nudity in images.') parser.add_argument('files', metavar='image', nargs='+', help='Images you wish to test') parser.add_argument('-r', '--resize', action='store_true', help='Reduce image size to increase speed of scanning') parser.add_argument('-v', '--visualization', action='store_true', help='Generating areas of skin image')
args = parser.parse_args()
for fname in args.files: if os.path.isfile(fname): n = Nude(fname) if args.resize: n.resize(maxheight=800, maxwidth=600) n.parse() if args.visualization: n.showSkinRegions() print(n.result, n.inspect()) else: print(fname, "is not a file")