如何在Linux下安装和使用Ansible?自动化运维的入门到进阶指南

安装Ansible最直接的方法是使用系统包管理器,如Ubuntu/Debian的apt或CentOS/RHEL的dnf,也可通过pip安装以支持跨平台或虚拟环境。安装后需配置inventory文件定义受管主机,随后可通过ad-hoc命令如ansible all -m ping测试连接。核心功能通过YAML格式的Playbook实现,如使用ansible.builtin.apt和ansible.builtin.service模块确保Nginx安装并运行,体现自动化部署优势。相比Shell脚本,Ansible具备幂等性、无代理架构、模块化设计和清晰的错误报告,更适合规模化运维。编写高效Playbook应遵循最佳实践:使用Roles组织代码、优先调用专用模块而非shell命令、合理利用变量提升灵活性、通过Handlers管理服务重启、用Tags和When实现选择性执行与条件控制。进阶用法包括动态清单(如aws_ec2.yml)自动同步云主机信息,避免静态维护IP列表;使用Ansible Vault加密敏感数据如密码和密钥;并将Ansible集成至CI/CD流程(如GitLab CI),实现从代码提交到生产部署的全自动化,显著提升发布效率与系统可靠性。

如何在linux下安装和使用ansible?自动化运维的入门到进阶指南

要在Linux上安装和使用Ansible,通常最直接的方法是利用你发行版的包管理器,比如Debian/Ubuntu系的

apt

或CentOS/RHEL系的

yum

/

dnf

。安装完成后,你需要配置一个清单文件(inventory),里面列出你要管理的所有服务器,然后就可以通过编写Playbook来自动化执行各种任务了。这个过程从简单的配置管理到复杂的应用部署都能覆盖,是自动化运维的基石之一。

解决方案

安装Ansible其实挺简单的,但不同的Linux发行版会有些许差异。我个人更倾向于使用系统自带的包管理器,因为它能很好地处理依赖关系,也方便后续更新。

1. 安装Ansible

对于Ubuntu/Debian系:通常,Ansible在默认的仓库里就能找到。

sudo apt updatesudo apt install ansible

有时候,如果需要最新版本或者遇到依赖问题,可以考虑添加PPA(Personal Package Archive):

sudo apt updatesudo apt install software-properties-commonsudo add-apt-repository --yes --update ppa:ansible/ansiblesudo apt install ansible

对于CentOS/RHEL/Fedora系:这些系统通常需要先安装EPEL (Extra Packages for Enterprise Linux) 仓库,因为Ansible不在默认的AppStream或BaseOS仓库中。

sudo dnf install epel-release # 或者 sudo yum install epel-releasesudo dnf install ansible      # 或者 sudo yum install ansible

使用Python的pip安装(跨平台通用,也适合虚拟环境):如果你想在一个特定的Python环境里运行Ansible,或者需要安装最新、非稳定版本,

pip

是个不错的选择。

sudo apt install python3-pip # 或 sudo dnf install python3-pippip3 install ansible

我个人建议,如果你只是在本地作为控制节点,并且不希望污染系统Python环境,最好在一个Python虚拟环境(

venv

)中安装Ansible。这样管理起来更干净。

2. 配置库存文件(Inventory)

Ansible通过SSH连接到远程服务器执行任务,它需要知道这些服务器的IP地址或主机名。这个信息就放在库存文件里,通常是

hosts

。默认情况下,Ansible会尝试读取

/etc/ansible/hosts

一个简单的

hosts

文件可能长这样:

[webservers]web1.example.comweb2.example.com[databases]db1.example.comdb2.example.com ansible_port=2222 # 如果SSH端口不是默认的22[all:vars]ansible_user=your_ssh_useransible_ssh_private_key_file=~/.ssh/id_rsa

这里我定义了两个组:

webservers

databases

all:vars

部分定义了所有主机的默认SSH用户和私钥路径。这些变量也可以针对特定组或特定主机定义。

3. 第一次尝试:ping一下!

安装并配置好inventory后,我们可以先用

ping

模块测试一下连接性。

ansible all -m ping

如果一切顺利,你会看到类似这样的输出:

web1.example.com | SUCCESS => {    "ansible_facts": {        "discovered_interpreter_python": "/usr/bin/python3"    },    "changed": false,    "ping": "pong"}

这表示Ansible成功通过SSH连接到了远程主机,并且可以执行命令。

4. 编写第一个Playbook

Playbook是Ansible的核心,它用YAML格式编写,描述了要在远程主机上执行的一系列任务。

我们来写一个简单的Playbook,比如确保Nginx服务已安装并运行:

install_nginx.yml

:

---- name: Install and configure Nginx web server  hosts: webservers  become: true # 需要root权限来安装和管理服务  tasks:    - name: Ensure Nginx is installed      ansible.builtin.apt: # 对于CentOS/RHEL使用ansible.builtin.dnf或ansible.builtin.yum        name: nginx        state: present        update_cache: yes # 更新apt缓存    - name: Ensure Nginx service is running and enabled at boot      ansible.builtin.service:        name: nginx        state: started        enabled: yes

保存这个文件,然后执行它:

ansible-playbook -i /path/to/your/hosts install_nginx.yml

-i

参数用来指定你的inventory文件路径,如果它不在默认位置。

执行后,Ansible会连接到

webservers

组中的所有主机,安装Nginx并确保它运行。这就是自动化运维的魅力所在,一次编写,多处执行。

Ansible与传统Shell脚本自动化有何优势?

坦白讲,刚接触自动化运维时,我曾一度觉得Shell脚本就够用了,毕竟它直接、灵活。但随着管理服务器数量的增加和任务复杂度的提升,Shell脚本的局限性就暴露无遗了。Ansible相比之下,优势简直是压倒性的。

首先,幂等性是Ansible一个核心且极其重要的概念。简单来说,就是无论你运行一个任务多少次,结果都将是一样的。比如,你要确保Nginx已安装,如果它已经装了,Ansible不会再尝试安装一次,而是直接告诉你“Nginx已存在”。而Shell脚本呢?你写一个

apt install nginx

,每次运行都会尝试安装,可能会报错,或者至少浪费资源。写一个幂等的Shell脚本,你需要加入大量的

if

判断和文件检查,这代码量和维护成本就上去了。我曾经花了好几个小时去调试一个因为重复执行而导致系统状态混乱的Shell脚本,那种痛苦,现在想起来都觉得不值。

其次,Agentless(无代理)架构。Ansible不需要在被管理端安装任何客户端软件,它仅仅通过标准的SSH协议连接过去执行任务。这意味着你的基础设施更轻量,没有额外的守护进程需要维护,也没有额外的端口需要开放。对于我这种喜欢简洁、不喜欢复杂部署的人来说,这简直是福音。很多其他的配置管理工具,比如Puppet或Chef,都需要在每台服务器上安装一个代理,这本身就是一项运维任务。

再者,可读性和模块化。Ansible Playbook使用YAML格式,这玩意儿比Shell脚本的各种语法和特殊字符要清晰得多,更容易阅读和理解。它内置了数以百计的模块,覆盖了文件操作、服务管理、包管理、用户管理、云资源操作等方方面面。这意味着你不需要自己去写复杂的命令来完成这些事,直接调用模块就行。比如,安装一个包,你不用去记

apt

yum

dnf

这些命令的区别,直接用

ansible.builtin.package

模块,Ansible会根据目标系统的类型自动选择正确的工具。这种抽象和模块化极大地提高了开发效率和可维护性。

最后,错误处理和报告。Ansible在执行Playbook时,会详细报告每个任务的执行状态,包括成功、失败、跳过、更改等。如果某个任务失败,它会停止并给出明确的错误信息。Shell脚本的错误处理就得你自己费心去写

set -e

trap

或者各种

if [ $? -ne 0 ]

,而且输出往往没那么友好。

总结来说,Shell脚本适合快速解决一次性、小规模的问题,但一旦涉及到规模化、可重复、状态管理和团队协作,Ansible的优势就体现得淋漓尽致了。它将你从繁琐、易错的重复劳动中解放出来,让你能更专注于更高价值的工作。

如何编写高效且可维护的Ansible Playbook?最佳实践与模块选择

编写Playbook不仅仅是把命令堆砌起来,更重要的是要考虑其效率、可读性和未来的可维护性。我的经验告诉我,遵循一些最佳实践能让你少走很多弯路。

1. 拥抱角色(Roles)

这是Playbook组织结构上最重要的一环。当你的自动化任务变得复杂,涉及多个应用、服务和配置时,一个巨大的Playbook会变得难以管理。角色允许你将相关的任务、变量、模板、文件和处理器(handlers)组织到一个独立的、可重用的结构中。

比如,一个

webserver

角色可能包含安装Nginx、配置虚拟主机、管理防火墙规则等所有与Web服务相关的任务。当我需要部署一个Web服务器时,我只需要在Playbook中调用这个角色,而不是复制代码块。这极大地提高了Playbook的模块化和复用性。

2. 优先使用模块,避免

command

shell

阶跃AI 阶跃AI

阶跃星辰旗下AI智能问答搜索助手

阶跃AI 291 查看详情 阶跃AI

Ansible提供了大量的模块来执行特定任务,比如

ansible.builtin.apt

ansible.builtin.service

ansible.builtin.copy

ansible.builtin.template

等等。这些模块是幂等的,并且通常会处理各种边缘情况和错误。

尽量避免直接使用

ansible.builtin.command

ansible.builtin.shell

模块,除非你确实没有找到合适的Ansible模块来完成你的任务。直接执行命令意味着你需要自己处理幂等性、错误检查和输出解析,这会增加Playbook的复杂性和出错的可能性。我见过太多Playbook因为过度依赖

shell

模块而变得难以维护和调试。

3. 充分利用变量

变量是Playbook灵活性的关键。不要在Playbook中硬编码IP地址、端口号、用户名等配置信息。将它们定义为变量,可以:

在Playbook中直接定义。在

group_vars

host_vars

文件中定义,实现分组和主机特定的配置。通过命令行参数传递(

--extra-vars

)。从外部文件导入。

这样,你的Playbook可以适应不同的环境(开发、测试、生产),而无需修改Playbook本身。比如,数据库密码、API密钥等敏感信息,就应该用Ansible Vault加密后作为变量使用。

4. 理解并善用处理器(Handlers)

处理器是当一个任务导致某个状态发生改变时才执行的特殊任务。最常见的例子是,当配置文件被修改后,你需要重启一个服务。你可以在修改配置文件的任务中通知一个处理器,让它在所有更改都完成后再重启服务。

- name: Copy Nginx config file  ansible.builtin.template:    src: nginx.conf.j2    dest: /etc/nginx/nginx.conf  notify: restart nginx # 通知处理器- name: restart nginx  ansible.builtin.service:    name: nginx    state: restarted  listen: restart nginx # 监听通知

这样可以避免不必要的服务重启,提高效率。

5. 标签(Tags)和条件(When)

标签(Tags):允许你只运行Playbook中的特定部分。比如,你可以给安装Nginx的任务打上

nginx_install

的标签,然后只执行

ansible-playbook your_playbook.yml --tags nginx_install

。这对于调试和选择性部署非常有用。条件(When):允许你根据特定条件来执行或跳过任务。比如,

when: ansible_os_family == "Debian"

可以确保某个任务只在Debian系系统上执行。

这些工具能让你的Playbook更加智能和灵活。

6. 保持Playbook的简洁和专注

一个Playbook或者一个角色应该专注于完成一个明确的目标。避免一个Playbook做太多不相关的事情。如果一个Playbook变得过于庞大,考虑将其拆分成多个更小的、更易于管理和测试的Playbook或角色。

Ansible进阶:动态清单、Vault与CI/CD集成实践

当你已经熟练掌握了Ansible的基础用法,并且管理着一个不断变化、规模庞大的基础设施时,就会发现一些进阶功能的重要性。动态清单、Ansible Vault以及与CI/CD流程的集成,是提升Ansible使用效率和安全性的关键。

1. 动态清单(Dynamic Inventory)

在云时代,服务器实例的生命周期往往很短,它们可能随时创建、销毁或改变IP地址。传统的静态

hosts

文件在这种场景下就显得力不从心了,你总不能手动去更新它。这就是动态清单发挥作用的地方。

动态清单通过执行一个外部脚本来获取服务器列表。这个脚本可以查询云服务提供商(如AWS EC2、Google Cloud Compute、Azure VM、VMware vCenter等)的API,或者你的CMDB系统,然后将结果以特定的JSON格式返回给Ansible。

例如,如果你在使用AWS EC2,Ansible官方提供了

aws_ec2.yml

作为动态清单插件。配置好后,Ansible就能自动发现你的EC2实例,并根据它们的标签、区域等属性自动分组。

# 假设你配置了AWS凭证ansible-inventory -i aws_ec2.yml --list

我个人觉得,对于任何基于云的基础设施,动态清单都是必不可少的。它将运维人员从繁琐的IP地址管理中解放出来,确保Ansible总能管理到最新的、正确的服务器集合。我曾经因为静态清单没有及时更新,导致Playbook运行在已经下线的服务器上,或者漏掉了新上线的服务器,那种错误真是让人头疼。

2. Ansible Vault:保护你的敏感数据

在Playbook中,我们经常需要处理数据库密码、API密钥、SSH私钥等敏感信息。直接将这些明文数据写在Playbook或变量文件中是极其危险的。Ansible Vault就是为了解决这个问题而生的。

Vault允许你加密文件、变量,甚至整个角色。这些加密的数据只有在提供正确的Vault密码后才能被Ansible解密和使用。

创建加密文件:

ansible-vault create secret.yml

这会提示你输入并确认一个密码,然后打开一个编辑器让你输入敏感数据。

编辑加密文件:

ansible-vault edit secret.yml

在Playbook中使用加密变量:你可以在Playbook中像使用普通变量一样引用Vault文件中的变量。执行Playbook时,通过

--ask-vault-pass

参数手动输入密码,或者使用

--vault-password-file

指定一个包含密码的文件(当然,这个文件本身也要妥善保管)。

# playbook.yml- name: Deploy application  hosts: appservers  vars_files:    - secret.yml # 引用加密的secret.yml文件  tasks:    - name: Configure database connection      ansible.builtin.template:        src: db_config.j2        dest: /etc/app/db_config.conf      vars:        db_password: "{{ app_db_password }}" # 从secret.yml中获取

Vault是生产环境中不可或缺的安全工具。任何时候处理敏感数据,都应该毫不犹豫地使用它。

3. 与CI/CD流程的集成

将Ansible集成到CI/CD(持续集成/持续部署)管道中,是实现真正DevOps的关键一步。它允许你自动化部署、配置管理和应用发布,从而加速开发周期,减少人为错误。

在CI阶段运行测试:你可以在CI管道中运行Ansible Playbook来配置一个测试环境,然后运行自动化测试。如果测试通过,则可以继续部署。

在CD阶段自动化部署:一旦代码通过了测试,CI/CD工具(如Jenkins, GitLab CI, GitHub Actions, CircleCI等)就可以触发Ansible Playbook来执行实际的部署操作。这可能包括:

更新服务器配置。部署新的应用代码。滚动更新服务。回滚到上一个稳定版本。

例如,在GitLab CI中,你的

.gitlab-ci.yml

文件可能包含一个阶段,该阶段使用

ansible-playbook

命令来执行部署:

deploy_production:  stage: deploy  script:    - ansible-playbook -i production_inventory.ini deploy_app.yml --vault-password-file ~/.ansible/vault_pass.txt  only:    - master  tags:    - ansible-runner

将Ansible与CI/CD结合,能将你的自动化能力提升到一个全新的水平。它不仅能确保部署的一致性和可靠性,还能显著缩短从代码提交到生产环境上线的时间。这种自动化流程的建立,是我个人在运维工作中感受最深,也觉得最有价值的变革之一。

当然,在实践中,你还会遇到各种挑战,比如SSH连接问题、权限配置错误、模块版本不兼容等等。这些都需要你在实践中不断积累经验,查阅文档,甚至深入理解Ansible的工作原理来解决。但总体来说,Ansible提供了一个强大且灵活的框架,足以应对绝大多数自动化运维场景。

以上就是如何在Linux下安装和使用Ansible?自动化运维的入门到进阶指南的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/412933.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月6日 23:58:29
下一篇 2025年11月6日 23:59:39

相关推荐

  • SASS 中的 Mixins

    mixin 是 css 预处理器提供的工具,虽然它们不是可以被理解的函数,但它们的主要用途是重用代码。 不止一次,我们需要创建多个类来执行相同的操作,但更改单个值,例如字体大小的多个类。 .fs-10 { font-size: 10px;}.fs-20 { font-size: 20px;}.fs-…

    2025年12月24日
    000
  • 如何解决本地图片在使用 mask JS 库时出现的跨域错误?

    如何跨越localhost使用本地图片? 问题: 在本地使用mask js库时,引入本地图片会报跨域错误。 解决方案: 要解决此问题,需要使用本地服务器启动文件,以http或https协议访问图片,而不是使用file://协议。例如: python -m http.server 8000 然后,可以…

    2025年12月24日
    200
  • 使用 Mask 导入本地图片时,如何解决跨域问题?

    跨域疑难:如何解决 mask 引入本地图片产生的跨域问题? 在使用 mask 导入本地图片时,你可能会遇到令人沮丧的跨域错误。为什么会出现跨域问题呢?让我们深入了解一下: mask 框架假设你以 http(s) 协议加载你的 html 文件,而当使用 file:// 协议打开本地文件时,就会产生跨域…

    2025年12月24日
    200
  • HTML、CSS 和 JavaScript 中的简单侧边栏菜单

    构建一个简单的侧边栏菜单是一个很好的主意,它可以为您的网站添加有价值的功能和令人惊叹的外观。 侧边栏菜单对于客户找到不同项目的方式很有用,而不会让他们觉得自己有太多选择,从而创造了简单性和秩序。 今天,我将分享一个简单的 HTML、CSS 和 JavaScript 源代码来创建一个简单的侧边栏菜单。…

    2025年12月24日
    200
  • 前端代码辅助工具:如何选择最可靠的AI工具?

    前端代码辅助工具:可靠性探讨 对于前端工程师来说,在HTML、CSS和JavaScript开发中借助AI工具是司空见惯的事情。然而,并非所有工具都能提供同等的可靠性。 个性化需求 关于哪个AI工具最可靠,这个问题没有一刀切的答案。每个人的使用习惯和项目需求各不相同。以下是一些影响选择的重要因素: 立…

    2025年12月24日
    000
  • 带有 HTML、CSS 和 JavaScript 工具提示的响应式侧边导航栏

    响应式侧边导航栏不仅有助于改善网站的导航,还可以解决整齐放置链接的问题,从而增强用户体验。通过使用工具提示,可以让用户了解每个链接的功能,包括设计紧凑的情况。 在本教程中,我将解释使用 html、css、javascript 创建带有工具提示的响应式侧栏导航的完整代码。 对于那些一直想要一个干净、简…

    2025年12月24日
    000
  • 布局 – CSS 挑战

    您可以在 github 仓库中找到这篇文章中的所有代码。 您可以在这里查看视觉效果: 固定导航 – 布局 – codesandbox两列 – 布局 – codesandbox三列 – 布局 – codesandbox圣杯 &#8…

    2025年12月24日
    000
  • 隐藏元素 – CSS 挑战

    您可以在 github 仓库中找到这篇文章中的所有代码。 您可以在此处查看隐藏元素的视觉效果 – codesandbox 隐藏元素 hiding elements hiding elements hiding elements hiding elements hiding element…

    2025年12月24日
    400
  • 居中 – CSS 挑战

    您可以在 github 仓库中找到这篇文章中的所有代码。 您可以在此处查看垂直中心 – codesandbox 和水平中心的视觉效果。 通过 css 居中 垂直居中 centering centering centering centering centering centering立即…

    2025年12月24日 好文分享
    300
  • 如何在 Laravel 框架中轻松集成微信支付和支付宝支付?

    如何用 laravel 框架集成微信支付和支付宝支付 问题:如何在 laravel 框架中集成微信支付和支付宝支付? 回答: 建议使用 easywechat 的 laravel 版,easywechat 是一个由腾讯工程师开发的高质量微信开放平台 sdk,已被广泛地应用于许多 laravel 项目中…

    2025年12月24日
    000
  • 如何在移动端实现子 div 在父 div 内任意滑动查看?

    如何在移动端中实现让子 div 在父 div 内任意滑动查看 在移动端开发中,有时我们需要让子 div 在父 div 内任意滑动查看。然而,使用滚动条无法实现负值移动,因此需要采用其他方法。 解决方案: 使用绝对布局(absolute)或相对布局(relative):将子 div 设置为绝对或相对定…

    2025年12月24日
    000
  • 移动端嵌套 DIV 中子 DIV 如何水平滑动?

    移动端嵌套 DIV 中子 DIV 滑动 在移动端开发中,遇到这样的问题:当子 DIV 的高度小于父 DIV 时,无法在父 DIV 中水平滚动子 DIV。 无限画布 要实现子 DIV 在父 DIV 中任意滑动,需要创建一个无限画布。使用滚动无法达到负值,因此需要使用其他方法。 相对定位 一种方法是将子…

    2025年12月24日
    000
  • 移动端项目中,如何消除rem字体大小计算带来的CSS扭曲?

    移动端项目中消除rem字体大小计算带来的css扭曲 在移动端项目中,使用rem计算根节点字体大小可以实现自适应布局。但是,此方法可能会导致页面打开时出现css扭曲,这是因为页面内容在根节点字体大小赋值后重新渲染造成的。 解决方案: 要避免这种情况,将计算根节点字体大小的js脚本移动到页面的最前面,即…

    2025年12月24日
    000
  • Nuxt 移动端项目中 rem 计算导致 CSS 变形,如何解决?

    Nuxt 移动端项目中解决 rem 计算导致 CSS 变形 在 Nuxt 移动端项目中使用 rem 计算根节点字体大小时,可能会遇到一个问题:页面内容在字体大小发生变化时会重绘,导致 CSS 变形。 解决方案: 可将计算根节点字体大小的 JS 代码块置于页面最前端的 标签内,确保在其他资源加载之前执…

    2025年12月24日
    200
  • Nuxt 移动端项目使用 rem 计算字体大小导致页面变形,如何解决?

    rem 计算导致移动端页面变形的解决方法 在 nuxt 移动端项目中使用 rem 计算根节点字体大小时,页面会发生内容重绘,导致页面打开时出现样式变形。如何避免这种现象? 解决方案: 移动根节点字体大小计算代码到页面顶部,即 head 中。 原理: flexível.js 也遇到了类似问题,它的解决…

    2025年12月24日
    000
  • 形状 – CSS 挑战

    您可以在 github 仓库中找到这篇文章中的所有代码。 您可以在此处查看 codesandbox 的视觉效果。 通过css绘制各种形状 如何在 css 中绘制正方形、梯形、三角形、异形三角形、扇形、圆形、半圆、固定宽高比、0.5px 线? shapes 0.5px line .square { w…

    2025年12月24日
    000
  • React 或 Vite 是否会自动加载 CSS?

    React 或 Vite 是否自动加载 CSS? 在 React 中,如果未显式导入 CSS,而页面却出现了 CSS 效果,这可能是以下原因造成的: 你使用的第三方组件库,例如 AntD,包含了自己的 CSS 样式。这些组件库在使用时会自动加载其 CSS 样式,无需显式导入。在你的代码示例中,cla…

    2025年12月24日
    000
  • 有哪些美观的开源数字大屏驾驶舱框架?

    开源数字大屏驾驶舱框架推荐 问题:有哪些美观的开源数字大屏驾驶舱框架? 答案: 资源包 [弗若恩智能大屏驾驶舱开发资源包](https://www.fanruan.com/resource/152) 软件 [弗若恩报表 – 数字大屏可视化组件](https://www.fanruan.c…

    2025年12月24日
    000
  • React 和 Vite 如何处理 CSS 加载?

    React 或 Vite 是否会自动加载 CSS? 在 React 中,默认情况下,使用 CSS 模块化时,不会自动加载 CSS 文件。需要手动导入或使用 CSS-in-JS 等技术才能应用样式。然而,如果使用了第三方组件库,例如 Ant Design,其中包含 CSS 样式,则这些样式可能会自动加…

    2025年12月24日
    000
  • ElementUI el-table 子节点选中后为什么没有打勾?

    elementui el-table子节点选中后没有打勾? 当您在elementui的el-table中选择子节点时,但没有出现打勾效果,可能是以下原因造成的: 在 element-ui 版本 2.15.7 中存在这个问题,升级到最新版本 2.15.13 即可解决。 除此之外,请确保您遵循了以下步骤…

    2025年12月24日
    200

发表回复

登录后才能评论
关注微信