IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> SuperPixel -> 正文阅读

[人工智能]SuperPixel

Gonzalez R. C. and Woods R. E. Digital Image Processing (Forth Edition).

单个像素的意义其实很小, 于是有了superpixel的概念, 即一簇pixels的集合(且这堆pixels共用一个值), 这会导致图片有非常有趣的艺术风格(下图便是取不同的superpixel大小形成的效果, 有种抽象画的感觉?):

经过superpixel的预处理后, 图片可以变得更加容易提取edge, region, 毕竟superpixel已经率先提取过一次了.

SLIC Superpixel algorithm

SLIC (simple linear iterative clustering) 算法是基于k-means的一种聚类算法.

Given: 需要superpixels的个数 n s p n_{sp} nsp?; 图片 f ( x , y ) = ( r , g , b ) , x = 1 , 2 , ? M , y = 1 , 2 , ? ? , N f(x, y) = (r, g, b), x = 1,2,\cdots M, y = 1, 2, \cdots, N f(x,y)=(r,g,b),x=1,2,?M,y=1,2,?,N;

  1. 根据图片以及其位置信息生成数据:
    z = [ r , g , b , x , y ] T , \bm{z} = [r, g, b, x, y]^T, z=[r,g,b,x,y]T,
    其中 r , g , b r, g, b r,g,b是颜色编码, x , y x, y x,y是位置信息.

  2. n t p = M N n_{tp} = MN ntp?=MN表示pixels的个数, 并计算网格大小:
    s = [ n t p / n s p ] 1 / 2 . s = [n_{tp} / n_{sp}]^{1/2}. s=[ntp?/nsp?]1/2.

  3. 将图片均匀分割为大小 s s s的网格, 初始化superpixels的中心:
    m i = [ r i , g i , b i , x i , y i ] T , i = 1 , 2 , ? ? , n s p , \bm{m}_i = [r_i, g_i, b_i, x_i, y_i]^T, i=1,2,\cdots, n_{sp}, mi?=[ri?,gi?,bi?,xi?,yi?]T,i=1,2,?,nsp?,
    为网格的中心. 或者, 为了防止噪声的影响, 选择中心 3 × 3 3 \times 3 3×3领域内梯度最小的点.

  4. 将图片的每个pixel的类别标记为 L ( p ) = ? 1 L(p) = -1 L(p)=?1, 距离 d ( p ) = ∞ d(p) = \infty d(p)=;

  5. 重复下列步骤直到收敛:

    1. 对于每个像素点 p p p, 计算其与 2 s × 2 s 2s \times 2s 2s×2s邻域内的中心点 m i \bm{m}_i mi?之间的距离 D i ( p ) D_i(p) Di?(p), 倘若 D i ( p ) < d ( p ) D_i(p) < d(p) Di?(p)<d(p):
      d ( p ) = D i , L ( p ) = i . d(p) = D_i, L(p) = i. d(p)=Di?,L(p)=i.

    2. C i C_i Ci?表示 L ( p ) = i L(p) = i L(p)=i的像素点的集合, 更新superpixels的中心:
      m i = 1 ∣ C i ∣ ∑ z ∈ C i z , i = 1 , 2 , ? ? , n s p . \bm{m}_i = \frac{1}{|C_i|} \sum_{\bm{z} \in C_i} \bm{z}, i=1, 2, \cdots, n_{sp}. mi?=Ci?1?zCi??z,i=1,2,?,nsp?.

  6. 将以 m i \bm{m}_i mi?为中心的区域中的点的(r, g, b)设定为与 m i \bm{m}_i mi?一致.

距离函数的选择

倘若 D D D采用的是和普通K-means一样的 ∥ ? ∥ 2 \|\cdot\|_2 ?2?显然是不合适的, 因为 ( r , g , b ) (r, g, b) (r,g,b) ( x , y ) (x, y) (x,y)显然不是一个尺度的. 故采用如下的距离函数:
D = [ ( d c d c m ) 2 + ( d s d s m ) 2 ] 1 / 2 , d c = [ ( r j ? r i ) 2 + ( g j ? g i ) 2 + ( b j ? b i ) 2 ] 1 / 2 , d s = [ ( x j ? x i ) 2 + ( y j ? y i ) 2 ] 1 / 2 , D = [(\frac{d_c}{d_{cm}})^2 + (\frac{d_s}{d_{sm}})^2]^{1/2}, \\ d_c = [(r_j - r_i)^2 + (g_j - g_i)^2 + (b_j - b_i)^2]^{1/2}, \\ d_s = [(x_j - x_i)^2 + (y_j - y_i)^2]^{1/2}, D=[(dcm?dc??)2+(dsm?ds??)2]1/2,dc?=[(rj??ri?)2+(gj??gi?)2+(bj??bi?)2]1/2,ds?=[(xj??xi?)2+(yj??yi?)2]1/2,
其中 d c m , d s m d_{cm}, d_{sm} dcm?,dsm?分别是 d c , d s d_c, d_s dc?,ds?可能取到的最大值, 相当于标准化了.

代码

skimage.segmentation.slic

import numpy as np


def _generate_data(img):
    img = img.astype(np.float64)
    if len(img.shape) == 2:
        img = img[..., None]
    M, N = img.shape[0], img.shape[1]
    loc = np.stack(np.meshgrid(range(M), range(N), indexing='ij'), axis=-1)
    classes = -np.ones((M, N))
    distances = np.ones((M, N)) * np.float('inf')
    data = np.concatenate((img, loc), axis=-1)
    return data, classes, distances

def _generate_means(data, size: int):
    M, N = data.shape[0], data.shape[1]
    x_splits = np.arange(0, M + size, size)
    y_splits = np.arange(0, N + size, size)
    means = []
    for i in range(len(x_splits) - 1):
        for j in range(len(y_splits) - 1):
            r1, r2 = x_splits[i:i+2]
            c1, c2 = y_splits[j:j+2]
            region = data[r1:r2, c1:c2]
            means.append(region.mean(axis=(0, 1)))
    return np.array(means)


def _unit_step(data, means, classes, distances, size, dis_fn):
    M, N = data.shape[0], data.shape[1]
    size = 2 * size
    for i, m in enumerate(means):
        # ..., x, y
        x, y = np.round(m[-2:])
        x, y = int(x), int(y)
        xl, xr = max(0, x - size), min(x + size, M)
        yb, yt = max(0, y - size), min(y + size, N)
        p = data[xl:xr, yb:yt]
        _dis = dis_fn(p, m)
        indices = _dis < distances[xl:xr, yb:yt]
        distances[xl:xr, yb:yt][indices] = _dis[indices]
        classes[xl:xr, yb:yt][indices] = i

    # update
    for i in range(len(means)):
        x_indices, y_indices = np.where(classes == i)
        if len(x_indices) == 0:
            continue
        means[i] = data[x_indices, y_indices].mean(axis=0)

def slic(img, size, max_iters=10, compactness=10):
    data, classes, distances = _generate_data(img)
    means = _generate_means(data, size)
    dsm = size
    dcm = (img.max(axis=(0, 1)) - img.min(axis=(0, 1))) * compactness
    dsc = np.concatenate((dcm, [dsm] * 2))
    def dis_func(p, m):
        _dis = ((p - m) / dsc) ** 2
        return _dis.sum(axis=-1)
    for _ in range(max_iters):
        _unit_step(data, means, classes, distances, size, dis_func)
    new_img = np.zeros_like(img, dtype=np.float)
    for i, m in enumerate(means):
        x_indices, y_indices = np.where(classes == i)
        if len(x_indices) == 0:
            continue
        new_img[x_indices, y_indices] = m[:-2]
    return new_img.astype(img.dtype)
    

from skimage import io, segmentation, filters
from freeplot.base import FreePlot
    
img = io.imread(r"Lenna.png")    

ours = slic(img, size=50, compactness=0.5)

def mask2img(mask, img):
    new_img = img.astype(np.float)
    masks = np.unique(mask)
    for m in masks:
        x, y = np.where(mask == m)
        mcolor = new_img[x, y].mean(axis=0)
        new_img[x, y] = mcolor
    return new_img.astype(img.dtype)


mask = segmentation.slic(img)
yours = mask2img(mask, img)

fp = FreePlot((1, 3), (10.3, 5), titles=('Lenna', 'ours', 'skimage.segmentation.slic'))
fp.imageplot(img, index=(0, 0))
fp.imageplot(ours, index=(0, 1))
fp.imageplot(yours, index=(0, 2))
fp.set_title()
fp.show()

skimage上实现的代码还有强制连通性, 我想这个是为什么它看起来这么流畅的原因. Compactness 越大, 聚类越倾向于空间信息, 所以越容易出现块状结构.

  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章           查看所有文章
加:2021-09-18 10:08:28  更:2021-09-18 10:11:26 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 17:12:20-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码