简明代码实现Unet眼底图像血管分割

项目工程文件结构如下:
在这里插入图片描述
参考了Retina_Unet项目,决定自己用代码来实现一遍,数据增强不是像Retina_Unet那样随机裁剪,而是将20个训练数据集按顺序裁剪,每张裁剪成48x48大小的144个patch,20张一共裁剪出2880个patch。

Unet模型:模型输入的张量形状为(Npatch,1,48,48),输出为(Npatch,2304,2)。Npatch表示训练集的样本数,本例中训练时为2880,预测时为144。
训练:把原作者代码中的SGD改为Adam,效果有提升。
预测:也需要先把待预测图像分割成48×48的小图,输入模型,然后把结果整理还原为完整图像,再和专家标注结果进行对比。代码中以测试集第一张图片为例,可自行修改为其他眼底图片路径。

util.py

import numpy as np
import cv2
import os
from PIL import Image

def read_image_and_name(path):
    imgdir = os.listdir(path)
    imglst = []
    imgs = []
    for v in imgdir:
        imglst.append(path + v)
        imgs.append(cv2.imread(path + v))
    print(imglst)
    print('original images shape: ' + str(np.array(imgs).shape))
    return imglst,imgs

def read_label_and_name(path):
    labeldir = os.listdir(path)
    labellst = []
    labels = []
    for v in labeldir:
        labellst.append(path + v)
        labels.append(np.asarray(Image.open(path + v)))
    print(labellst)
    print('original labels shape: ' + str(np.array(labels).shape))
    return labellst,labels

def resize(imgs,resize_height, resize_width):
    img_resize = []
    for file in imgs:
        img_resize.append(cv2.resize(file,(resize_height,resize_width)))
    return img_resize

#将N张576x576的图片裁剪成48x48
def crop(image,dx):
    list = []
    for i in range(image.shape[0]):
        for x in range(image.shape[1] // dx):
            for y in range(image.shape[2] // dx):
                list.append(image[ i,  y*dx : (y+1)*dx,  x*dx : (x+1)*dx]) #这里的list一共append了20x12x12=2880次所以返回的shape是(2880,48,48)
    return np.array(list)

# 网络预测输出转换成图像子块
# 网络预测输出 size=[Npatches, patch_height*patch_width, 2]
def pred_to_imgs(pred, patch_height, patch_width, mode="original"):
    assert (len(pred.shape)==3)  #3D array: (Npatches,height*width,2)
    assert (pred.shape[2]==2 )  #check the classes are 2  # 确认是否为二分类
    pred_images = np.empty((pred.shape[0],pred.shape[1]))  #(Npatches,height*width)
    if mode=="original": # 网络概率输出
        for i in range(pred.shape[0]):
            for pix in range(pred.shape[1]
  • 7
    点赞
  • 103
    收藏
    觉得还不错? 一键收藏
  • 25
    评论
UNet是一种常用于图像分割的深度学习网络结构,可以用于眼底血管分割。以下是一个基于Matlab实现UNet眼底血管分割代码示例: 1. 首先需要准备眼底图像和标签图像,可以使用公开数据集或自己标注得到。 2. 加载数据并进行预处理 ```matlab % 加载眼底图像和标签图像 img = imread('retina_img.png'); label = imread('retina_label.png'); % 将图像归一化到0-1之间 img = im2double(img); label = im2double(label); % 对图像和标签进行裁剪,使其大小是2的整数次幂 img = img(1:512, 1:512); label = label(1:512, 1:512); % 将标签图像转换为分类的one-hot编码 label = categorical(label, [0 1], [1 0]); ``` 3. 定义UNet网络结构 ```matlab % 定义UNet的编码器部分 encoder1 = convolution2dLayer(3, 64, 'Padding', 'same'); encoder1 = batchNormalizationLayer(encoder1); encoder1 = reluLayer(encoder1); encoder2 = maxPooling2dLayer(2, 'Stride', 2); encoder2 = convolution2dLayer(3, 128, 'Padding', 'same'); encoder2 = batchNormalizationLayer(encoder2); encoder2 = reluLayer(encoder2); encoder3 = maxPooling2dLayer(2, 'Stride', 2); encoder3 = convolution2dLayer(3, 256, 'Padding', 'same'); encoder3 = batchNormalizationLayer(encoder3); encoder3 = reluLayer(encoder3); encoder4 = maxPooling2dLayer(2, 'Stride', 2); encoder4 = convolution2dLayer(3, 512, 'Padding', 'same'); encoder4 = batchNormalizationLayer(encoder4); encoder4 = reluLayer(encoder4); % 定义UNet的解码器部分 decoder1 = transposedConv2dLayer(2, 512, 'Stride', 2); decoder1 = convolution2dLayer(3, 512, 'Padding', 'same'); decoder1 = batchNormalizationLayer(decoder1); decoder1 = reluLayer(decoder1); decoder2 = transposedConv2dLayer(2, 256, 'Stride', 2); decoder2 = convolution2dLayer(3, 256, 'Padding', 'same'); decoder2 = batchNormalizationLayer(decoder2); decoder2 = reluLayer(decoder2); decoder3 = transposedConv2dLayer(2, 128, 'Stride', 2); decoder3 = convolution2dLayer(3, 128, 'Padding', 'same'); decoder3 = batchNormalizationLayer(decoder3); decoder3 = reluLayer(decoder3); decoder4 = transposedConv2dLayer(2, 64, 'Stride', 2); decoder4 = convolution2dLayer(3, 64, 'Padding', 'same'); decoder4 = batchNormalizationLayer(decoder4); decoder4 = reluLayer(decoder4); % 定义UNet的输出层 outputLayer = convolution2dLayer(1, 2, 'Padding', 'same'); outputLayer = softmaxLayer(outputLayer); outputLayer = classificationLayer(outputLayer); % 将编码器和解码器部分连接起来,形成UNet网络 layers = [ encoder1 encoder2 encoder3 encoder4 decoder1 decoder2 decoder3 decoder4 outputLayer ]; ``` 4. 训练UNet网络 ```matlab % 设置训练选项 options = trainingOptions('adam', ... 'InitialLearnRate', 1e-3, ... 'MaxEpochs', 30, ... 'MiniBatchSize', 16, ... 'Shuffle', 'every-epoch', ... 'ValidationData', {img, label}, ... 'ValidationFrequency', 5, ... 'Plots', 'training-progress'); % 开始训练 net = trainNetwork(img, label, layers, options); ``` 5. 使用UNet网络进行眼底血管分割 ```matlab % 加载测试图像 testImg = imread('test_image.png'); testImg = im2double(testImg); % 对测试图像进行预处理 testImg = testImg(1:512, 1:512); % 使用UNet网络进行分割 predictedLabel = classify(net, testImg); % 将预测结果转换为二值图像 predictedLabel = double(predictedLabel) - 1; predictedLabel = imbinarize(predictedLabel); % 显示原图像分割结果 figure; subplot(1, 2, 1); imshow(testImg); title('Original Image'); subplot(1, 2, 2); imshow(predictedLabel); title('Segmentation Result'); ``` 以上是一个基于Matlab实现UNet眼底血管分割代码示例,可以根据实际需求进行调整和修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 25
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值