第一種:==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 三個通道的值 # [x, y] 是 [(x,y)] 的簡便寫法 r = pixels[x, y][0] # red g = pixels[x, y][1] # green b = pixels[x, y][2] # blue # 判斷當前像素是否為膚色像素 isSkin = True if self._classify_skin(r, g, b) else False # 給每個像素分配唯一 id 值(1, 2, 3...height*width) # 注意 x, y 的值從零開始 _id = x + y * self.width + 1 # 為每個像素創建一個對應的 Skin 對象,并添加到 self.skin_map 中 self.skin_map.append(self.Skin(_id, isSkin, None, x, y))
# 用來記錄相鄰像素中膚色像素所在的區域號,初始化為 -1 region = -1 # 遍歷每一個相鄰像素的索引 for index in check_indexes: # 嘗試索引相鄰像素的 Skin 對象,沒有則跳出循環 try: self.skin_map[index] except IndexError: break # 相鄰像素若為膚色像素: if self.skin_map[index].skin: # 若相鄰像素與當前像素的 region 均為有效值,且二者不同,且尚未添加相同的合并任務 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) : # 那么這添加這兩個區域的合并任務 self._add_merge(region, self.skin_map[index].region) # 記錄此相鄰像素所在的區域號 region = self.skin_map[index].region
# 基于像素的膚色檢測技術 def _classify_skin(self, r, g, b): # 根據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 # 根據處理后的 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()方法主要是對self.merge_regions操作,而self.merge_regions 的元素都是包含一些 int 對象(區域號)的列表,列表中的區域號代表的區域都是待合并的區。self._add_merge()方法接收兩個區域號,將之添加到self.merge_regions中。
# 遍歷每個 self.merge_regions 的元素 for index, region in enumerate(self.merge_regions): # 遍歷元素中的每個區域號 for r_index in region: if r_index == _from: from_index = index if r_index == _to: to_index = index
# 若兩個區域號都存在于 self.merge_regions 中 if from_index != -1 and to_index != -1: # 如果這兩個區域號分別存在于兩個列表中 # 那么合并這兩個列表 if from_index != to_index: self.merge_regions[from_index].extend(self.merge_regions[to_index]) del(self.merge_regions[to_index]) return
# 若兩個區域號都不存在于 self.merge_regions 中 if from_index == -1 and to_index == -1: # 創建新的區域號列表 self.merge_regions.append([_from, _to]) return # 若兩個區域號中有一個存在于 self.merge_regions 中 if from_index != -1 and to_index == -1: # 將不存在于 self.merge_regions 中的那個區域號 # 添加到另一個區域號所在的列表 self.merge_regions[from_index].append(_to) return # 若兩個待合并的區域號中有一個存在于 self.merge_regions 中 if from_index == -1 and to_index != -1: # 將不存在于 self.merge_regions 中的那個區域號 # 添加到另一個區域號所在的列表 self.merge_regions[to_index].append(_from) return
# 將 merge_regions 中的元素中的區域號代表的所有區域合并 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] = []
# 添加剩下的其余皮膚區域到 new_detected_regions for region in detected_regions: if len(region) > 0: new_detected_regions.append(region)
# 皮膚區域清理函數 # 只保存像素數大于指定數量的皮膚區域 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")