深度学习与总结JVM专辑(二):垃圾回收基础(图文+代码)

news2025/1/13 9:36:44

垃圾回收

    • 基础概念
      • 什么是垃圾回收
      • 为什么要进行垃圾回收
    • 垃圾
      • 什么是垃圾
      • 你是垃圾吗?让我用算法来测测你。
        • 引用计数算法
        • 可达性分析算法
          • 对象引用
          • 对象,真的死了吗
    • 方法区回收
      • 废弃常量
      • 无用类
    • 垃圾回收算法
      • 回收类型
      • 分代收集理论
      • 标记清除算法(Mark-Sweep)
      • 标记-复制算法
      • 分代收集算法
    • 经典垃圾收集器
      • Serial收集器
      • Serial Old收集器
      • ParNew收集器
      • Paralled Old 收集器
      • Parallel Scavenge收集器
      • CMS收集器
      • G1收集器(Garbage-First)

基础概念

什么是垃圾回收

垃圾回收是JVM垃圾回收器提供的一种用在空闲时间不定时回收无任何对象引用的对象所占据的内存空间的一种机制。
垃圾收集器在一个Java程序汇总的执行是自动的,不能强制执行,我们能做的就是通过调用System.gc方法来建议执行垃圾收集器,但是到底是否可执行,什么时候执行都是不可知的,虽然这是它的缺点,但瑕不掩瑜。

为什么要进行垃圾回收

如果你了解过C++,你肯定明白对象所占的内存在程序结束前一直被占用,在明确释放之前不能分配给其他对象;
在Java中,垃圾回收能自动释放内存空间,减轻编程负担,JVM的一个系统级线程会自动释放该内存块。

这使得C++程序员最头疼的内存管理问题迎刃而解,而且因为垃圾回收机制,Java中的对象也不再有“对象域”的概念,只有对象的引用才有“对象域”。
此外,垃圾回收可以有效防止内存泄漏,有效的使用空间内存。
小段子,在餐厅里吃饭,吃完把餐盘端走清理,是C++程序员,而吃完直接走的,是Java程序员。

垃圾

先看一下内存区域
在这里插入图片描述

什么是垃圾

垃圾,这里指的是可以销毁的对象,其占有的空间是可以回收。
根据JVM架构划分,几乎所有的对象实例都在堆中存放,所以垃圾回收主要是针对堆来进行的。
JVM的眼中,垃圾就是指在堆中存在,但是已经“死亡”的对象。
而对于死亡的定义,我们简单理解为“不可能再被任何途径使用的对象”。
问题来了,你怎么证明对象是死是活呢?

你是垃圾吗?让我用算法来测测你。

JVM没有明确说使用哪种垃圾回收算法,但是任何一种垃圾回收算法一般要做2件基本的事情:
1:找到所有存活对象
2:回收被无用对象占用的内存空间,使该空间可被程序再次使用

引用计数算法

给对象添加一个引用计数器:
当对象增加一个引用时,计数器+1;
当对象失效一个引用时,计数器-1;
两个对象如果出现循环引用的情况,此时引用计数器永不为0,导致无法对它们进行回收。
所以因为有循环引用存在,JVM不使用引用计数算法(经典白雪)。
代码如下:

public class ReferenceCountingGC {

    public Object instance = null;

    public static void main(String[] args) {
        ReferenceCountingGC objectA = new ReferenceCountingGC();
        ReferenceCountingGC objectB = new ReferenceCountingGC();
        objectA.instance = objectB;
        objectB.instance = objectA;
    }
}

可达性分析算法

现代虚拟机基本都是采用这种算法来判断对象的存活。
通过叫做GC Roots的对象为起点出发,引出它们指向的下一个节点,再以下个节点为起点,引出此节点指向的下一个节点(这样通过GC Root串成的一条线就叫引用链),直到所有节点都遍历完毕,如果相关对象不在任意一个以GC Root为起点的引用链中,则这些对象都会被判断为垃圾,被GC回收。
在这里插入图片描述
那么有哪些对象有机会可以作为GC Roots,分为以下几种:
1.虚拟机栈(栈帧中本地变量表)中引用的对象,比如各个线程被调用的方法堆栈中使用到的参数,局部变量,临时变量等
2.在方法区中类静态变量引用的对象
3.在方法区中常量引用的对象
4.在本地方法中JNI(Native方法)引用的对象
5.在JVM内部的引用,如基本数据类型对应的Class对象,一些常驻异常对象及系统类加载器
6.所有被同步锁持有的对象
7.反应 Java 虚拟机内部情况的 JMXBean,JVMTI 中注册的回调,本地代码缓存等。
除了这些固定的GC Roots集合以外,根据用户所选用的垃圾收集器以及当前回收的内存区域不同,还可能会有其他对象“临时性”加入,共同构成完整的GC Roots集合。

对象引用

可达性分析是基于引用链进行判断的,在JDK1.2后,Java将引用关系分为以下四类:

  • 强引用(Strongly Reference):最传统的引用,如Object obj=new Object()。无论任何情况下,只要强引用关系还存在,垃圾收集器就永远不会回收掉被引用的对象。
  • 软引用(Soft Reference):用于描述一些还有用,但非必需的对象。只要被软引用关联着的对象,在系统将要发生内存溢出异常之前,会被列入回收范围进行第二次回收,如果这次回收后还没有足够的内存,才会抛出内存溢出异常。
  • 弱引用(Weak Reference):用于描述那些非必需的对象,强度比较比软引用弱。被弱引用关联对象只能生存到下一次垃圾收集发生时,无论当前内存是否足够,弱引用都会被回收。
  • 虚引用(Phantom Reference):最弱的引用关系。为一个对象设置虚引用关联的唯一目的只是为了能在这个对象被回收时收到一个系统通知。
对象,真的死了吗

要宣告一个对象死亡,需要经过至少两次标记过程:
1.如果对象进行可达性分析后发现GC Roots不可达,将会进行第一次标记;
2.随后进行一次筛选,条件是——此对象是否有必要执行finalized()方法。
如果对象没有覆盖finalized()方法,或者finalized方法以及被JVM调用过了,那么都会被视为没有必要执行。
如果判断结果是有必要执行,此时对象会被放入名为F-Queue的队列,收集器会进行第二次小规模的标记,如果对象在finalized()方法中重新将自己与引用链上的任何一个对象进行了关联,那么它完成了自我拯救,第二次标记会将其移除“即将回收”的集合,否则该对象就将被真正回收,走向死亡。

方法区回收

在Java堆上进行对象回收的性价比通常比较高,因为大多数对象都是朝生夕灭。
而方法区由于回收条件比较苛刻,对应的回收性价比通常比较低,主要回收两部分内容:
废弃常量和无用的类。

废弃常量

与堆中对象回收类似,以常量池字面量回收为例,如果字符串“abc”进入了常量池,但是没有任何一个String对象引用它,也没有任何地方引用了这个字面量,如果发生内存回收,有必要的话,这个常量会被系统清理出常量池。

无用类

  • 该类所有的实例都已经被回收了(堆中不存在该类的任何实例)
  • 加载该类的ClassLoader已经被回收
  • 该类对应的Java.lang.Class对象没有在任何地方被引用,无法在任何地方通过反射访问该类的方法。
    JVM可以对满足以上三个条件的无用类进行回收,但不是一定被回收。
    是否对类回收HotSpot提供了-Xnoclassgc参数进行控制。

垃圾回收算法

当前大多数虚拟机都遵循“分代收集”的理论进行设计,它建立在强弱两个分代假说下:

  • 弱分代假说(Weak Generational Hypothesis):绝大数对象都是朝生夕灭(IBM 公司的专业研究表明,有将近98%的对象是朝生夕死)。
  • 强分代假说(Strong Generational Hypothesis):熬过越多次垃圾收集过程的对象就越难以消亡。
  • 跨带引用假说(Intergenerational Reference Hypothesis):基于上面两条假说还可以得出一条隐含推论:存在相互引用关系的两个对象,应该倾向于同时生存或者同时消亡。

强弱分代假说奠定了垃圾收集器的设计原则:
收集器应该将Java堆划分不同的区域,然后将回收对象依据年龄(对象经历垃圾收集的次数)分配到不同的区域中进行存储。
如果一个区域中的对象都是朝生夕灭的,那么收集器只需要关注少量对象的存活而不是去标记那些大量要回收的对象,此时就能以较小代价获取较大空间。
最后再将难以消亡的对象集中到一起,根据强分代假说,它们很难消亡,所以JVM可以使用较低频率进行回收,兼顾了时间和内存空间的开销。
在这里插入图片描述

回收类型

根据分代收集理论,收集范围可以分为以下几种类型:

  • 部分收集(Partial GC):具体分为:
    新生代收集(Minor GC/Young GC):只对新生代进行垃圾收集;
    老年代收集(Major GC/Old GC):只对老年代进行垃圾收集。
    混合收集(Mixed GC):对整个新生代和部分老年代进行垃圾收集。
  • 整堆收集(Full GC):收集整个Java堆和方法区。

分代收集理论

我们通过了可达性算法来识别哪些数据是垃圾,那么如何高效的对这些垃圾回收呢?
由于JVM规范并没有对如何实现垃圾收集器做出明确的规定,因此各个厂商的虚拟机可以采用不同的方式来实现垃圾收集器,这里主要有下面几种方式。

标记清除算法(Mark-Sweep)

标记清除算法是基础垃圾回收算法,分为两部分:
1.先把内存区域中的这些对象标记(哪些属于可回收标记出来)
2.把这些垃圾拎出来清理掉。
在这里插入图片描述

缺点:
1.虽然逻辑十分清晰,但是会产生大量内存碎片,从而可能导致无法为大对象分配足够的连续内存。
2.执行效率不稳定,如果Java堆上包含大量需要回收的对象,则需要进行大量标记和清除动作。

标记-复制算法

标记-复制算法基于“半区复制”算法:
1.它将可用内存按容量划分为大小相等的两块,每次只使用其中一块。
2.当这一块的内存使用完了,就将还存活着的对象复制到另外一块上面。
3.将使用过的那块内存空间一次性清理掉。
优点:避免了内存空间碎片化的问题
缺点:
1.如果内存中多数对象都是存活的,这种算法将产生大量的复制开销;
2.浪费内存空间,内存空间变为了原有的一半。
在这里插入图片描述### 标记-整理算法
在标记完成之后,让所有存活对象都向内存的一端移动,然后直接清理掉边界以外的内存。
优点:可以避免内存空间碎片化,也可以充分利用内存空间;
缺点:
1.它对内存变动的更频繁,需要整理所有存活对象的引用地址,效率上比复制算法要差很多。
2.在移动存活对象可能要全程暂停用户程序。
在这里插入图片描述

分代收集算法

商业虚拟机采用分代手机算法,根据对象存活周期将内存划分为几块,不同块采用适当的收集算法,严格来说并不是一种思想或者理论,而是融合三种基础的算法思想,针对不同情况采用不同算法的一套组合拳。
一般将堆分为新生代和老年代。

  • 新生代使用:复制算法。
  • 老年代使用:标记-清除或者标记-整理算法。

经典垃圾收集器

并行和并发是并发编程中的专有名词,在谈论垃圾收集器的上下文语境中,它们的含义如下:

  • 并行(Parallel):并行描述的是多条垃圾收集器线程之间的关系,说明同一时间有多条这样的线程在协同工作,默认用户线程处于等待状态。
  • 并发(Concurrent):并发描述的是垃圾收集器线程与用户线程之间的关系,说明同一时间垃圾收集器线程和用户线程都在运行。但由于垃圾收集器线程会占一部分系统资源,所以程序的吞吐量依然会受到一定影响,除了CMS和G1之外,其他垃圾收集器都是以串行方式执行。
    下图中的连线代表了可以配合使用。
    在这里插入图片描述

Serial收集器

最基础,历史最悠久的收集器,进行垃圾回收时,必需暂停其他所有的工作线程,直到收集结束。
缺点:要暂停其他所有的工作线程,直到收集结束。
优点:单线程避免了多线程复杂的上下文切换,因此在单线程环境下收集效率非常高。
在这里插入图片描述

Serial Old收集器

它是Serial收集器的老年代版本,同样是一个单线程收集器,采用标记-整理算法,主要用于给客户端模式下的HotSpot虚拟机使用:
在这里插入图片描述

ParNew收集器

Serial收集器的多线程版本,可以使用多条线程进行垃圾回收:
在这里插入图片描述

Paralled Old 收集器

它是Parallel Scavenge收集器的老年代版本,支持多线程并发收集,采用标记-整理算法实现:
在这里插入图片描述

Parallel Scavenge收集器

与ParNew一样是多线程收集器
其他收集器的关注点在于尽可能缩短垃圾收集时用户现场的停顿时间,而它的目标是达到一个可控制的吞吐量,被称为“吞吐量优先”收集器。这里的吞吐量是指CPU用于运行用户代码的时间占总时间的比值
(吞吐量 = 运行用户代码时间 \ (运行用户代码时间 + 运行垃圾收集时间))
停顿时间越短就越适合需要与用户相互的程序,良好的响应速度能提升用户体验。
缩短停顿时间是以牺牲吞吐量和新生代空间来换取的:新生代空间越小,垃圾回收越频繁,导致吞吐量下降。
高吞吐量则可以高效率地利用CPU时间,尽快完成程序的运算任务,主要适合在后台运算而不需要太多交互的任务。

CMS收集器

它是一种以获取最短回收停顿时间为目标的收集器,基于标记-清除算法实现,整个收集过程分为以下四个阶段:
1.初始标记(inital mark):标记GC Roots能直接关联到的对象,耗时短但需要暂停用户线程;
2.并发标记(concurrent mark):从GC Roots能直接关联到对象开始遍历整个对象图,耗时长但不需要暂停用户线程。
3.重新标记(remark):采用增量更新算法,对并发标记阶段因为用户线程运行而产生变动的那部分对象进行重新标记,耗时比初始标记稍长且需要暂停用户线程;
4.并发清除(inital sweep):并发清楚掉已经死亡的对象,耗时长但不需要暂停用户线程。
在整个过程中耗时最长的并发标记和并发清除过程中,收集器线程都可以与用户线程一起工作,不需要进行停顿。
在这里插入图片描述
缺点:

  • 吞吐量低:低停顿时间是以牺牲吞吐量为代价,导致CPU利用率不够高。
  • 无法处理浮动垃圾:
    浮动垃圾是指并发清除阶段由于用户线程继续运行而产生的垃圾。
    这部分垃圾只能到下一次GC时才能进行回收。由于浮动垃圾的存在,因此需要预留出一部分内存,意味着CMS收集不能像其他收集器那样等待老年代快满时再回收。
    如果预留的内存不够存放浮动垃圾,就会出现Concurrent Mode Failure,这时虚拟机会临时启动Serial Old来替代CMS。
  • 标记-清除法导致的空间碎片:往往出现老年代空间剩余,但是无法找到足够大连续空间来分配当前对象,不得不提前触发一次Full GC。

G1收集器(Garbage-First)

面向服务端应用的垃圾收集器,在多CPU和大内存的场景下有很好的性能。HotSpot开发团队赋予它的使命是未来可以替换掉CMS收集器。

堆被分为新生代和老年代,其他收集器进行手机的范围是整个新生代或者老年代,而G1可以直接对新生代和老年代一起回收。

在这里插入图片描述
G1把堆划分为多个大小相等的独立区域,新生代和老年代不再物理隔离。
(H代表了大对象)
在这里插入图片描述

通过引入Region的概念,从而将原来的一整块内存空间划分为多个小空间,使得每个小空间可以进行单独的垃圾回收。
每个Region都可以根据不同的需求来扮演不同的新老年代空间。
这种划分带来了很大的灵活性,使得可预测的停顿事件模型成为可能。
通过记录每个Region垃圾回收时间以及回收所获得的空间(这两个值是通过过去回收的经验获得),维护一个优先列表,每次根据允许的收集时间,优先回收价值最大的Region。

每个Region都有一个Remembered Set,用来记录该Region对象的引用对象所在的Region。
通过使用Remembered Set ,在做可达性分析的时候就可以避免全堆扫描。

在不计算维护Remembered Set的操作,G1收集器的运作大致可划分为以下几个步骤:

  • 初始标记:标记GC Roots能直接关联到的对象,并且修改TAMS(Top at Mark Start)指针的值,让下一阶段用户线程并发运行时,能够正确的在Region中分配新对象。
    G1为每个Region都涉及了两个名为TAMS的指针,新分配的对象必须位于这两个指针位置以上,位于这两个指针位置以上的对象默认被隐式标记为存活,不会纳入回收范围。
  • 并发标记:从GC Roots能直接关联到的对象开始遍历整个对象图。
    遍历完成后,还需要处理SATB记录中变动的对象。SATB(snapshot-at-the-beginning,开始阶段快照)能够有效的解决并发标记阶段因为用户线程运行而导致的对象变动,其效率比CMS重新标记阶段所使用的增量更新算法效率更高。
  • 最终标记:为了修正并发标记期间用户程序继续运作而导致标记变动的那一部分标记记录,JVM将这段时间对象变化记录在线程的Remembered Set Logs里面,最终标记阶段需要把Remembered Set Logs的数据合并到Remembered Set中。这阶段需要停顿线程,但是可并行执行。
  • 筛选回收:首先对各个Region中的回收价值和成本进行排序,根据用户所期望的GC停顿时间来制定回收计划。此阶段其实也可以做到与用户程序一起并发执行,但是因为只回收一部分Region,时间是用户可控制的,而且停顿用户线程将大幅度提高收集效率。
    在这里插入图片描述

具备的特点:
空间整合:整体来看是基于“标记-整理”算法实现的收集器,从局部(两个Region之间)来看是基于复制算法实现,这意味着运行期间不会产生内存空间碎片。
可预测的停顿:能让使用者明确指定在一个长度为M毫秒的时间片段内,消耗在Gc上的时间不得超过N毫秒。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Hack The Box】linux练习-- Irked

HTB 学习笔记 【Hack The Box】linux练习-- Irked 🔥系列专栏:Hack The Box 🎉欢迎关注🔎点赞👍收藏⭐️留言📝 📆首发时间:🌴2022年11月17日🌴 &#x1f36…

[附源码]计算机毕业设计JAVA基于ssm的电子网上商城

[附源码]计算机毕业设计JAVA基于ssm的电子网上商城 项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM my…

基于FPGA的PID控制器开发与实现

欢迎订阅《FPGA学习入门100例教程》、《MATLAB学习入门100例教程》 效果预览: 目录 一、理论基础 二、核心程序 三、测试结果

数据结构与算法之顺序表详解

标题:猜数字小游戏 作者:Ggggggtm 寄语:与其忙着诉苦,不如低头赶路,奋路前行,终将遇到一番好风景 文章目录: 一、顺序表的概念与结构 1、线性表的解释 2、顺序表概念解释 二、顺序表的思路及代码…

ERP系统三种部署方式的区别

ERP系统被认为是一种 "企业应用程序",指的是为满足企业的软件需求和提高业务绩效而设计的软件。今天有许多不同的ERP系统可供使用,其范围很广,取决于企业的规模、功能和需求。ERP系统的类型根据其部署方式划分,包括云ER…

二、Redis分布式锁

一、什么是分布式锁 分布式锁是一种跨进程的,跨机器节点的一种互斥锁。保证在多个机器节点对共享资源访问的一个排他性。 分布式与单机情况下最大的不同在于分布式锁是多进程的而单机锁是单进程多线程的。 二、为什么需要分布式锁 与分布式锁相对就的是单机锁&…

BHQ-1 amine,1308657-79-5,BHQ染料通过FRET和静态猝灭的组合工作

英文名称:BHQ-1 amine 中文名称:BHQ-1 氨基 CAS:1308657-79-5 外观:深紫色粉末 分子式:C25H29N7O3 分子量:475.55 结构式: 溶解性:溶于大部分有机溶剂,溶于水 储存…

luffy-(13)

内容概览 支付宝支付介绍支付宝支付二次封装订单相关表设计生成订单接口支付前端支付宝回调接口 支付宝支付介绍 """ 项目中有需要在线支付功能,可以使用支付宝支付(沙箱环境)微信支付(需要有备案过的域名)云闪付我们的项目以支付宝支付为例支付流程使用官方…

【动手学深度学习】关于“softmax回归的简单实现”报错的解决办法(含源代码)

目录:关于“softmax回归的简单实现”报错的解决办法一、前言二、实现步骤2.1 导包2.2 初始化模型参数2.3 重新审视Softmax的实现2.4 优化算法2.5 训练2.6 源代码三、问题出现四、问题的解决五、再跑代码六、改正后的函数源代码一、前言 在之前的学习中,…

题库系统(公众号免费调用)

题库系统(公众号免费调用) 本平台优点: 多题库查题、独立后台、响应速度快、全网平台可查、功能最全! 1.想要给自己的公众号获得查题接口,只需要两步! 2.题库: 查题校园题库:查题校园题库后台&#xff0…

arthas 源码构建

arthas 源码构建 git下载代码 git clone https://github.com/alibaba/arthas.git 若github被墙,可以在gitee搜索下载 maven clean 可以在项目目录执行 mvn clean , ide可以执行界面执行 maven package 可以在项目目录执行mvn package 问题记录 ja…

四肽Suc-AAPD-对硝基苯胺,165174-58-3

粒酶B底物succ - aapd - pna。也被ICE劈开了。 编号: 177581中文名称: 四肽Suc-Ala-Ala-Pro-Asp-对硝基苯胺CAS号: 165174-58-3单字母: Suc-AAPD-pNA三字母: Suc-Ala-Ala-Pro-Asp-pNA氨基酸个数: 4分子式: C25H32O11N6平均分子量: 592.56精确分子量: 592.21等电点(PI): -pH7.0时…

新进场的獴哥健康、至真健康们,讲不出互联网医疗的新故事

文丨智能相对论 作者丨沈浪 曾几何时,互联网医疗风靡一时,现如今潮水退去,当市场回归理性,赛道竞争趋于同质化,一批互联网医疗企业正在试图通过讲好新故事,来拉开品牌与品牌之间的商业差距,寻…

ArcGIS Pro 转换Smart3D生成的倾斜3D模型数据osgb——创建集成网格场景图层包

最近在做Arcgis 批处理的一些工作,然后再学习Python的同时,偶然觉得arcgis Pro是个好东西呢?然后结合近期的Smart3D倾斜3D模型数据,是否可以在arcgis里查看呢?带着这样的疑问和好奇,开始了arcgis Pro的学习…

【408专项篇】C语言笔记-第四章(选择与循环)

第四章:选择、循环 第一节:选择if-else 1. 关系表达式与逻辑表达式 if-else的判断条件结果还是真与假,即1或0,一般是关系表达式或逻辑表达式。 算术运算符的优先级高于关系运算符,关系运算符的优先级高于逻辑与和逻…

Webpack 5 超详细解读(四)

31.proxy 代理设置 为什么开发阶段需要设置代理,在开发阶段,我们需要请求后端接口,但是一般后端接口地址和我们本地的不在同一个服务中提供,这时进行访问就会存在跨域的问题,所以我们需要对我们的请求进行转啊操作。模拟跨域请求代码如下: https://api.github.com/users…

高项 沟通管理论文

3个过程: 1,规划沟通管理:根据干系人的信息需要和要求及组织的可用资产情况,制订合适的项目沟通方式和计划的过程。 2,管理沟通:根据沟通管理计划,生成、收集、分发、储存、检索及最终处置项目信息的过程…

Spring boot 实践Rabbitmq消息防丢失

之前看很多网上大佬的防丢失的文章,文章中理论知识偏多,所以自己想着实践一下,实践过程中也踩了一些坑,因此写出了这篇文章。如果文章有误人子弟的地方,望在评论区指出。 导致消息出现丢失的原因 发送时失败&#xff…

295348-87-7,AF 594 Succinimidyl Ester可用于成像和流式细胞分析

理论分析: 中文名:AF 594活性酯 英文名:AF 594 Succinimidyl Ester,Alexa Fluor 594 NHS Ester,AF 594 NHS Ester CAS号:295348-87-7 化学式:C39H37N3O13S2 分子量:819.85 ex/em : 5…

招投标业务总结

最近接了一个招标投标的项目,开发完成后,整个招投标的流程也就理清楚了,简介一下业务过程。业务主流程可以分为3个阶段: 招标方建立招标项目发布招标公告投标人参与竞标招标方开标评选公示 系统可以划分为两套,一个是给招标方使…