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

【排序算法】八大排序(上)(c语言实现)(附源码)

🌟🌟作者主页:ephemerals__

🌟🌟所属专栏:算法

目录

前言

写一串测试数据

交换两元素的函数

一、冒泡排序

二、选择排序

三、插入排序

四、希尔排序

程序全部代码

总结


前言

        排序算法是计算机科学领域的基石之一,它不仅在算法的理论研究中占据重要地位,更是实际开发当中解决数据组织,检索,处理等问题的关键工具。现如今数据日益增长,理解并掌握这些排序算法的原理、特点及其适用场景,对于提升程序效率、优化用户体验至关重要。

        八大排序算是排序算法当中知名度较高的了,它们不仅涵盖了简单直观的排序方法,以便初学者学习理解;也包含了高效复杂的排序策略,广泛应用于实际开发。

        本篇文章,作者主要介绍并实现八大排序算法的其中四种:冒泡排序、选择排序、插入排序、希尔排序

正文开始

写一串测试数据

        首先,我们写一个乱序的数组,便于后续排序测试:

#include <stdio.h>int main()
{int arr[] = { 5,9,4,0,2,7,8,0,0,1,4,4,1,3,5,6,3,2,9,7 };int sz = sizeof(arr) / sizeof(arr[0]);//计算出数组元素个数for (int i = 0; i < sz; i++)//打印数组{printf("%d ", arr[i]);}return 0;
}

交换两元素的函数

        在这里,我们将交换两个元素的步骤封装成一个函数,便于后续多次调用。

void Swap(int* x, int* y)
{int tmp = *x;*x = *y;*y = tmp;
}

一、冒泡排序

        冒泡排序是一种简单的排序算法,易于初学者理解和学习。它的核心思想就是重复遍历数组,比较相邻两个元素,如果它们的顺序错误,则交换之。直到数组中没有顺序错误的情况,则排序已经完成

        它的具体步骤描述如下:

1.遍历数组,比较所有的相邻元素,如果前者大于后者(默认升序),则交换它们

2.当遍历到数组最后一对相邻元素后,数组中最后一个元素会是最大的数。此时一趟冒泡排序完成

3.重新遍历数组比较相邻元素(最后一个元素除外,因为已经是最大的了)。一趟结束后,数组中第二大的元素将位于倒数第二个位置

4.重复进行上述步骤(已经就位的元素除外),直到所有元素均不需要再交换

动图表示:

接下来,我们尝试实现冒泡排序:

void BubbleSort(int* arr, int n)
{//外层循环控制排序的趟数for (int i = 0; i < n - 1; i++)//每一趟排序使一个元素就位,n个元素的数组需要n-1趟排序(最后一趟会使前两个元素就位){//内层循环控制需要比较的相邻元素for (int j = 0; j < n - 1 - i; j++)//每一趟结束后,需要比较的元素便减少一个,所以要减去i{if (arr[j] > arr[j + 1])//若前者大于后者,说明顺序错误,需要交换{Swap(&arr[j], &arr[j + 1]);//交换两元素}}}
}

不难发现,如果没有到达排序的趟数,但是数组已经有序,程序就会继续进行不必要的元素比较,运行效率降低。基于这一点,我们可以做出如下改进:

void BubbleSort(int* arr, int n)
{//外层循环控制排序的趟数for (int i = 0; i < n - 1; i++)//每一趟排序使一个元素就位,n个元素的数组需要n-1趟排序(最后一趟会使前两个元素就位){int flag = 1;//假设数组已经有序//内层循环控制需要比较的相邻元素for (int j = 0; j < n - 1 - i; j++)//每一趟结束后,需要比较的元素便减少一个,所以要减去i{if (arr[j] > arr[j + 1])//若前者大于后者,说明顺序错误,需要交换{Swap(&arr[j], &arr[j + 1]);//交换两元素flag = 0;//发生了交换,说明数组并非有序}}if (flag == 1)//如果数组已经有序,则不需要排序,直接退出循环{break;}}
}

接下来,我们对测试数组进行排序:

int main()
{int arr[] = { 5,9,4,0,2,7,8,0,0,1,4,4,1,3,5,6,3,2,9,7 };int sz = sizeof(arr) / sizeof(arr[0]);BubbleSort(arr, sz);//冒泡排序for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}

运行结果:

可以看到,排序成功了。

冒泡排序的特性总结

空间复杂度:O(1)

时间复杂度:O(N^2)

稳定性(相同值的元素排序前后的相对次序是否保持不变):稳定

无论数组是否有序,都有大量元素被重复地进行比较和交换,运行效率不高

二、选择排序

        选择排序是一种比较接近人类思想的排序算法,它的核心思想非常简单:从数组中寻找最小(最大)元素,将其放于数组第一个位置(最后一个位置)处,然后再从剩余的元素中寻找最小(最大)值,放到数组第二个位置(倒数第二个位置)处......

        具体步骤如下:

1.首先遍历数组,寻找到数组中最小的元素,将其与数组的首元素进行交换

2.开始遍历数组的剩下部分,寻找最小值,与该部分的首元素进行交换。

3.重复第二步,直到“该部分”只有一个元素为止,说明数组已经排序好

动图表示:

接下来,我们尝试代码实现选择排序:

void SelectSort(int* arr, int n)
{//外层循环控制遍历次数以及遍历的起始位置for (int i = 0; i < n; i++){int mini = i;//假设最小值位于遍历部分的首元素处//内层循环,寻找最小值for (int j = mini + 1; j < n; j++){if (arr[j] < arr[mini])//将更小的元素标记为mini{mini = j;}}Swap(&arr[i], &arr[mini]);//将最小值与遍历部分的首元素交换}
}

运行测试:

选择排序特性总结

空间复杂度:O(1)

时间复杂度:O(N^2)

稳定性:不稳定

使用寻找最值进行交换的方式,虽然在效率上相比冒泡排序有所提升,但是依然不是很实用。

三、插入排序

        插入排序是一种适用于对少量数据进行排序的算法,它的效率要略高于冒泡排序和选择排序。就像玩扑克牌一样,它的核心思想是:将一个个数据不断插入到已经有序的数组的合适位置,从而使整个数组有序。

        具体步骤如下:

1.将数组首元素视为有序。

2.将有序数组的下一个元素取出,然后开始对该元素之前的部分进行遍历,找到合适位置并插入

3.将这部分有序数组看成整体,重复第二步,直到将最后一个元素调整完成

动图表示:

代码实现如下:

void InsertSort(int* arr, int n)
{//外层循环确定遍历次数for (int i = 0; i < n - 1; i++){int end = i;//记录有序数组的末尾位置int tmp = arr[end + 1];//保存需要插入的元素tmp//内层循环控制对有序数组的遍历while (end >= 0)//当end小于0时,一次遍历结束{if (arr[end] > tmp){arr[end + 1] = arr[end];//该值后移一位,留空缺end--;//继续向前走}else//当tmp值比有序数组中某元素小或相等时,说明tmp应该插入在该元素之后,终止遍历{break;}}arr[end + 1] = tmp;//插入到该元素之后的位置}
}

代入测试:

插入排序性能总结

空间复杂度:O(1)

时间复杂度:O(N^2)

稳定性:稳定

虽然在时间复杂度上没有什么变化,但是相比冒泡排序和选择排序,性能有所提升。由于它是按照顺序进行扫描并且插入的,所以相同值元素的相对位置不会改变,是一种稳定的排序算法,是排序少量数据的首选。不过面对的数据量庞大时,依然不够看。

四、希尔排序

        希尔排序又称为缩小增量排序,是一种效率很高的排序算法,算是插入排序的升级版。它的核心思想也比较简单:首先对数组进行预排序,使其接近有序,然后进行插入排序

        这里介绍一下排序的思路:

        首先选定一个增量gap(这里我们使gap=数组元素n/3),将数组中相差gap个位置的元素视为一个子序列,然后分别对这些子序列进行插入排序。当所有子序列排序结束之后,使gap的值逐渐减小(再除以3),然后再将相差gap个位置的元素构成的子序列进行插入排序......直到gap为1时,构成的序列就是数组本身,对其进行一次插入排序即可。

        当然,这里的增量gap并不是只有n/3这一种取法。他还有很多种取法,例如n/2、n/4等。但要注意:变化的gap值应尽量满足没有除1以外的公因子,并且最后一次gap的值是1

接下来,我们尝试实现希尔排序:

void ShellSort(int* arr, int n)
{int gap = n;//定义增量gap//循环控制预排序的次数,以及进行最后一次插入排序while (gap > 1){gap = gap / 3 + 1;//首先调整gap值//进行插入排序//注意细节处理,是对相隔gap的元素进行排序for (int i = 0; i < n - gap; i++)//一次循环结束后,i++就走向了下一个子序列的起始位置{int end = i;//记录有序部分的末尾位置int tmp = arr[end + gap];//end+gap位置的元素视为子序列中的下一个元素while (end >= 0){if (arr[end] > tmp){arr[end + gap] = arr[end];end -= gap;//一次向前走gap个位置}else{break;}}arr[end + gap] = tmp;}}
}

从代码角度看,它相比插入排序又多了一层循环,效率怎么会高于插入排序呢?实际上,有了预排序之后,最后一次插入排序所节省的时间要远远大于预排序消耗的时间

代入数据测试:

希尔排序性能总结

空间复杂度:O(1)

时间复杂度:O(N^1.3)(计算过程相对复杂)

稳定性:不稳定

相比冒泡排序,选择排序和插入排序,希尔排序的时间复杂度较低,运行效率更高,更适用于大规模数据的排序。但是它的性能不稳定,易受数据和增量选择的影响。

程序全部代码

程序全部代码如下:

#include <stdio.h>
#include <stdlib.h>//交换函数
void Swap(int* x, int* y)
{int tmp = *x;*x = *y;*y = tmp;
}//冒泡排序
void BubbleSort(int* arr, int n)
{//外层循环控制排序的趟数for (int i = 0; i < n - 1; i++)//每一趟排序使一个元素就位,n个元素的数组需要n-1趟排序(最后一趟会使前两个元素就位){int flag = 1;//假设数组已经有序//内层循环控制需要比较的相邻元素for (int j = 0; j < n - 1 - i; j++)//每一趟结束后,需要比较的元素便减少一个,所以要减去i{if (arr[j] > arr[j + 1])//若前者大于后者,说明顺序错误,需要交换{Swap(&arr[j], &arr[j + 1]);//交换两元素flag = 0;//发生了交换,说明数组并非有序}}if (flag == 1)//如果数组已经有序,则不需要排序,直接退出循环{break;}}
}//选择排序
void SelectSort(int* arr, int n)
{//外层循环控制遍历次数以及遍历的起始位置for (int i = 0; i < n; i++){int mini = i;//假设最小值位于遍历部分的首元素处//内层循环,寻找最小值for (int j = mini + 1; j < n; j++){if (arr[j] < arr[mini])//将更小的元素标记为mini{mini = j;}}Swap(&arr[i], &arr[mini]);//将最小值与遍历部分的首元素交换}
}//插入排序
void InsertSort(int* arr, int n)
{//外层循环确定遍历次数for (int i = 0; i < n - 1; i++){int end = i;//记录有序数组的末尾位置int tmp = arr[end + 1];//保存需要插入的元素tmp//内层循环控制对有序数组的遍历while (end >= 0)//当end小于0时,一次遍历结束{if (arr[end] > tmp){arr[end + 1] = arr[end];//该值后移一位,留空缺end--;//继续向前走}else//当tmp值比有序数组中某元素小或相等时,说明tmp应该插入在该元素之后,终止遍历{break;}}arr[end + 1] = tmp;//插入到该元素之后的位置}
}//希尔排序
void ShellSort(int* arr, int n)
{int gap = n;//定义增量gap//循环控制预排序的次数,以及进行最后一次插入排序while (gap > 1){gap = gap / 3 + 1;//首先调整gap值//进行插入排序//注意细节处理,是对相隔gap的元素进行排序for (int i = 0; i < n - gap; i++)//一次循环结束后,i++就走向了下一个子序列的起始位置{int end = i;//记录有序部分的末尾位置int tmp = arr[end + gap];//end+gap位置的元素视为子序列中的下一个元素while (end >= 0){if (arr[end] > tmp){arr[end + gap] = arr[end];end -= gap;//一次向前走gap个位置}else{break;}}arr[end + gap] = tmp;}}
}int main()
{int arr[] = { 5,9,4,0,2,7,8,0,0,1,4,4,1,3,5,6,3,2,9,7 };int sz = sizeof(arr) / sizeof(arr[0]);//BubbleSort(arr, sz);//SelectSort(arr, sz);//InsertSort(arr, sz);ShellSort(arr, sz);for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}

总结

        今天,我们学习了八大排序的其中四种:冒泡排序,选择排序,插入排序和希尔排序。在理解这些排序思想和实现它们的过程当中,我们感受到了算法之美,也加强了分析问题、解决问题的能力。之后博主会和大家分享剩下的几种排序算法。如果你觉得博主讲的还不错,就请留下一个小小的赞在走哦,感谢大家的支持❤❤❤

相关文章:

【排序算法】八大排序(上)(c语言实现)(附源码)

&#x1f31f;&#x1f31f;作者主页&#xff1a;ephemerals__ &#x1f31f;&#x1f31f;所属专栏&#xff1a;算法 目录 前言 写一串测试数据 交换两元素的函数 一、冒泡排序 二、选择排序 三、插入排序 四、希尔排序 程序全部代码 总结 前言 排序算法是计算机科…...

Python版《超级玛丽+源码》-Python制作超级玛丽游戏

小时候最喜欢玩的小游戏就是超级玛丽了&#xff0c;有刺激有又技巧&#xff0c;通关真的很难&#xff0c;救下小公主还被抓走了&#xff0c;唉&#xff0c;心累&#xff0c;最后还是硬着头皮继续闯&#xff0c;终于要通关了&#xff0c;之后再玩还是没有那么容易&#xff0c;哈…...

互联网私有IP地址列表

最近因为业务需要&#xff0c;要判断用户的IP是否私有IP&#xff0c; 以前知道的私有IP&#xff0c;基本上只有如下几个(注意&#xff1a;这不是正确答案)&#xff1a; 10.0.0.0/8&#xff08;10.0.0.0-10.255.255.255&#xff09;172.16.0.0/12&#xff08;172.16.0.0-172.31…...

光伏项目管理软件为什么那么多光伏人在用?

在光伏行业迅速发展的今天&#xff0c;光伏项目管理软件已成为众多光伏从业者不可或缺的得力助手。那么&#xff0c;为何这款软件能够受到如此广泛的青睐和应用呢&#xff1f; 一、提高项目管理效率 光伏项目管理软件通过数字化、智能化的手段&#xff0c;对光伏项目的各个环节…...

《AOP实战》— 自定义注解

承接上文&#xff08;传送门 —>《面试必考》 — AOP-CSDN博客&#xff09;&#xff0c;在被面试官拷打的时候&#xff0c;会被问到一个致命问题&#xff1a;“你了解aop吗&#xff1f;有具体的使用经验吗&#xff1f;” 你&#xff1a;......... 言尽于此&#xff0c;此篇…...

微前端架构下的单页应用实现策略

随着Web应用的复杂性日益增加&#xff0c;传统的多页应用&#xff08;MPA&#xff09;模式已经难以满足现代Web开发的需求。单页应用&#xff08;SPA&#xff09;以其流畅的用户体验和高效的页面加载速度&#xff0c;逐渐成为Web开发的主流模式。然而&#xff0c;在微前端架构下…...

JWT(JSON Web Token)工作原理及特点

JWT定义 概念&#xff1a;JWT是一种开放标准&#xff08;RFC 7519&#xff09;&#xff0c;用于在网络上安全传输信息&#xff0c;常用于身份验证。比喻&#xff1a;类似于电子通行证&#xff0c;包含用户身份信息&#xff0c;用于身份验证和享受服务。 JWT组成部分 头部&am…...

【体检】程序人生之健康检查,全身体检与预防疫苗,五大传染病普筛,基因检测等

程序员养生指南之 【体检】程序人生之健康检查&#xff0c;全身体检项目分类&#xff0c;五大传染病普筛&#xff0c;基因检测等 文章目录 一、全身体检与预防疫苗&#xff08;年检&#xff09;1、实验室检测&#xff1a;生化全套检查2、医技检查&#xff1a;辅助诊疗科室3、科…...

汇编语言中的指令锁定:解锁高效并发编程

标题&#xff1a;汇编语言中的指令锁定&#xff1a;解锁高效并发编程 在汇编语言的微观世界中&#xff0c;指令锁定&#xff08;Instruction Locking&#xff09;是一种确保数据一致性和操作原子性的关键机制。通过使用特定的lock前缀&#xff0c;开发者可以告诉CPU在执行多处…...

《人工智能时代:金融投资决策的潜在系统性风险及防范策略》

在当今数字化飞速发展的时代&#xff0c;人工智能&#xff08;AI&#xff09;在金融领域的应用日益广泛&#xff0c;特别是在投资决策方面展现出了巨大的潜力。然而&#xff0c;随着其影响力的不断扩大&#xff0c;我们也必须警惕潜在的系统性风险。 人工智能在金融投资决策中…...

MT7621+MT7915(MT7905)+MT7975 (W7621A6G-SDK)编译固件与升级固件方法

一、搭建开发环境&#xff0c;编译固件。 1、安装在Ubuntu 14.04.5 x86_64系统后&#xff0c;然后安装下面命令行。 $ sudo apt-get install git g make libncurses5-dev subversion libssl-dev gawk libxml-parser-perl unzip wget python xz-utils vim zlibc zlib1g zlib1g…...

[php:\\filter]

写入 #题目 <?php $filename$_GET[filename]; $content$_POST[content]; file_put_contents($filename,<?php exit();.$content); highlight_file(__FILE__); ?> 源码如上&#xff0c;需要再服务器上写入一句话木马 payload如下&#xff1a; #<?php phpinf…...

Linux-环境变量

文章目录 第6章 Linux 环境变量6.1 环境变量简介&#xff1f;6.2 全局变量6.3 局部环境变量6.4 设置用户自定义变量6.4.1 设置局部用户自定义变量6.4.2 设置全局环境变量6.4.3 删除环境变量 6.5 默认的shell环境变量6.6 设置PATH环境变量6.7 定位系统环境变量6.7.1 登录shell6.…...

DISCUZ论坛中 “阅读权限10“这几个字的修改教程以及后台目录路径修改后的管理路径

第一篇&#xff1a;修改“阅读权限10”这几个字 首先找到目录&#xff1a; source\language\lang_message.php 找到这个文件 查找&#xff1a; thread_nopermission 首发地址&#xff1a;玖毅论坛 第二篇&#xff1a;后台管理路径 看到好多人在网上问discuz管理路径怎么…...

springboot 整合spring-boot-starter-data-elasticsearch

依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-elasticsearch</artifactId></dependency> 配置 spring:elasticsearch:rest:uris: "http://localhost:9200" # Elastics…...

Element UI中el-dialog作为子组件如何由父组件控制显示/隐藏~

1、这里介绍的是将el-dialog作为组件封装便于复用&#xff0c;如何通过父组件控制子组件dialog的显示与隐藏。 2、思路:首先el-dialog是通过dialogVisible的值是否为true或false来控制显示与隐藏的。那么我们可以通过父传子props来将true&#xff08;即showFlag的值&#xff0…...

【vue讲解:es6导入导出语法、 vue-router简单使用、登录跳转案例、scoped的使用、elementui使用】

1 es6导入导出语法 # 做项目&#xff1a;肯定要写模块--》导入使用# 默认导出和导入 在某个js中 # 命名导出和导入1.1 默认导出和导入 // #########导出语法########### // export default name // 只导出变量 // export default add // 只导出函数// export default {nam…...

#beego的orm一直引入失败#

在导入beego的orm的时候&#xff0c;一直导入失败&#xff0c;orm显示红色,表示导入失败 解决办法: 1:升级go,由1.7升级到1.8 2&#xff1a;执行以下命令 go clean go get github.com/astaxie/beego/orm go mod tidy go mod vendor 3:测试在vendor中可以看到…...

Vue插值:双大括号标签、v-text、v-html、v-bind 指令

创建应用程序实例后&#xff0c;需要通过插值进行数据绑定。数据绑定是 Vue.js 最核心的一个特性。建立数据绑定后&#xff0c;数据和视图会相互关联&#xff0c;当数据发生变化时&#xff0c;视图会自动进行更新。这样就无须手动获取 DOM 的值&#xff0c;使代码更加简洁&…...

实验五之用Processing绘画

1.案例代码如下&#xff1a; import generativedesign.*; import processing.pdf.*; import java.util.Calendar; Tablet tablet; boolean recordPDF false; float x 0, y 0; float stepSize 5.0; PFont font; String letters "Sie hren nicht die folgenden Gesnge…...

Apache CloudStack Official Document 翻译节选(七)

关于 Apache CloudStack 的 最佳实践 &#xff08;一&#xff09; Best Practices 部署Apache CloudStack是极具挑战性的&#xff0c;在整个部署过程中需要你做出形形色色的技术性选择。Apache CloudStack的配置条目是相当灵活的&#xff0c;这是因为在组合和配置具体条目时有…...

动态创建 Delphi 按钮的完整指南:基于配置文件的 `TGridPanel` 实现

在 Delphi 开发中&#xff0c;我们经常需要根据不同的配置动态生成 UI 元素。本文将带你通过一个完整的示例&#xff0c;演示如何根据配置文件动态创建按钮&#xff0c;并将它们排列在一个 TGridPanel 中。每个按钮的标题、链接、颜色和大小都将从配置文件中读取。 “C:\myApp\…...

【设计模式】工厂模式和抽象工厂模式

工厂模式 function User(role, pages) {this.role role;this.pages pages; }// new User(admin, [home, user, setting]); // new User(user, [home, user]); // new User(guest, [home]);function UserFactory(role) {switch (role) {case admin:return new User(role, [ho…...

【xilinx】Versal Adaptive SoC DDRMC - NoC QoS 选项卡未出现

在 2024.1 之前的 Vivado 版本中&#xff0c;用户在使用 NoC 验证块设计时可以访问 NoC 对象窗口和 QoS 选项卡。 Vivado 2024.1 中存在一个已知问题&#xff0c;即 NoC 对象窗口和 QoS 选项卡不出现。 要显示 NoC 对象窗口和 QoS 选项卡&#xff0c;请保存块设计&#xff0c;…...

融合创新:EasyCVR视频汇聚平台云计算技术与AI技术共筑雪亮工程智能防线

随着信息技术的飞速发展&#xff0c;视频云计算技术作为云计算领域的一个重要分支&#xff0c;正逐步在公共安全、社会治理等领域展现出其独特的优势。特别是在雪亮工程这一群众性治安防控工程中&#xff0c;视频云计算技术更是发挥了不可替代的作用。本文将从视频云计算技术的…...

keepalived的技术原理及其在负载均衡场景中的应用

keepalived的技术原理及其在负载均衡场景中的应用 深入探讨Keepalived及其在负载均衡场景中的应用1. **Keepalived概述**2. **Keepalived的技术原理**2.1 **VRRP协议**2.2 **健康检查机制**2.3 **脚本管理** 3. **Keepalived与LVS的结合应用**3.1 **LVS优缺点** 4. **Nginx与HA…...

树的重心 by江河湖海

引入 重心是什么? 想象你有一个由线悬挂的秋千,秋千的两端坐着两个人,如果这两个人坐在秋千的重心上,秋千就会保持平衡。在树的结构中,重心就是那个让所有节点到它那里的“距离”(可以理解为线的长度)总和最小的点。 重心为什么最多只有两个? 假设树的重心有两个,…...

MySQL存储过程深入指南

MySQL存储过程深入指南 存储过程是MySQL中一个强大的功能,能够显著提升数据库操作的效率和灵活性。本文将全面介绍存储过程的概念、语法、使用方法及最佳实践,帮助读者熟练掌握存储过程的使用。 1. 什么是存储过程? 存储过程(Stored Procedure)是预先编译并存储在数据库…...

牛客算法小题

目录 牛客.求和​编辑 牛客.abb 牛客.合并k个有序链表 牛客.滑雪&#xff08;暴力->递归->记忆化搜索&#xff09; 牛客.旋转字符串 牛客.求和 我没想到是dfs&#xff0c;另外我的dfs能力确实也不强&#xff0c;另外难度大的是他的那个输出 import java.util.Scanne…...

小米SU7销量超特斯拉,新车明年上半年发布

小米 SU7&#xff0c;一款国内新能源车品牌纯血新势力旗下首款轿车&#xff0c;上市短短 4 个月卖出超 4 万台&#xff0c;月均销量过万。 该说不说&#xff0c;这放在整个新能源汽车工业史上也足以称得上是一件小刀喇拍屁股&#xff0c;让人开了眼的事儿。 就在本月初&#x…...