当前位置: 首页 > news >正文

Linux如何将文件或目录打成rpm包?-- rpmbuild打包详解

👨‍🎓博主简介

  🏅CSDN博客专家
  🏅云计算领域优质创作者
  🏅华为云开发者社区专家博主
  🏅阿里云开发者社区专家博主
💊交流社区:运维交流社区 欢迎大家的加入!
🐋 希望大家多多支持,我们一起进步!😄
🎉如果文章对你有帮助的话,欢迎 点赞 👍🏻 评论 💬 收藏 ⭐️ 加关注+💗


文章目录

  • 前言
  • 一、使用rpmbuild将python环境及依赖包打成rpm包
    • 1.1 准备打包环境
    • 1.2 创建 Python 环境的 SPEC 文件
    • 1.3 准备 Python 环境
    • 1.4 构建 RPM 包
    • 1.5 安装、卸载 RPM 包
      • 1.5.1 安装rpm包
      • 1.5.2 查找刚刚安装rpm包
      • 1.5.3 卸载刚刚安装的rpm包
  • 二、使用rpmbuild将docker离线安装包打成rpm包
    • 2.1 准备打包环境
    • 2.2 创建 docker 安装包的 SPEC 文件
    • 2.3 准备 docker安装包 环境
    • 2.4 构建 RPM 包
    • 2.5 安装、卸载 RPM 包
      • 2.5.1 安装rpm包
      • 2.5.2 查找刚刚安装rpm包
      • 2.5.3 卸载刚刚安装的rpm包
  • 三、相关文章

前言

  最近因为遇到一个服务器受系统限制,只能上传rpm包才能使用,而我们的服务都是文件,那么怎么将文件都打成rpm包呢???我也是找了好几个,终于找到了一个简单好用的打包方式,下面来给大家讲解一下部署及打包、安装;

rpm是什么呢?

  rpm是一种安装包的格式。就像在Windows系统上我们常见的安装包格式是exe和msi一样,在linux上常见的安装包格式是deb和rpm。一般在红帽系列的系统上,不支持deb,所以我们需要将程序打包成rpm安装。

本文已docker离线安装包python环境两种类型文件打包为例子,将其打成rpm包,并在新的服务器上安装;

操作系统为:Centos 7.6、中科方德开发版系统

一、使用rpmbuild将python环境及依赖包打成rpm包

在使用 rpmbuild 将 Python 环境打包成 RPM 包之前,需要确保已经安装 rpmbuild 工具,并配置好相关的 RPM 打包环境。以下是具体的步骤和说明:

1.1 准备打包环境

首先,确保安装了 rpmbuild,可以通过以下命令安装:

sudo yum install rpm-build -y
# 安装 rpmdevtools
yum install rpmdevtools

rpmdevtools为开发者提供了极大的便利,用于制作RPM包,涵盖了从创建rpmbuild开发树、MD5验证、spec文件的生成,到解压归档文件、归档文件前后版本对比diff等一系列功能。

创建打包文件夹

创建打包文件夹有两种方法:

方法一:使用命令 rpmdev-setuptree 来创建,但是创建的文件夹是在用户主目录(home)如果是root就是在(root)下,我们可以将rpmbuild整个文件夹拷贝到项目文件夹,这样可以方便我们项目管理。

方法二:手动的去创建文件夹,具体文件夹结构可以参考下面的目录树

# 方法一:使用命令创建
rpmdev-setuptree# 方法二:手动创建文件夹
mkdir -p ~/rpmbuild/{BUILD,RPMS,SOURCES,SPECS,SRPMS}# 目录树
rpmbuild- BUILD           // 编译时用到的暂存目录- RPMS            // 打包后生成的 rpm 包会放在这里- SOURCES         // 源码压缩包- SPECS           // 放 xx.spec 文件- SRPMS           // 打包后生成的 srpm 包会放在这里 

1.2 创建 Python 环境的 SPEC 文件

~/rpmbuild/SPECS 目录中创建一个 .spec 文件,用于定义 RPM 包的相关信息,例如 python_env.spec。以下是一个基本的 spec 文件模板:

或者使用rpmdev-newspec python_env.spec命令创建*.spec模板

Name: python_env
Version: 1.0
Release: 1%{?dist}
Summary: Python Environment PackageLicense: YourLicense
Source0: %{name}-%{version}.tar.gz%description
This package provides a Python environment including dependencies.%prep
# 静默模式解压,并进入解压后的目录,常用:%setup -q%build
# 编译过程%install
# 安装过程
# 假设你已经将Python环境打包为.tar.gz文件,将其解压到 /opt/
tar -xzvf %{SOURCE0} -C %{buildroot}/opt/%files
/opt/python_env/*%changelog
* Tue Nov 12 2024 Your Name <youremail@example.com> - 1.0-1
- Initial package for Python environment

以下是这个 RPM SPEC 文件的配置内容的详细解析:


基本配置信息

  • Name: python_env

    • 定义了 RPM 包的名称,这里指定为 python_env
  • Version: 1.0

    • 指定了 RPM 包的版本号。在后续版本更新中,可以根据需要调整这个数字。
  • Release: 1%{?dist}

    • 定义了 RPM 包的发布版本,1 表示初次发布,%{?dist} 会自动添加当前的发布版本分发标记(如 .el7 表示适用于 CentOS 7)。
  • Summary: Python Environment Package

    • 这是一个简短的描述,用于概括 RPM 包的内容。
  • License: YourLicense

    • 指定了软件包的许可证。这里应替换为实际的许可证名称(如 MITGPL 等)。
  • Source0: %{name}-%{version}.tar.gz

    • 定义了源代码文件的名称,通常与 SPEC 文件同名,格式为 Name-Version.tar.gz%{name}%{version} 会分别替换为 python_env1.0,因此实际值为 python_env-1.0.tar.gz

包的详细描述

  • %description
    • 该字段提供了包的详细描述,将会显示在 RPM 的信息中。这里说明此包包含了 Python 环境及其依赖项。

构建和安装过程

  • %prep

    • 这个部分用于准备构建环境,通常会执行一些解压缩或预处理操作。此例中,注释提到可以在此处解压环境包,但没有实际命令。
  • %build

    • 定义了构建过程中需要执行的操作,例如编译代码。但对于不需要编译的包,这部分可以为空。
  • %install

    • 在安装阶段中,指定了将内容安装到目标目录的操作。
    • tar -xzvf %{SOURCE0} -C %{buildroot}/opt/python/:将 Source0(即 python_env-1.0.tar.gz)解压到安装路径中,以便包含完整的 Python 环境。

包含的文件

  • %files
    • 指定在安装包中包含哪些文件。
    • /opt/python_env/*:将 /opt/python_env/ 中的所有内容添加到最终生成的 RPM 包中。

变更日志

  • %changelog
    • 记录包的变更历史。每条日志包括日期、作者、版本及更新说明。在此例中:
      • 日期是 2024 年 11 月 12 日
      • 作者是 Your Name <youremail@example.com>
      • 更新描述是 “Initial package for Python environment”,表示首次打包。

这个 SPEC 文件的作用是将 Python 环境打包为一个 RPM,安装时会解压到指定目录,并可方便安装在其他系统上。

1.3 准备 Python 环境

可以使用 virtualenvvenv 来创建 Python 虚拟环境,并安装所需的库,然后将整个环境打包为 .tar.gz 文件,例如:

# 创建python3虚拟环境
python3 -m venv python_env【要创建的虚拟环境名称(也就是到时候打包的名称)】
# 激活python3虚拟环境
source python_env/bin/activate
# 然后就可以在虚拟环境里安装需要的东西了。

安装完之后,就可以退出虚拟环境打包了;

# 打包python虚拟环境格式为:包名-版本.tar.gz(版本为spec文件里写的版本)
tar -czvf python_env-1.0.tar.gz python_env
# 将打好的包移动到rpmbuild/SOURCES/下,用于打包rpm文件
mv python_env-1.0.tar.gz ~/rpmbuild/SOURCES/

1.4 构建 RPM 包

使用以下命令在 rpmbuild 环境中创建 RPM 包:

rpmbuild -ba ~/rpmbuild/SPECS/python_env.spec

执行后,生成的 RPM 包会保存在 ~/rpmbuild/RPMS/noarch/ 目录中。

noarch是根据你系统生成的,x86的就是x86_64

1.5 安装、卸载 RPM 包

首先要将打的rpm包上传到服务器上;

1.5.1 安装rpm包

默认安装路径是你写的SPEC文件里的路径【本文是/opt/】

rpm -ivh python_env-1.0-1.noarch.rpm
# 强制安装
rpm -ivh python_env-1.0-1.noarch.rpm --nodeps

1.5.2 查找刚刚安装rpm包

rpm -qa | grep python_env-1.0

1.5.3 卸载刚刚安装的rpm包

rpm -e python_env-1.0-1.x86_64
# 强制卸载
rpm -e python_env-1.0-1.x86_64 --nodeps

二、使用rpmbuild将docker离线安装包打成rpm包

在使用 rpmbuilddocker离线安装包打包成 RPM 文件之前,需要确保已经安装 rpmbuild 工具,并配置好相关的 RPM 打包环境。以下是具体的步骤和说明:

2.1 准备打包环境

首先,确保安装了 rpmbuild,可以通过以下命令安装:

sudo yum install rpm-build -y
# 安装 rpmdevtools
yum install rpmdevtools

rpmdevtools为开发者提供了极大的便利,用于制作RPM包,涵盖了从创建rpmbuild开发树、MD5验证、spec文件的生成,到解压归档文件、归档文件前后版本对比diff等一系列功能。

创建打包文件夹

创建打包文件夹有两种方法:

方法一:使用命令 rpmdev-setuptree 来创建,但是创建的文件夹是在用户主目录(home)如果是root就是在(root)下,我们可以将rpmbuild整个文件夹拷贝到项目文件夹,这样可以方便我们项目管理。

方法二:手动的去创建文件夹,具体文件夹结构可以参考下面的目录树

# 方法一:使用命令创建
rpmdev-setuptree# 方法二:手动创建文件夹
mkdir -p ~/rpmbuild/{BUILD,RPMS,SOURCES,SPECS,SRPMS}# 目录树
rpmbuild- BUILD           // 编译时用到的暂存目录- RPMS            // 打包后生成的 rpm 包会放在这里- SOURCES         // 源码压缩包- SPECS           // 放 xx.spec 文件- SRPMS           // 打包后生成的 srpm 包会放在这里 

2.2 创建 docker 安装包的 SPEC 文件

~/rpmbuild/SPECS 目录中创建一个 .spec 文件,用于定义 RPM 包的相关信息,例如 docker.spec。以下是一个基本的 spec 文件模板:

或者使用rpmdev-newspec docker.spec命令创建*.spec模板

Name: docker
Version: 24.0.5
Release: 1%{?dist}
Summary: docker install PackageLicense: MIT
# 尽量不要使用中文名
Source0: %{name}%{version}.zip%description
This install docker package.%prep
# 这里可以解压环境包%build
# 打包需要的构建操作可以在这里定义%install
# 假设你已经将把docker安装包已经打好了为.zip文件,将其解压到 /opt/
unzip  %{SOURCE0} -d %{buildroot}/opt/%files
/opt/docker24.0.5/*%changelog
* Tue Nov 14 2024 Your Name <youremail@example.com> - 1.0-1
- Initial package for Python environment

2.3 准备 docker安装包 环境

将docker的安装包放到rpmbuild/SOURCES/

# 复制过去
cp -ar docker24.0.5.zip ~/rpmbuild/SOURCES/
# 或者移动过去
mv docker24.0.5.zip ~/rpmbuild/SOURCES/

2.4 构建 RPM 包

使用以下命令在 rpmbuild 环境中创建 RPM 包:

rpmbuild -ba ~/rpmbuild/SPECS/docker.spec

执行后,生成的 RPM 包会保存在 ~/rpmbuild/RPMS/noarch/ 目录中。

noarch是根据你系统生成的,x86的就是x86_64

2.5 安装、卸载 RPM 包

首先要将打的rpm包上传到服务器上;

2.5.1 安装rpm包

默认安装路径是你写的SPEC文件里的路径【本文是/opt/】

rpm -ivh docker-24.0.5-1.nfs.x86_64.rpm
# 强制安装
rpm -ivh docker-24.0.5-1.nfs.x86_64.rpm --nodeps

2.5.2 查找刚刚安装rpm包

rpm -qa | grep docker-24.0.5

2.5.3 卸载刚刚安装的rpm包

rpm -e docker-24.0.5-1.nfs.x86_64
# 强制卸载
rpm -e docker-24.0.5-1.nfs.x86_64 --nodeps

三、相关文章

文章标题文章链接
Linux如何将文件或目录打成rpm包? – fpm打包详解https://blog.csdn.net/liu_chen_yang/article/details/134270559
Linux如何将文件或目录打成rpm包?-- rpmbuild打包详解https://blog.csdn.net/liu_chen_yang/article/details/143715019

🐋 希望大家多多支持,我们一起进步!😄
🎉如果文章对你有帮助的话,欢迎 点赞 👍🏻 评论 💬 收藏 ⭐️ 加关注+💗

相关文章:

Linux如何将文件或目录打成rpm包?-- rpmbuild打包详解

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;CSDN博客专家   &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01…...

RabbitMQ-死信队列(golang)

1、概念 死信&#xff08;Dead Letter&#xff09;&#xff0c;字面上可以理解为未被消费者成功消费的信息&#xff0c;正常来说&#xff0c;生产者将消息放入到队列中&#xff0c;消费者从队列获取消息&#xff0c;并进行处理&#xff0c;但是由于某种原因&#xff0c;队列中的…...

爬虫开发工具与环境搭建——环境配置

第二章&#xff1a;爬虫开发工具与环境搭建 第二节&#xff1a;环境配置 在进行爬虫开发之前&#xff0c;首先需要配置好开发环境。一个良好的开发环境不仅能提高开发效率&#xff0c;还能避免因环境不一致带来的问题。以下是环境配置的详细步骤&#xff0c;涵盖了Python开发…...

15.UE5等级、经验、血条,魔法恢复和消耗制作

2-17 等级、经验、血条、魔法消耗_哔哩哔哩_bilibili 目录 1.制作UI&#xff0c;等级&#xff0c;经验&#xff0c;血条 ​2.为属性面板绑定角色真实的属性&#xff0c;实现动态更新 3.魔法的消耗和恢复 1.制作UI&#xff0c;等级&#xff0c;经验&#xff0c;血条 创建控…...

【Homework】【5】Learning resources for DQ Robotics in MATLAB

Lesson 5 代码-TwoDofPlanarRobot.m 表示一个 2 自由度平面机器人。该类包含构造函数、计算正向运动学模型的函数、计算平移雅可比矩阵的函数&#xff0c;以及在二维空间中绘制机器人的函数。 classdef TwoDofPlanarRobot%TwoDofPlanarRobot - 表示一个 2 自由度平面机器人类…...

vue3中 ref和reactive的区别

ref的主要作用 ref 函数接受的参数数据类型可以是原始数据类型也可以是引用数据类型。在模板中使用 ref 时&#xff0c;我们不需要加 .value&#xff0c;因为当 ref 在模板中作为顶层属性被访问时&#xff0c;它们会被自动解包&#xff0c; <p>count: {{ count }}</…...

第十四章 Spring之假如让你来写AOP——雏形篇

Spring源码阅读目录 第一部分——IOC篇 第一章 Spring之最熟悉的陌生人——IOC 第二章 Spring之假如让你来写IOC容器——加载资源篇 第三章 Spring之假如让你来写IOC容器——解析配置文件篇 第四章 Spring之假如让你来写IOC容器——XML配置文件篇 第五章 Spring之假如让你来写…...

群控系统服务端开发模式-应用开发-前端个人资料开发

一、总结 其实程序开发到现在&#xff0c;简单的后端框架就只剩下获取登录账号信息及获取登录账号菜单这两个功能咯。详细见下图&#xff1a; 1、未登录时总业务流程图 2、登录后总业务流程图 二、获取登录账号信息对接 在根目录下src文件夹下store文件夹下modules文件夹下的us…...

动态规划技巧点

动规五部曲&#xff08;来自b站卡哥&#xff09;&#xff1a;1、确定DP数组中i、j…的含义。2、确定DP推导式。3、DP数组初始化。4、DP数组遍历顺序。5、DP数组打印校验。 父问题、子问题有些许区别&#xff1a;LeetCode343.整数拆分 今天在哔哩哔哩上刷到了一个非常有意思的l…...

深度学习之pytorch常见的学习率绘制

文章目录 0. Scope1. StepLR2. MultiStepLR3. ExponentialLR4. CosineAnnealingLR5. ReduceLROnPlateau6. CyclicLR7. OneCycleLR小结参考文献 https://blog.csdn.net/coldasice342/article/details/143435848 0. Scope 在深度学习中&#xff0c;学习率&#xff08;Learning R…...

Spring Boot集成SQL Server快速入门Demo

1.什么是SQL Server&#xff1f; SQL Server是由Microsoft开发和推广的以客户/服务器&#xff08;c/s&#xff09;模式访问、使用Transact-SQL语言的关系数据库管理系统&#xff08;DBMS&#xff09;&#xff0c;它最初是由Microsoft、Sybase和Ashton-Tate三家公司共同开发的&…...

低代码牵手 AI 接口:开启智能化开发新征程

一、低代码与 AI 接口的结合趋势 低代码开发平台近年来在软件开发领域迅速崛起。随着企业数字化转型的需求不断增长&#xff0c;低代码开发平台以其快速构建应用程序的优势&#xff0c;满足了企业对高效开发的需求。例如&#xff0c;启效云低代码平台通过范式化和高颗粒度的可配…...

【已解决】git push一直提示输入用户名及密码、fatal: Could not read from remote repository的问题

问题描述&#xff1a; 在实操中&#xff0c;git push代码到github上一直提示输入用户名及密码&#xff0c;并且跳出的输入框输入用户名和密码后&#xff0c;报错找不到远程仓库 实际解决中&#xff0c;发现我环境有两个问题解决&#xff1a; git push一直提示输入用户名及密码…...

python语言基础-4 常用模块-4.13 其他模块

声明&#xff1a;本内容非盈利性质&#xff0c;也不支持任何组织或个人将其用作盈利用途。本内容来源于参考书或网站&#xff0c;会尽量附上原文链接&#xff0c;并鼓励大家看原文。侵删。 4.13 其他模块 除此之外python中还有大量的功能模块&#xff0c;如&#xff1a; pill…...

微信小程序=》基础=》常见问题=》性能总结

文章目录 微信小程序开发应用 实例小程序生命周期 以及 各生命周期应用实例小程序图片 展示方案 小程序打包应用方案技术细节&#xff08;分包应用实例&#xff09;技术细节&#xff08;压缩处理&#xff09;一、准备工作二、JavaScript 代码压缩三、WXML 文件优化&#xff08…...

JWT深度解析:Java Web中的安全传输与身份验证

标题&#xff1a;JWT深度解析&#xff1a;Java Web中的安全传输与身份验证 引言 JSON Web Token&#xff08;JWT&#xff09;是一种轻量级的身份验证和授权标准&#xff0c;它允许在各方之间安全地传输信息。在Java Web开发中&#xff0c;JWT因其无状态、可扩展性和跨域支持而…...

使用Java爬虫获取商品订单详情:从API到数据存储

在电子商务日益发展的今天&#xff0c;获取商品订单详情成为了许多开发者和数据分析师的需求。无论是为了分析用户行为&#xff0c;还是为了优化库存管理&#xff0c;订单数据的获取都是至关重要的。本文将详细介绍如何使用Java编写爬虫&#xff0c;通过API获取商品订单详情&am…...

Mybatis中批量插入foreach优化

数据库批量入库方常见方式&#xff1a;Java中foreach和xml中使用foreach 两者的区别&#xff1a; 通过Java的foreach循环批量插入&#xff1a; 当我们在Java通过foreach循环插入的时候&#xff0c;是一条一条sql执行然后将事物统一交给spring的事物来管理&#xff08;Transa…...

Word VBA如何间隔选中多个(非连续)段落

实例需求&#xff1a;Word文档中的有多个段落&#xff0c;段落总数量不确定&#xff0c;现在需要先选中所有基数段落&#xff0c;即&#xff1a;段落1&#xff0c;段落3 … &#xff0c;然后一次性设置粗体格式。 也许有的读者会认为这个无厘头的需求&#xff0c;循环遍历遍历文…...

Linux系统常用操作与命令指南

一、快捷分类 1、移动光标 h&#xff0c; j&#xff0c; k&#xff0c; l 左&#xff0c; 下&#xff0c; 上&#xff0c; 右 Ctrl-F:下翻一页 Ctrl-B:上翻一页 Ctrl-U:上翻半页 Ctrl-d:下翻半页 0:跳至行首&#xff0c;不管有无缩进&#xff0c;就是跳到第0个字…...

大数据学习栈记——Neo4j的安装与使用

本文介绍图数据库Neofj的安装与使用&#xff0c;操作系统&#xff1a;Ubuntu24.04&#xff0c;Neofj版本&#xff1a;2025.04.0。 Apt安装 Neofj可以进行官网安装&#xff1a;Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...

论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(二)

HoST框架核心实现方法详解 - 论文深度解读(第二部分) 《Learning Humanoid Standing-up Control across Diverse Postures》 系列文章: 论文深度解读 + 算法与代码分析(二) 作者机构: 上海AI Lab, 上海交通大学, 香港大学, 浙江大学, 香港中文大学 论文主题: 人形机器人…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

无法与IP建立连接,未能下载VSCode服务器

如题&#xff0c;在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈&#xff0c;发现是VSCode版本自动更新惹的祸&#xff01;&#xff01;&#xff01; 在VSCode的帮助->关于这里发现前几天VSCode自动更新了&#xff0c;我的版本号变成了1.100.3 才导致了远程连接出…...

关于 WASM:1. WASM 基础原理

一、WASM 简介 1.1 WebAssembly 是什么&#xff1f; WebAssembly&#xff08;WASM&#xff09; 是一种能在现代浏览器中高效运行的二进制指令格式&#xff0c;它不是传统的编程语言&#xff0c;而是一种 低级字节码格式&#xff0c;可由高级语言&#xff08;如 C、C、Rust&am…...

多种风格导航菜单 HTML 实现(附源码)

下面我将为您展示 6 种不同风格的导航菜单实现&#xff0c;每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...

何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡

何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡 背景 我们以建设星云智控官网来做AI编程实践&#xff0c;很多人以为AI已经强大到不需要程序员了&#xff0c;其实不是&#xff0c;AI更加需要程序员&#xff0c;普通人…...

ArcPy扩展模块的使用(3)

管理工程项目 arcpy.mp模块允许用户管理布局、地图、报表、文件夹连接、视图等工程项目。例如&#xff0c;可以更新、修复或替换图层数据源&#xff0c;修改图层的符号系统&#xff0c;甚至自动在线执行共享要托管在组织中的工程项。 以下代码展示了如何更新图层的数据源&…...

深度解析云存储:概念、架构与应用实践

在数据爆炸式增长的时代&#xff0c;传统本地存储因容量限制、管理复杂等问题&#xff0c;已难以满足企业和个人的需求。云存储凭借灵活扩展、便捷访问等特性&#xff0c;成为数据存储领域的主流解决方案。从个人照片备份到企业核心数据管理&#xff0c;云存储正重塑数据存储与…...

mcts蒙特卡洛模拟树思想

您这个观察非常敏锐&#xff0c;而且在很大程度上是正确的&#xff01;您已经洞察到了MCTS算法在不同阶段的两种不同行为模式。我们来把这个关系理得更清楚一些&#xff0c;您的理解其实离真相只有一步之遥。 您说的“select是在二次选择的时候起作用”&#xff0c;这个观察非…...