【Diffusers库】第二篇 快速生成图片

2024-03-08 18:28

本文主要是介绍【Diffusers库】第二篇 快速生成图片,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 写在前面的话
  • 提速的几个条件
    • 1. 硬件
    • 2. 精度
    • 3. 推理步数
    • 3. 内存
    • 4. 管道组件
  • 提质的几个条件
    • 1. 模型
    • 2. prompt

写在前面的话

  这是我们研发的用于 消费决策的AI助理 ,我们会持续优化,欢迎体验与反馈。微信扫描二维码,添加即可。
  官方链接:https://ailab.smzdm.com/

************************************************************** 分割线 *******************************************************************

  随着AI技术的发展,模型参数量越来越大,运行效率成了一个问题。在AIGC方面同样有这样的问题 ,本篇文章介绍一些方法,用于提升图像生成的效率。 本篇是以文生图为例子的,图生图同样适用!
  这篇文章的 搬运性质 多一些!哈哈~ 没有梯子的话,就老老实实在这里看吧!

提速的几个条件

  闲话不说,先实例化模型,使用的模型是"runwayml/stable-diffusion-v1-5",这个模型可能要5个G左右。下载起来确实很费事。

from diffusers import DiffusionPipeline
# 实例化对象
model_id = "runwayml/stable-diffusion-v1-5"
pipeline = DiffusionPipeline.from_pretrained(model_id, use_safetensors=True)

  本教程将使用的提示词是 “portrait photo of a old warrior chief”,但是你可以随心所欲的想象和构造自己的提示词:

prompt = "portrait photo of a old warrior chief"

1. 硬件

  加速推理的最简单方法之一是将 pipeline 放在 GPU 上 ,就像使用任何 PyTorch 模块一样:

import torchpipeline = pipeline.to("cuda")# 为确保您可以使用相同的图像并对其进行改进,使用 Generator 方法,然后设置一个随机数种子 以确保其 复现性:
generator = torch.Generator("cuda").manual_seed(0)  # 生成图像
image = pipeline(prompt, generator=generator).images[0]
image

  现在,你可以生成一个图像:
在这里插入图片描述

2. 精度

  在 T4 GPU 上,这个过程大概要30秒(如果你的 GPU 比 T4 好,可能会更快,笔者用的V100所以会快一些)。在默认情况下,DiffusionPipeline 使用完整的 float32 精度进行 50 步推理。你可以通过降低精度(如 float16 )或者减少推理步数来加速整个过程

  让我们把模型的精度降低至 float16 ,提升速度是比较明显的,然后生成一张图像:

import torch# 实例化对象
pipeline = DiffusionPipeline.from_pretrained(model_id, torch_dtype=torch.float16, use_safetensors=True)
pipeline = pipeline.to("cuda")# 生成随机种子 
generator = torch.Generator("cuda").manual_seed(0)
image = pipeline(prompt, generator=generator).images[0]
image      

在这里插入图片描述

3. 推理步数

  推理步数的调整会涉及到调度器,因为有的调度器可能4步就搞定了,有的调度器可能20步在可以。
  你可以选择一个更高效的调度器 (scheduler) 可以减少推理步数同时保证输出质量。您可以在 [DiffusionPipeline] 中通过调用compatibles方法找到与当前模型兼容的调度器 (scheduler)。

pipeline.scheduler.compatibles
[diffusers.schedulers.scheduling_lms_discrete.LMSDiscreteScheduler,diffusers.schedulers.scheduling_unipc_multistep.UniPCMultistepScheduler,diffusers.schedulers.scheduling_k_dpm_2_discrete.KDPM2DiscreteScheduler,diffusers.schedulers.scheduling_deis_multistep.DEISMultistepScheduler,diffusers.schedulers.scheduling_euler_discrete.EulerDiscreteScheduler,diffusers.schedulers.scheduling_dpmsolver_multistep.DPMSolverMultistepScheduler,diffusers.schedulers.scheduling_ddpm.DDPMScheduler,diffusers.schedulers.scheduling_dpmsolver_singlestep.DPMSolverSinglestepScheduler,diffusers.schedulers.scheduling_k_dpm_2_ancestral_discrete.KDPM2AncestralDiscreteScheduler,diffusers.schedulers.scheduling_heun_discrete.HeunDiscreteScheduler,diffusers.schedulers.scheduling_pndm.PNDMScheduler,diffusers.schedulers.scheduling_euler_ancestral_discrete.EulerAncestralDiscreteScheduler,diffusers.schedulers.scheduling_ddim.DDIMScheduler,
]

  Stable Diffusion 模型默认使用的是 PNDMScheduler ,通常要大概50步推理, 但是像 DPMSolverMultistepScheduler 这样更高效的调度器只要大概 20 或 25 步推理. 使用 ConfigMixin.from_config() 方法加载新的调度器:

from diffusers import DPMSolverMultistepSchedulerpipeline.scheduler = DPMSolverMultistepScheduler.from_config(pipeline.scheduler.config)  

  现在将 num_inference_steps 设置为 20:

generator = torch.Generator("cuda").manual_seed(0)
image = pipeline(prompt, generator=generator, num_inference_steps=20).images[0]
image  

在这里插入图片描述

  太棒了!你成功把推理时间缩短到 4 秒(官方4秒,我的更快)!

3. 内存

  改善 pipeline 性能的另一个关键是减少内存的使用量,这间接意味着速度更快,因为你经常试图最大化每秒生成的图像数量。要想知道你一次可以生成多少张图片,最简单的方法是尝试不同的batch size,直到出现Out Of Memory Error (OOM)。

  创建一个函数,为每一批要生成的图像分配提示词和 Generators 。请务必为每个Generator 分配一个种子,以便于复现良好的结果。

def get_inputs(batch_size=1):generator = [torch.Generator("cuda").manual_seed(i) for i in range(batch_size)]prompts = batch_size * [prompt]num_inference_steps = 20return {"prompt": prompts, "generator": generator, "num_inference_steps": num_inference_steps}     

  设置 batch_size=4 ,然后看一看我们消耗了多少内存:

from diffusers.utils import make_image_grid images = pipeline(**get_inputs(batch_size=4)).images
make_image_grid(images, 2, 2)

  除非你有一个更大内存的GPU, 否则上述代码会返回 OOM 错误! 大部分内存被 cross-attention 层使用。按顺序运行可以节省大量内存,而不是在批处理中进行。你可以为 pipeline 配置 enable_attention_slicing() 函数:

pipeline.enable_attention_slicing()

  现在尝试把 batch_size 增加到 8

images = pipeline(**get_inputs(batch_size=8)).images
make_image_grid(images, rows=2, cols=4)

  以前你不能一批生成 4 张图片,而现在你可以在一张图片里面生成八张图片而只需要大概3.5秒!这可能是 T4 GPU 在不牺牲质量的情况运行速度最快的一种方法。
在这里插入图片描述

4. 管道组件

随着diffusers库的更新,管道组件也随着丰富,相继推出了一些高效的出图组件,下面介绍一些新的组件

from diffusers import AutoencoderKLvae = AutoencoderKL.from_pretrained("stabilityai/sd-vae-ft-mse", torch_dtype=torch.float16).to("cuda")
pipeline.vae = vae
images = pipeline(**get_inputs(batch_size=8)).images
image_grid(images, rows=2, cols=4)

在这里插入图片描述

提质的几个条件

1. 模型

  稳定扩散模型是一个很好的开始,自正式发布以来,还发布了几个改进版本。然而,使用更新的版本并不意味着你会得到更好的结果。你仍然需要自己尝试不同的模型,并做一些研究。例如想生成的图形是二次元、真人、场景、科幻、3D风格等等,根据自己想要的风格去选择合适的模型事半功倍。可以去hugging face 和 civital 网站上去找,里面有很多图像和模型,可以选择。
在这里插入图片描述

  另外,Lora的使用也是可以提升质量的,但他需要训练,也属于模型,所以也放在这里。

2. prompt

  prompt的写法是有讲究的。它属于一个比较庞大的模块。需要花一些时间去学习一下,这里只是简单点一下。推荐去 civital 网站上去看看,网站上的图片有对应的详细参数,其中包括prompt。
  我理解的是prompt 主要包括3个部分:主体、场景、修饰
  主体和场景是根据实际描述,穿插介绍的,修饰词一般放在最后。
在这里插入图片描述
  另外还有一些 起到强调作用的写法, 比如:(被强调词:1.5)。给个例子:
Prompt

masterpiece,professional macro photography, small sprouting  Clover  plant  (symbol of hope, love and faith) in the war zone field,chaotic background, fire,war,  soft bounced lighting, amazing depth of field, shot from a low angle, shot on Lumix GH5 (cinematic bokeh, dynamic range, vibrant colors)

Negative prompt

anime, cartoon, deformed, glitch, noisy, low contrast

在这里插入图片描述

这篇关于【Diffusers库】第二篇 快速生成图片的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/787989

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

C#中图片如何自适应pictureBox大小

《C#中图片如何自适应pictureBox大小》文章描述了如何在C#中实现图片自适应pictureBox大小,并展示修改前后的效果,修改步骤包括两步,作者分享了个人经验,希望对大家有所帮助... 目录C#图片自适应pictureBox大小编程修改步骤总结C#图片自适应pictureBox大小上图中“z轴

使用Python将长图片分割为若干张小图片

《使用Python将长图片分割为若干张小图片》这篇文章主要为大家详细介绍了如何使用Python将长图片分割为若干张小图片,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. python需求的任务2. Python代码的实现3. 代码修改的位置4. 运行结果1. Python需求

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

shell脚本快速检查192.168.1网段ip是否在用的方法

《shell脚本快速检查192.168.1网段ip是否在用的方法》该Shell脚本通过并发ping命令检查192.168.1网段中哪些IP地址正在使用,脚本定义了网络段、超时时间和并行扫描数量,并使用... 目录脚本:检查 192.168.1 网段 IP 是否在用脚本说明使用方法示例输出优化建议总结检查 1

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

Rust中的Option枚举快速入门教程

《Rust中的Option枚举快速入门教程》Rust中的Option枚举用于表示可能不存在的值,提供了多种方法来处理这些值,避免了空指针异常,文章介绍了Option的定义、常见方法、使用场景以及注意事... 目录引言Option介绍Option的常见方法Option使用场景场景一:函数返回可能不存在的值场景

详解Java中如何使用JFreeChart生成甘特图

《详解Java中如何使用JFreeChart生成甘特图》甘特图是一种流行的项目管理工具,用于显示项目的进度和任务分配,在Java开发中,JFreeChart是一个强大的开源图表库,能够生成各种类型的图... 目录引言一、JFreeChart简介二、准备工作三、创建甘特图1. 定义数据集2. 创建甘特图3.

使用 Python 和 LabelMe 实现图片验证码的自动标注功能

《使用Python和LabelMe实现图片验证码的自动标注功能》文章介绍了如何使用Python和LabelMe自动标注图片验证码,主要步骤包括图像预处理、OCR识别和生成标注文件,通过结合Pa... 目录使用 python 和 LabelMe 实现图片验证码的自动标注环境准备必备工具安装依赖实现自动标注核心