在 Python 中使用 Pillow 进行图像处理【3/4】

第三部分

一、腐蚀和膨胀

        您可以查看名为 的图像文件dot_and_hole.jpg,您可以从本教程链接的存储库中下载该文件:

用于演示 Python Pillow 中的腐蚀和膨胀的示例图像

        该二值图像的左侧显示黑色背景上的白点,而右侧显示纯白色部分中的黑洞。

        侵蚀是从图像边界去除白色像素的过程。您可以通过使用二进制图像 ImageFilter.MinFilter(3)作为该方法的参数来实现此目的.filter()3x3此过滤器将像素的值替换为以该像素为中心的阵列中九个像素的最小值。在二值图像中,这意味着如果某个像素的任何相邻像素为零,则该像素的值将为零。

    ImageFilter.MinFilter(3)通过对图像应用多次,您可以看到侵蚀的效果dot_and_hole.jpg。您应该继续使用与上一节中相同的 REPL 会话:

>>>

>>> from PIL import ImageFilter
>>> filename = "dot_and_hole.jpg"

>>> with Image.open(filename) as img:
...     img.load()
...

>>> for _ in range(3):
...     img = img.filter(ImageFilter.MinFilter(3))
...

>>> img.show()

         您已使用循环应用了三次过滤器for。此代码给出以下输出:

Python Pillow 中的侵蚀

        由于侵蚀,圆点缩小了,但孔却变大了。

        膨胀是与腐蚀相反的过程。白色像素被添加到二值图像的边界。您可以使用 来实现膨胀ImageFilter.MaxFilter(3),如果某个像素的任何邻居是白色,则该像素将其转换为白色。

        您可以对包含点和孔的同一图像应用膨胀,您可以再次打开并加载该图像:

>>>

>>> with Image.open(filename) as img:
...     img.load()
...

>>> for _ in range(3):
...     img = img.filter(ImageFilter.MaxFilter(3))
...

>>> img.show()

        点现在变大了,洞缩小了:

Python Pillow 中的膨胀

        您可以同时使用腐蚀和膨胀来填充孔洞并从二值图像中删除小对象。使用带有点和孔的图像,您可以执行十次腐蚀循环来删除点,然后执行十次膨胀循环以将孔恢复到其原始大小:

>>>

>>> with Image.open(filename) as img:
...     img.load()
...

>>> for _ in range(10):
...     img = img.filter(ImageFilter.MinFilter(3))
...

>>> img.show()

>>> for _ in range(10):
...     img = img.filter(ImageFilter.MaxFilter(3))
...

>>> img.show()

        您可以使用第一个for循环执行十个侵蚀循环。此阶段的图像如下:

Python Pillow 中的侵蚀

        该点已消失,并且该孔比原始图像中的孔更大。第二个for循环执行十个膨胀周期,使孔返回到其原始大小:

Python Pillow 中的膨胀

        然而,该点不再出现在图像中。侵蚀和膨胀修改了图像以保留孔但移除点。所需的腐蚀和膨胀的数量取决于图像和您想要实现的目标。通常,您需要通过反复试验找到正确的组合。

        您可以定义函数来执行多个腐蚀和膨胀循环:

>>>

>>> def erode(cycles, image):
...     for _ in range(cycles):
...          image = image.filter(ImageFilter.MinFilter(3))
...     return image
...

>>> def dilate(cycles, image):
...     for _ in range(cycles):
...          image = image.filter(ImageFilter.MaxFilter(3))
...     return image
...

        这些函数使对图像进行腐蚀和膨胀实验变得更加容易。当您继续将猫放入修道院时,您将在下一节中使用这些函数。

二、使用阈值分割图像

        您可以对之前获得的阈值图像使用一系列侵蚀和膨胀,以删除遮罩中不代表猫的部分,并填充包含猫的区域中的任何间隙。一旦您尝试了腐蚀和膨胀,您将能够在试错过程中使用有根据的猜测来找到腐蚀和膨胀的最佳组合,以实现理想的蒙版。

        从您之前获得的图像开始img_cat_threshold,您可以从一系列腐蚀开始,以删除代表原始图像中背景的白色像素。您应该继续在与前面部分相同的 REPL 会话中工作:

>>>

>>> step_1 = erode(12, img_cat_threshold)
>>> step_1.show()

        腐蚀后的阈值图像不再包含代表图像背景的白色像素:

阈值处理后的腐蚀和膨胀

        然而,剩下的面具比猫的整体轮廓要小,并且内部有孔和间隙。您可以执行扩张来填补空白:

>>>

>>> step_2 = dilate(58, step_1)
>>> step_2.show()

        五十八个膨胀周期填充了掩模中的所有孔,得到以下图像:

        不过,这个面具太大了。因此,您可以通过一系列侵蚀来完成该过程:

>>>

>>> cat_mask = erode(45, step_2)
>>> cat_mask.show()

结果是一个可以用来分割猫图像的蒙版:

Python Pillow 中的图像分割掩码

您可以通过模糊此蒙版来避免二元蒙版的锐边。您必须首先将其从二值图像转换为灰度图像:

>>>

>>> cat_mask = cat_mask.convert("L")
>>> cat_mask = cat_mask.filter(ImageFilter.BoxBlur(20))
>>> cat_mask.show()

过滤BoxBlur()器返回以下掩码:

Python Pillow 中的图像分割掩码

面具现在看起来像一只猫!现在您已准备好从背景中提取猫的图像:

>>>

>>> blank = img_cat.point(lambda _: 0)
>>> cat_segmented = Image.composite(img_cat, blank, cat_mask)
>>> cat_segmented.show()

        首先,创建一个与 img_cat 大小相同的空白图像。您可以使用 .point() 并将所有值设置为零,从 img_cat 创建一个新的 Image 对象。接下来,使用 PIL.Image 中的 composite() 函数创建由 img_cat 和空白组成的图像,并使用 cat_mask 来确定使用每个图像的哪些部分。合成图像如下所示:

Python Pillow 中猫的分割图像

您已经分割了猫的图像并从背景中提取了猫。

三、图像叠加使用Image.paste()

您可以更进一步,将猫的分割图像从本教程的图像存储库粘贴到修道院庭院的图像中:

>>>

>>> filename_monastery = "monastery.jpg"
>>> with Image.open(filename_monastery) as img_monastery:
...     img_monastery.load()

>>> img_monastery.paste(
...     img_cat.resize((img_cat.width // 5, img_cat.height // 5)),
...     (1300, 750),
...     cat_mask.resize((cat_mask.width // 5, cat_mask.height // 5)),
... )

>>> img_monastery.show()

        您曾经.paste()将一张图像粘贴到另一张图像上。该方法可以与三个参数一起使用:

  • 第一个参数是要粘贴的图像//。您使用整数除法运算符 ( ) 将图像大小调整为其大小的五分之一。
  • 第二个参数是主图像中要粘贴第二张图片的位置。该元组包含主图像中要放置要粘贴的图像左上角的坐标。
  • 如果您不想粘贴整个图像,第三个参数提供您希望使用的蒙版。

您已使用通过阈值处理、腐蚀和膨胀过程获得的蒙版来粘贴没有背景的猫。输出如下图所示:

叠加在修道院图像上的猫的分割图像

        您已将猫从一张图像中分割出来,并将其放入另一张图像中,以显示猫安静地坐在修道院庭院中,而不是原始图像中它坐在田野中。

四、创建水印

        本示例中的最终任务是将 Real Python 徽标作为水印添加到图像中。您可以从本教程随附的存储库中获取带有 Real Python 徽标的图像文件:

        获取图像: 单击此处访问您将使用 Pillow 操作和处理的图像。

您应该继续在同一个 REPL 会话中工作:

>>>

>>> logo = "realpython-logo.png"
>>> with Image.open(logo) as img_logo:
...     img_logo.load()
...

>>> img_logo = Image.open(logo)
>>> img_logo.show()

这是全尺寸的彩色徽标:

真正的Python标志

您可以将图像更改为灰度并使用阈值.point()将其转换为黑白图像。您还可以缩小其尺寸并将其转换为轮廓图像:

>>>

>>> img_logo = img_logo.convert("L")
>>> threshold = 50
>>> img_logo = img_logo.point(lambda x: 255 if x > threshold else 0)
>>> img_logo = img_logo.resize(
...     (img_logo.width // 2, img_logo.height // 2)
... )
>>> img_logo = img_logo.filter(ImageFilter.CONTOUR)
>>> img_logo.show()

输出显示 Real Python 徽标的轮廓。该轮廓非常适合用作图像上的水印:

Python Pillow 中真实 Python 徽标的轮廓

要将其用作水印,您需要反转颜色,使背景为黑色,只有要保留的轮廓为白色。您可以.point()再次使用以下方法来实现此目的:

>>>

>>> img_logo = img_logo.point(lambda x: 0 if x == 255 else 255)
>>> img_logo.show()

您已经转换了值为 的像素255并为它们分配了值0,将它们从白色像素转换为黑色像素。您将剩余的像素设置为白色。反转轮廓标志如下所示:

Python Pillow 中真实 Python 徽标的轮廓

最后一步是将这个轮廓粘贴到坐在修道院庭院里的猫的图像上。您可以.paste()再次使用:

>>>

>>> img_monastery.paste(img_logo, (480, 160), img_logo)
>>> img_monastery.show()

第一个参数.paste()表示您要粘贴的图像,第三个参数表示蒙版。在本例中,您将使用相同的图像作为蒙版,因为该图像是二值图像。第二个参数提供要粘贴图像的区域的左上角坐标。

该图像现在包含一个真正的 Python 水印:

带有 Real Python 水印的修道院里的猫图像

水印具有矩形轮廓,这是您之前使用的轮廓过滤器的结果。如果您希望删除此轮廓,可以使用 裁剪图像.crop()。这是一个您可以自己尝试的练习。

五、使用 NumPy 和 Pillow 进行图像处理

        Pillow 具有多种内置功能和过滤器可供选择。然而,有时您需要进一步操作超出 Pillow 中现有功能的图像。

        您可以借助NumPy进一步操作图像。NumPy 是一个非常流行的用于处理数值数组的 Python 库,它是与 Pillow 一起使用的理想工具。您可以在NumPy 教程:使用 Python 进入数据科学的第一步中了解有关 NumPy 的更多信息。

        将图像转换为 NumPy 数组时,您可以直接对数组中的像素执行所需的任何转换。在 NumPy 中完成处理后,您可以Image使用 Pillow 将数组转换回对象。您需要为此部分安装 NumPy:

(venv) $ python -m pip install numpy

        现在您已经安装了 NumPy,您可以使用 Pillow 和 NumPy 来发现两个图像之间的差异。

5.1 使用 NumPy 相互减去图像

        看看您是否能找出以下两幅图像之间的差异:

使用 Python Pillow 找出差异

        这不是一件难事!然而,你决定作弊并编写一个 Python 程序来为你解决这个难题。您可以从本教程附带的存储库下载图像文件house_left.jpghouse_right.jpg(图像来源):

        获取图像: 单击此处访问您将使用 Pillow 操作和处理的图像。

第一步是使用 Pillow 读取图像并将其转换为 NumPy 数组:

>>>

>>> import numpy as np
>>> from PIL import Image

>>> with Image.open("house_left.jpg") as left:
...     left.load()
...
>>> with Image.open("house_right.jpg") as right:
...     right.load()
...

>>> left_array = np.asarray(left)
>>> right_array = np.asarray(right)

>>> type(left_array)
<class 'numpy.ndarray'>
>>> type(right_array)
<class 'numpy.ndarray'>

        由于left_arrayright_array是 类型的对象numpy.ndarray,因此您可以使用 NumPy 中提供的所有工具来操作它们。您可以从一个数组中减去另一个数组以显示两个图像之间不同的像素:

>>>

>>> difference_array =  right_array - left_array
>>> type(difference_array)
<class 'numpy.ndarray'>

        当您从另一个相同大小的数组中减去一个数组时,结果是另一个与原始数组具有相同形状的数组。Image.fromarray()您可以使用Pillow将此数组转换为图像:

>>>

>>> difference = Image.fromarray(difference_array)
>>> difference.show()

将一个 NumPy 数组减去另一个 NumPy 数组并转换为 Pillow 的结果Image是如下所示的差异图像:

使用 Python Pillow 找出差异

        差异图像仅显示原始图像的三个区域。这些区域突出了两个图像之间的差异。您还可以看到云和栅栏周围有一些噪点,这是由于这些项目周围区域的原始 JPEG 压缩发生了微小变化。

5.2 使用 NumPy 创建图像

        您可以更进一步,使用 NumPy 和 Pillow 从头开始​​创建图像。您可以从创建灰度图像开始。在此示例中,您将创建一个包含正方形的简单图像,但您可以用相同的方式创建更复杂的图像:

>>>

>>> import numpy as np
>>> from PIL import Image

>>> square = np.zeros((600, 600))
>>> square[200:400, 200:400] = 255

>>> square_img = Image.fromarray(square)
>>> square_img
<PIL.Image.Image image mode=F size=600x600 at 0x7FC7D8541F70>

>>> square_img.show()

        您创建一个大小随处包含零的数组600x600。接下来,将数组中心的一组像素的值设置为255

        您可以使用行和列对 NumPy 数组进行索引。在此示例中,第一个切片表示的200:400行。逗号后面的第二个切片表示的列。200399200:400200399

        您可以使用Image.fromarray()将 NumPy 数组转换为 类型的对象Image。上面代码的输出如下所示:

使用 NumPy 和 Python Pillow 生成的图像

        您已经创建了一个包含正方形的灰度图像。当您使用 时,会自动推断图像的模式Image.fromarray()。在这种情况下,使用模式"F",它对应于具有 32 位浮点像素的图像。如果您愿意,可以将其转换为更简单的 8 位像素灰度图像:

>>>

>>> square_img = square_img.convert("L")

        您还可以更进一步,创建彩色图像。您可以重复上述过程来创建三张图像,一张对应于红色通道,另一张对应于绿色通道,最后一张对应于蓝色通道:

>>>

>>> red = np.zeros((600, 600))
>>> green = np.zeros((600, 600))
>>> blue = np.zeros((600, 600))
>>> red[150:350, 150:350] = 255
>>> green[200:400, 200:400] = 255
>>> blue[250:450, 250:450] = 255

>>> red_img = Image.fromarray(red).convert("L")
>>> green_img = Image.fromarray(green).convert("L")
>>> blue_img = Image.fromarray((blue)).convert("L")

        您从每个 NumPy 数组创建一个Image对象,并将图像转换为 mode "L",它表示灰度。现在,您可以使用以下命令将这三个单独的图像合并为一个 RGB 图像Image.merge()

>>>

>>> square_img = Image.merge("RGB", (red_img, green_img, blue_img))
>>> square_img
<PIL.Image.Image image mode=RGB size=600x600 at 0x7FC7C817B9D0>

>>> square_img.show()

        第一个参数Image.merge()是图像输出的模式。第二个参数是包含各个单波段图像的序列。此代码创建以下图像:

使用 NumPy 和 Python Pillow 生成的彩色图像

        您已将单独的波段组合成 RGB 彩色图像。在下一节中,您将更进一步,使用 NumPy 和 Pillow 创建 GIF 动画。

六、创建动画

        在上一节中,您创建了一个彩色图像,其中包含三个不同颜色的重叠正方形。在本部分中,您将创建一个动画,显示这三个方块合并为一个白色方块。您将创建包含三个正方形的图像的多个版本,并且连续图像之间的正方形位置会略有不同:

>>>

>>> import numpy as np
>>> from PIL import Image

>>> square_animation = []
>>> for offset in range(0, 100, 2):
...     red = np.zeros((600, 600))
...     green = np.zeros((600, 600))
...     blue = np.zeros((600, 600))
...     red[101 + offset : 301 + offset, 101 + offset : 301 + offset] = 255
...     green[200:400, 200:400] = 255
...     blue[299 - offset : 499 - offset, 299 - offset : 499 - offset] = 255
...     red_img = Image.fromarray(red).convert("L")
...     green_img = Image.fromarray(green).convert("L")
...     blue_img = Image.fromarray((blue)).convert("L")
...     square_animation.append(
...         Image.merge(
...             "RGB",
...             (red_img, green_img, blue_img)
...         )
...     )
...

        您创建一个名为 的空列表square_animation,用于存储您生成的各种图像。在for循环中,您为红色、绿色和蓝色通道创建 NumPy 数组,如上一节中所做的那样。包含绿色层的数组始终相同,代表图像中心的一个正方形。

        红色方块从移至中心左上角的位置开始。在每个连续帧中,红色方块都会向中心移动,直到在循环的最终迭代中到达中心。蓝色方块最初向右下角移动,然后随着每次迭代向中心移动。

        请注意,在此示例中,您正在迭代range(0, 100, 2),这意味着变量offset以 2 为步长增加。

        您之前了解到可以Image使用 将对象保存到文件中Image.save()。您可以使用相同的功能保存到包含图像序列的 GIF 文件。您调用Image.save()序列中的第一张图像,这是您存储在列表中的第一张图像square_animation

>>>

>>> square_animation[0].save(
...     "animation.gif", save_all=True, append_images=square_animation[1:]
... )

        第一个参数.save()是要保存的文件的文件名。文件名中的扩展名告诉我们.save()需要输出什么文件格式。您还可以在中包含两个关键字参数.save()

  • save_all=True确保序列中的所有图像都被保存,而不仅仅是第一个图像。
  • append_images=square_animation[1:]允许您将序列中的剩余图像附加到 GIF 文件。

        此代码保存animation.gif到文件,然后您可以使用任何图像软件打开 GIF 文件。GIF 默认情况下应该循环,但在某些系统上,您需要添加关键字参数loop=0.save()确保 GIF 循环。您得到的动画如下:

蓝色、绿色和红色方块合并成一个以黑色背景为中心的白色方块

        三个不同颜色的方块合并成一个白色方块。您可以使用不同的形状和不同的颜色创建自己的动画吗?

七、结论

        您已经学习了如何使用 Pillow 处理图像并执行图像处理。如果您喜欢处理图像,您可能想一头扎进图像处理的世界。关于图像处理的理论和实践还有很多东西需要学习。一个很好的起点是Gonzalez 和 Woods 的《数字图像处理》,这是该领域的经典教科书。

        Pillow 并不是唯一可以在 Python 中用于图像处理的库。如果您的目标是执行一些基本处理,那么您在本教程中学到的技术可能就是您所需要的。如果您想深入了解更先进的图像处理技术,例如机器学习和计算机视觉应用,那么您可以使用 Pillow 作为其他库(例如 OpenCV 和 scikit-image)的垫脚石。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/103170.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何创建前端绘图和图表?

聚沙成塔每天进步一点点 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 欢迎来到前端入门之旅&#xff01;感兴趣的可以订阅本专栏哦&#xff01;这个专栏是为那些对Web开发感兴趣、刚刚踏入前端领域的朋友们量身打造的。无论你是完全的新手还是有一些基础的开发…

【excel技巧】excel单元格内如何换行?

Excel表格&#xff0c;在制作完成之后&#xff0c;在输入数据的时候&#xff0c;总是会遇到内容长度太长导致无法全部显示或者破坏表格整体格式。几天分享4个单元格换行的方法给大家。 方法一&#xff1a; 首先我们先介绍一个&#xff0c;通过调整列宽的方式来达到显示全部内…

使用Python的Flask框架开发验证码登录功能

目录 一、安装和配置Flask 二、生成验证码 三、处理用户输入和验证验证码 四、实现安全的用户认证 五、创建HTML模板 总结 验证码登录功能是现代Web应用程序中常见的安全特性之一&#xff0c;它有助于防止自动化机器人或恶意用户进行非法登录。在本文中&#xff0c;我们将…

hadoop伪分布式安装部署

首先jdk安装完毕 jdk安装文档参考&#xff1a; Linux 环境下安装JDK1.8并配置环境变量_linux安装jdk1.8并配置环境变量_Xi-Yuan的博客-CSDN博客 准备好hadoop的安装包 我的下载地址如下&#xff1a; We Transfer Gratuit. Envoi scuris de gros fichiers. 将hadoop包上传到随…

华为云 CodeArts Snap 智能编程助手 PyCharm 插件安装与使用指南

1 插件安装下载 1.1 搜索插件 打开 PyCharm&#xff0c;选择 File&#xff0c;点击 Settings。 选择 Plugins&#xff0c;点击 Marketplace&#xff0c;并在搜索框中输入 Huawei Cloud CodeArts Snap。 1.2 安装插件 如上图所示&#xff0c;点击 Install 按钮安装 Huawei Cl…

Azure - 机器学习企业级服务概述与介绍

目录 一、什么是 Azure 机器学习&#xff1f;大规模生成业务关键型机器学习模型 二、Azure 机器学习适合哪些人群&#xff1f;三、Azure 机器学习的价值点加快价值实现速度协作并简化 MLOps信心十足地开发负责任地设计 四、端到端机器学习生命周期的支持准备数据生成和训练模型…

Spark简单回顾

星光下的赶路人star的个人主页 大鹏一日同风起&#xff0c;扶摇直上九万里 文章目录 1、Spark1.1 Spark入门1.1.1 Spark部署模式1.1.2 常用端口 1.2 SparkCore1.2.1 RDD不可变和五大属性1.2.2 RDD的弹性1.2.3 cache和Checkpoint的区别1.2.4 算子 1.3 SparkSQL1.4 内核1.4.1提交…

软件测试(五)自动化 selenium

文章目录 自动化测试单元测试&#xff1a;单元测试&#xff1a;UI自动化 selenium工具定义特点&#xff1a;原理&#xff1a;seleniumjava环境搭建SeleniumAPI获取测试结果&#xff1a;添加等待浏览器操作键盘事件鼠标事件多层框架/窗口定位下拉框处理弹窗处理上传文件操作关闭…

Windows电脑如何录制电脑桌面?

如果你使用的电脑是Windows系统&#xff0c;那你是不是想知道如何在Windows电脑上录制电脑桌面&#xff1f; 本文以win10为例&#xff0c;好消息是&#xff0c;Windows 10电脑自带录屏工具&#xff0c;你可以直接使用此录屏工具轻松录制视频&#xff0c;而无需下载其他第三方软…

Android切换主题生命周期流程与onSaveInstanceState和onRestoreInstanceState,Kotlin

Android切换主题生命周期流程与onSaveInstanceState和onRestoreInstanceState&#xff0c;Kotlin import android.os.Bundle import android.util.Log import androidx.appcompat.app.AppCompatActivityclass MainActivity : AppCompatActivity() {private val TAG "fly&…

基于侏儒猫鼬优化的BP神经网络(分类应用) - 附代码

基于侏儒猫鼬优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码 文章目录 基于侏儒猫鼬优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码1.鸢尾花iris数据介绍2.数据集整理3.侏儒猫鼬优化BP神经网络3.1 BP神经网络参数设置3.2 侏儒猫鼬算法应用 4.测试结果…

列表推导式、集合推导式、字典推导式、生成器

列表推导式 可以与三目运算符搭配使用 dict1 {name: "by", "age": 20} dict2 {name: "ss", "age": 25} dict3 {name: "sa", "age": 24} dict4 {name: "xs", "age": 27} list1 [dict1, …

软件设计包括了四个既独立又相互联系的活动

软件设计包括了四个既独立又相互联系的活动

windows协议详解之-RPC/SMB/LDAP/LSA/SAM域控协议关系

如果你在windows域控环境中&#xff0c;例如企业的网络中开启wireshark抓包&#xff0c;你一定会遇到一大堆各种各样的协议。不同于互联网服务&#xff08;大多基于HTTP&#xff09;&#xff0c;为了实现域控中各种各样的服务&#xff0c;windows的域控环境中采用了非常多的协议…

【广州华锐互动】VR消防员模拟灭火:身临其境的火场救援

随着科技的不断发展&#xff0c;虚拟现实&#xff08;VR&#xff09;技术已经逐渐渗透到各个领域&#xff0c;为我们带来了前所未有的沉浸式体验。在这其中&#xff0c;VR模拟消防员灭火体验无疑是一种极具创新性和实用性的应用。通过这项技术&#xff0c;人们可以亲身体验到消…

QT 自定义窗体加载完成函数

使用信号和槽函数&#xff0c;具体如下&#xff1a; QT-如何在窗口/对话框显示后自动执行指定任务_qt 界面显示完在调用函数-CSDN博客文章目录QT-如何在窗口/对话框显示后自动执行指定任务一、如何实现在窗口展示出来后&#xff0c;执行某个函数二、如何成功实现判断条件后选择…

前端实现菜单按钮级权限

核心思想就是通过登录请求此用户对应的权限菜单&#xff0c;然后跳转首页&#xff0c;触发全局前置导航守卫&#xff0c;在全局导航守卫中通过 addRoute 添加动态路由进去。addRoute有一个需要注意的地方&#xff0c;就是我们添加完动态路由后&#xff0c;地址栏上立即访问添加…

django建站过程(3)定义模型与管理页

定义模型与管理页 定义模型[models.py]迁移模型向管理注册模型[admin.py]注册模型使用Admin.site.register(模型名)修改Django后台管理的名称定义管理列表页面应用名称修改管理列表添加查询功能 django shell交互式shell会话 认证和授权 定义模型[models.py] 模仿博客形式&…

鸿鹄工程项目管理系统 Spring Cloud+Spring Boot+Mybatis+Vue+ElementUI+前后端分离构建工程项目管理系统项目背景

鸿鹄工程项目管理系统 Spring CloudSpring BootMybatisVueElementUI前后端分离构建工程项目管理系统 1. 项目背景 一、随着公司的快速发展&#xff0c;企业人员和经营规模不断壮大。为了提高工程管理效率、减轻劳动强度、提高信息处理速度和准确性&#xff0c;公司对内部工程管…

Dynamics 365 重写自带按钮

必备工具&#xff1a;Ribbon Workbench 步骤&#xff1a; 1、查看默认按钮使用的方法名称 右键按钮选择自定义命令(Customise Command)&#xff0c;然后查看command使用的命令 2、在前台chrome浏览器中搜索对应的命令&#xff0c;查看命令细节 3、基于命令细节新建command&am…