八大排序超详解(动图+源码)

news2025/1/16 5:13:56

💓博主个人主页:不是笨小孩👀
⏩专栏分类:数据结构与算法👀 刷题专栏👀 C语言👀
🚚代码仓库:笨小孩的代码库👀
⏩社区:不是笨小孩👀
🌹欢迎大家三连关注,一起学习,一起进步!!💓

在这里插入图片描述

排序算法

  • 排序的概念
  • 插入排序
  • 希尔排序
  • 选择排序
  • 冒泡排序
  • 堆排序
  • 快速排序
  • 归并排序
  • 计数排序

排序的概念

排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。
稳定性:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次 序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排 序算法是稳定的;否则称为不稳定的。
内部排序:数据元素全部放在内存中的排序。
外部排序:数据元素太多不能同时放在内存中,根据排序过程的要求不能在内外存之间移动数据的排序。

常见的排序算法:

在这里插入图片描述
除了这些排序以外,该有一个很奇怪的排序,计数排序,我们待会将,我们接下来,就从第一个排序开始:

插入排序

插入排序的思想很简单就是把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列 。插入排序可以理解为就是我们打扑克牌摸排的过程,摸一张排,依次比较然后将它插入的合适的位置。

我们看图:
在这里插入图片描述
这个排序很简单,根据图我们就可以把第一个数据当成有序的数据,然后后面的数据依次插入,直到将数据插入完,这样就有序了。

代码如下:

void InsertSort(int* arr, int n)
{
	for (int i = 0; i < n-1; i++)
	{
		//end表示有序数据的最后一数的下标
		int end = i;
		//tmp保存需要插入的值
		int tmp = arr[end+1]; 
		while (end >= 0)
		{
		//依次比较如果比需要插入的数大,就往后移,否则就跳出循环
			if (arr[end] > tmp)
			{
				arr[end + 1] = arr[end];
				end--;
			}
			else
			{
				break;
			}
		}
		//跳出循环后将需要插入的数据放到end后面的位置
		arr[end + 1] = tmp;
	}
}

总结:

  1. 元素集合越接近有序,直接插入排序算法的时间效率越高
  2. 时间复杂度:O(N^2)
  3. 空间复杂度:O(1),它是一种稳定的排序算法
  4. 稳定性:稳定

希尔排序

希尔排序法又称缩小增量法。希尔排序法的基本思想是:先选定一个整数gap,把待排序文件中所有记录分成gap个组,所有距离为gap的记录分在同一组内,并对每一组内的记录进行排序。然后,重复上述分组和排序的工作。当到达=1时,所有记录在统一组内排好序。

插入排序第一步我们需要预排序
预排序后插入排序就很快了,直接使用插入排序就可以了。但是当我们的gap=1是,希尔排序就相当于插入排序了。这里gap可以取很多值,但是要保证最后一次gap=1.

在这里插入图片描述

代码如下:

void ShellSort(int* arr, int n)
{
	int gap = n;
	//要进行多趟排序
	while (gap > 1)
	{
	//+1是为了保证gap最后一次等于1
		gap = gap / 3 + 1;
		for (int i = 0; i < n - gap; i++)
		{
		//每次分别排gap组数据,每组间隔gap个数据,一共gap组
			int end = i;
			int tmp = arr[i + gap];
			while (end >= 0)
			{
				if (arr[end] > tmp)
				{
					arr[end + gap] = arr[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			arr[end + gap] = tmp;
		}
	}
}

总结:

  1. 希尔排序是对直接插入排序的优化。
  2. 当gap > 1时都是预排序,目的是让数组更接近于有序。当gap == 1时,数组已经接近有序的了,这样就 会很快。这样整体而言,可以达到优化的效果。我们实现后可以进行性能测试的对比。
  3. 希尔排序的时间复杂度不好计算,因为gap的取值方法很多,导致很难去计算,因此在好些树中给出的 希尔排序的时间复杂度都不固定:我们记住大约就等于O(N^1.3)
  4. 稳定性:不稳定

选择排序

选择排序是每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完 。

在这里插入图片描述

我们这里实现的是依次找大的,然后放到最后面,和图不太一样,但是思想都一样。

代码如下:

void SelectSort(int* arr, int n)
{

	int end = n - 1;
	while (end>0)
	{
		//每次初始化最大在0处,防止maxi到已经在排好序的位置
		int maxi = 0;
		for (int i = 0; i <= end; i++)
		{
			if (arr[i] > arr[maxi])
			{
				maxi = i;
			}
		}
		//找到后和最后一个数据交换
		Swap(&arr[maxi], &arr[end]);
		end--;
	}
}

选择排序我们这里可以优化一下,就是每次选出最小的和最大的,然后最小的放到左边,最大的放到右边,然后接着找剩余数据的最大最小,直到结束。

代码如下:

void SelectSort(int* arr, int n)
{
	int begin = 0;
	int end = n - 1;

	while (begin < end)
	{
		int maxi = begin;
		int mini = begin;
		//依次找大和找小
		for (int i = begin; i <= end; i++)
		{
			if (arr[mini] > arr[i])
			{
				mini = i;
			}
			if (arr[maxi] < arr[i])
			{
				maxi = i;
			}
		}
		//找到后将大的数据放到后面
		Swap(&arr[maxi], &arr[end]);
		//防止最小的数据在最后面被换走了,及时修正
		if (mini == end)
		{
			mini = maxi;
		}

		Swap(&arr[mini], &arr[begin]);

		begin++;
		end--;
	}
}

总结:

  1. 直接选择排序思考非常好理解,但是效率不是很好。实际中很少使用
  2. 时间复杂度:O(N^2)
  3. 空间复杂度:O(1)
  4. 稳定性:不稳定

冒泡排序

冒泡排序大多数人应该都知道,它的基本思想就是依次比较,将大的数据冒到最后然后重复前面的过程,就可以完成排序。

在这里插入图片描述
代码如下:

void BubbleSort(int* arr, int n)
{
	for (int i = 0; i < n - 1; i++)
	{
		int flag = 1;
		for (int j = 1; j < n - i; j++)
		{
			if (arr[j] < arr[j - 1])
			{
				Swap(arr + j, arr + j - 1);
				flag = 0;
			}
		}
		if (flag == 1)
		{
			break;
		}
	}
}

总结:

  1. 冒泡排序是一种非常容易理解的排序
  2. 时间复杂度:O(N^2)
  3. 空间复杂度:O(1)
  4. 稳定性:稳定

堆排序

堆排序前面已经讲过一次了,这里就不做过多的解释了,想要详细了解请戳。
这里是引用

总结:

  1. 堆排序使用堆来选数,效率就高了很多。
  2. 时间复杂度:O(N*logN)
  3. 空间复杂度:O(1)
  4. 稳定性:不稳定

快速排序

快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素序列中
的某元素作为基准值,按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右
子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止。

我们每次可以将数组划分为两部分,keyi是那个选出来的数的最终的下标,然后第一次排序后就是[left,keyi-1],keyi,[keyi+1,right],我们每一趟要保证的是keyi左边的数据逗比key小,右边的都比它大,然后左区间重复这个操作,右区间也重复这个操作,这就有点像二叉树的前序遍历,直到每个区间只剩下一个值,或者区间不存在时,我们结束递归。

快排的整体框架:

void QuickSort(int* arr, int left, int right)
{
	if (left >= right)
	{
		return;
	}
	int keyi = partSort(arr,left,right);
	
	QuickSort(arr,left, keyi - 1);
	QuickSort(arr,keyi + 1, right);

}

这里的partSort就是我们的单趟排序,我们讲三种方法:

  1. hoare版本
    在这里插入图片描述

我们需要两个指针,一个从左边开始走,一个从右边开始走,再定义一个key,和keyi,keyi保存key的小标,如果左边左key就右边先走,右边左key就左边先走,,然后左边找比key大的数,右边找比key小的数,找到后交换,然后接着走,直到相遇,然后把相遇的位置和key交换一下。

为什么左边做key右边先走呢?

因为这样可以保证相遇的位置一定是比key小等于的数,相遇无非就是两种情况,L遇到R,R遇到L,如果是L遇到R,我们让右边先走,R停下的位置一定是比key小的数,如果是R遇L,假设数组中的数都比key大,所以key遇到L是就是等于key,所以我们左边做key让右边先走,是可以保证相遇位置一定比key小的。

代码如下:

int partSort1(int* arr, int left, int right)
{
	int keyi = left;
	while (left < right)
	{
		//右边找小
		while (left < right && arr[right] >= arr[keyi])
		{
			right--;
		}

		//左边找大
		while (left < right && arr[left] <= arr[keyi])
		{
			left++;
		}

		Swap(&arr[left], &arr[right]);
	}

	Swap(&arr[left], &arr[keyi]);

	return left;
}

  1. 挖坑法
    在这里插入图片描述

我们还是将左边做key,然后保存它的值,然后它就是一个坑,还是两个指针,由于左边有一个坑,所以右边就要找小的数来填这个坑,然后将右边的那个位置变成新的坑,然后左边找大,找到后接着填坑,更新坑的位置,L和R一定有一个是坑,所以,当他们相遇时,那个位置一定是坑,然后将key放进去即可。

代码如下:

int partSort2(int* arr, int left, int right)
{
	int hole = arr[left];
	int keyi = left;
	while (left < right)
	{
		while (left < right && arr[right] >= hole)
		{
			right--;
		}

		arr[keyi] = arr[right];
		keyi = right;
		while (left < right && arr[left] <= hole)
		{
			left++;
		}
		arr[keyi] = arr[left];
		keyi = left;
	}

	arr[keyi] = hole;
	return keyi;
}

  1. 前后指针法
    在这里插入图片描述

定义两个指针一个prev一个cur,cur用来遍历数组,还是用左边的值来做key,然后将cur找到比key小的值就和++prev位置的数交换直到遍历结束,然后再把prev位置的值可key交换即可。

代码如下:

int partSort3(int* arr, int left, int right)
{
	int keyi = left;
	int cur = left + 1;
	int prev = left;
	while (cur <= right)
	{
		if (arr[cur] < arr[keyi]&&++prev!=cur)
		{
			Swap(&arr[prev], &arr[cur]);
		}

		cur++;
	}
	Swap(&arr[prev], &arr[keyi]);

	return prev;
}

快速排序的非递归版本

非递归的话,我们用队列和栈都是可以的,但是想要模仿递归的路径的话我们就要使用栈,我们先把数组的整个区间放到栈里面,然后在进行一趟排序后,我们把排出来的左区间和右区间入栈,由于先走左边,所以就要先把右边的区间压栈,然后依次进行,只要区间存在,我们就压,只要栈不为空,就代表一直有区间未处理。所以我们就一直重复操作,当然单趟排序的话,用上面的那种方法都可以。

代码如下:

void QuickSortNonR(int* arr, int left, int right)
{
	Stack st;
	StackInit(&st);
	StackPush(&st,right);
	StackPush(&st, left);
	while (!StackEmpty(&st))
	{
		int begin = StackTop(&st);
		StackPop(&st);
		int end = StackTop(&st);
		StackPop(&st);

		int keyi = partSort1(arr, begin, end);

		if (keyi + 1 < end)
		{
			StackPush(&st, end);
			StackPush(&st, keyi + 1);

		}

		if (keyi - 1 > begin)
		{
			StackPush(&st, keyi - 1);
			StackPush(&st, begin);
		}
	}
}

快速排序,还可以优化,他的效率和选的key的关系很大,所以我们有种方法叫做三数取中,左边的值、右边的值、中间的值,然都找到这三个数中间的数,把他换到左边,就可以了。

总结:

  1. 快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫快速排序
  2. 时间复杂度:O(N*logN)
  3. 空间复杂度:O(logN)
  4. 稳定性:不稳定

归并排序

归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

归并排序需要将区间分为两部分,然后两部分都要有序才能归并,那左右区间又可以分割,以此类推,当区间只有一个数的时候就可以认为有序,这时我们可以走一个类似与二叉树后序遍历的思路,我们想归并左右区间,但是左右区间都无序,我们就递归左边让左边有序,在递归右边让右边有序,最后再左右归并,就可以排好了。

在这里插入图片描述

我们这里就需要一个数组来保存我们归并的值,我们取两段区间的值依次比较,拿小的尾插到tmp数组中,等归并完再拷回原数组,即可。

代码如下:

void _MergeSort(int* arr, int left, int right,int* tmp)
{
	if (left >= right)
	{
		return;
	}
	//分割区间
	int mid = (left + right) / 2;
	
	_MergeSort(arr, left, mid, tmp);
	_MergeSort(arr, mid+1, right, tmp);

	int begin1 = left, end1 = mid;
	int begin2 = mid + 1, end2 = right;
	int k = left;
	while (begin1 <= end1 && begin2 <= end2)
	{
		//选小的来尾插
		if (arr[begin1] <= arr[begin2])
		{
			tmp[k++] = arr[begin1++];
		}
		else
		{
			tmp[k++] = arr[begin2++];
		}
	}
	//不管哪个没有拷贝完,因为区间是有序地,直接尾插就可以
	while (begin1 <= end1)
	{
		tmp[k++] = arr[begin1++];
	}
	while (begin2 <= end2)
	{
		tmp[k++] = arr[begin2++];
	}
	//拷贝回原数组
	memcpy(arr + left, tmp + left, (right - left + 1) * sizeof(int));

}
void MergeSort(int* arr, int left, int right)
{
	int* tmp = (int*)malloc(sizeof(int) * (right - left + 1));
	//不能在这个函数中递归,不然每次都要开辟数组
	_MergeSort(arr, left, right,  tmp);

	free(tmp);
}

归并排序的非递归版本

我们会发现归并排序用队列和栈都用不了,但是我们可以使用循环来解决它,首先我们需要一个gap来记录每组归并的数据有几个,然后控制区间,来进行归并。
但是在归并中,会存在很多的越界问题,比如end1越界了,或者begin1越界了,但是这两种情况我们都很好处理,等处理到这种错误时我们可以看成只剩下一组数据,就可以不用动,放在原数就好,等待下一轮归,直接break跳出就可以,还有一种情况是end2越界了,这时还有一部分数据需要归并,那我们就调整end2为n-1就可以了。

在这里插入图片描述

代码如下:

void MergeSortNonR(int* arr, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	int gap = 1;

	//gap表示每组数据的长度
	while (gap < n)
	{
		for (int i = 0; i < n; i += 2 * gap)
		{
			//控制区间
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + 2 * gap - 1;
			int k = i;
			//越界即使调整或退出
			if (end1 >= n || begin2 >= n)
			{
				break;
			}
			if (end2 >= n)
			{
				end2 = n - 1;
			}

			while (begin1 <= end1 && begin2 <= end2)
			{
				if (arr[begin1] <= arr[begin2])
				{
					tmp[k++] = arr[begin1++];
				}
				else
				{
					tmp[k++] = arr[begin2++];
				}
			}

			while (begin1 <= end1)
			{
				tmp[k++] = arr[begin1++];
			}
			while (begin2 <= end2)
			{
				tmp[k++] = arr[begin2++];
			}
			//每次归并完拷贝会原数组
			memcpy(arr+i, tmp+i, sizeof(int)*(end2-i+1));
		}
		gap *= 2;
	}
	free(tmp);
}

总结:

  1. 归并的缺点在于需要O(N)的空间复杂度,归并排序的思考更多的是解决在磁盘中的外排序问题。
  2. 时间复杂度:O(N*logN)
  3. 空间复杂度:O(N)
  4. 稳定性:稳定

计数排序

计数排序是非常奇怪的排序,它不需要比较任何数据,他开辟一个和最大值一样的数组,然后将该数组初始化为0,然后原遍历数组,将原数组的值对应到我们开辟数组的下标,出现一次我们就++该位置,然后统计每个位置出现的次数,然后在依次拷贝回原数组,就可以了。
但是如果数据很大很集中,我们就没必要开那么大,会很浪费,我们需要找到最大值和最小值,然后使用相对位置,就可以了,每个数对应到减去最小值的那个小下标,这样我们数组也不用开的很大。

代码如下:

void CountSort(int* arr, int n)
{
	int min = arr[0];
	int max = arr[0];
	//找最大值和最小值
	for (int i = 0; i < n; i++)
	{
		if (max < arr[i])
		{
			max = arr[i];
		}

		if (min > arr[i])
		{
			min = arr[i];
		}
	}
	//计算区间方便开数组
	int c =max-min+1;

	int* nums = (int*)malloc(sizeof(int) * c);
	memset(nums, 0, c * sizeof(int));
	//统计
	for (int i = 0; i < n; i++)
	{
		nums[arr[i]-min]++;
	}
	int k = 0;
	//拷贝回原数组
	for (int i = 0; i < c; i++)
	{
		while (nums[i]--)
		{
			arr[k++] = i+min;
		}
	}
	
	free(nums);
}

总结:

  1. 计数排序在数据范围集中时,效率很高,但是适用范围及场景有限。
  2. 时间复杂度:O(MAX(N,范围))
  3. 空间复杂度:O(范围)
  4. 稳定性:稳定

各大排序的比较:
在这里插入图片描述
今天的分享就到这里感谢大家的关注和支持!

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

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

相关文章

高效提升工作效率,亚马逊云科技热门课程带你入门生成式AI

当前人工智能仍处于飞速发展阶段&#xff0c;作为当下最先进的科学技术之一&#xff0c;相信大家对AIGC关注已久。今天&#xff0c;引用亚马逊云科技最新发布的七项生成式AI新功能来跟大家聊聊近期的热门生成式AI&#xff01; 有人说&#xff0c;生成式AI将带来充满创造性的新世…

AMEYA360:村田共模扼流线圈,针对车载应用的高频噪声

近年来&#xff0c;随着ADAS(高级驾驶辅助系统)精度的提高&#xff0c;汽车行业开始安装大量毫米波雷达、LiDAR等高速传感设备。如果噪声从外部进入这些设备&#xff0c;系统可能无法正常工作。相反&#xff0c;如果这些设备产生噪声&#xff0c;则可能会对其他设备产生不利影响…

mysql 02 数据库的约束

为防止错误的数据被插入到数据表&#xff0c;MySQL中定义了一些维护数据库完整性的规则&#xff1b;这些规则常称为表的约束。常见约束如下&#xff1a; 主键约束 主键约束即primary key用于唯一的标识表中的每一行。被标识为主键的数据在表中是唯一的且其值不能为空。这点类似…

ICT产教融合创新实训基地软件测试实训室建设方案

一 、系统概述 ICT产教融合创新&#xff0c;简单来说&#xff0c;就是信息与通信技术&#xff08;ICT&#xff09;与产业界、教育界的融合创新。这个概念强调了在现代社会中&#xff0c;信息技术与产业发展以及教育培训之间相互关联的重要性。 ICT产教融合创新的核心思想包括以…

【正点原子STM32连载】第十章 跑马灯实验 摘自【正点原子】APM32F407最小系统板使用指南

1&#xff09;实验平台&#xff1a;正点原子stm32f103战舰开发板V4 2&#xff09;平台购买地址&#xff1a;https://detail.tmall.com/item.htm?id609294757420 3&#xff09;全套实验源码手册视频下载地址&#xff1a; http://www.openedv.com/thread-340252-1-1.html# 第十…

数字孪生技术对环境保护有哪些作用?

数字孪生技术在环境保护中的作用不容忽视&#xff0c;为我们创造了全新的可能性和解决方案。在追求可持续发展的今天&#xff0c;数字孪生以其独特的能力&#xff0c;正逐渐改变着环境保护的方式。 数字孪生技术首先在环境监测方面发挥了巨大作用。通过传感器和数据采集设备&a…

JVM整体回忆笔记

模块三 内存的分配 指针碰撞 空闲列表 分配内存的时候出现并发问题&#xff0c;几个线程同时抢同一块内存区域 CAS方法解决 本地线程分配缓冲&#xff08;-XX&#xff1a;UseTLAB&#xff09;,jvm默认开启 对象的组成 对象头、实例数据、对齐填充&#xff08;保证对象8个字节…

Sonar:Class variable fields should not have public accessibility

公有类变量字段不遵守封闭原则&#xff0c;主要有三个问题&#xff1a; 无法添加验证等附加行为。内部表示暴露在外&#xff0c;事后无法更改。成员值可能会在代码的任何地方发生变化&#xff0c;并且可能不符合程序员的假设。 通过使用私有属性和访问方法&#xff08;set 和…

输出倒逼输入系列之 阻塞 or 等待?

操作系统线程状态 《趣谈Linux 操作系统》第12章节&#xff0c;讲到 进程的数据结构&#xff0c;其中涉及到任务状态&#xff0c;这里简单概述一下 首先在操作系统中&#xff0c;不管是进程还是线程&#xff0c;都统一是 任务的概念&#xff0c;它的数据结构 为 task_strcut。…

深度学习1:通过模型评价指标优化训练

P(Positive)表示预测为正样本&#xff0c;N(negative)表示预测为负样本&#xff0c;T(True)表示预测正确,F(False)表示预测错误。 TP&#xff1a;正样本预测正确的数量&#xff08;正确检测&#xff09; FP&#xff1a;负样本预测正确数量&#xff08;误检测&#xff09; TN…

Python入门--安装和环境配置

要开始使用Python&#xff0c;您需要先安装Python环境。下面是安装Python环境的简单步骤&#xff1a; 1.访问Python官网&#xff08;https://www.python.org/&#xff09;&#xff0c;点击“下载”按钮。 2.选择适合您系统的版本下载&#xff0c;可选择Windows、macOS、Linux…

uniapp中map使用点聚合渲染marker覆盖物

效果如图&#xff1a; 一、什么是点聚合 当地图上需要展示的标记点 marker 过多时&#xff0c;可能会导致界面上 marker 出现压盖&#xff0c;展示不全&#xff0c;并导致整体性能变差。针对此类问题&#xff0c;推出点聚合能力。 点聚合官网教程 二、基本用法 template…

如何让CSDN学习成就个人能力六边形全是100分:解析个人能力雷达图的窍门

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

【Linux C】在sprintf中打印双引号

0x00 前言 最后更新日期&#xff1a;2023.8.16 0x01 在sprintf中打印双引号 在字符串中有双引号”的地方前面加上一个反斜杠“\”即可&#xff0c;例如&#xff1a; char szProQuery[256] {0}; char name[256] "XiaoMing"; sprintf(szProQuery,"%s said :…

使用druid连接池对配置文件密码进行加密

想让别人看不到你的数据库密码该怎么办呢&#xff1f;&#xff1f;&#xff1f; 1.建立maven项目添加依赖。 <dependencies><!-- com.mysql/mysql-connector-j --><dependency><groupId>com.mysql</groupId><artifactId>mysql-connector-j…

MyBatis速成

一、MyBatis简介 1、基本概念 MyBatis是一个持久层框架&#xff0c;用于简化JDBC开发。 表现层&#xff1a;页面展示&#xff1b; 业务层&#xff08;service&#xff09;&#xff1a;逻辑处理&#xff1b; 持久层&#xff08;dao&#xff09;&#xff1a;将数据保存到…

Access开发ERP简单主框架搭建

哈喽,哈喽,大家好! 今天我们接着来讲ERP,我们已经把理论知识与表结构讲完了,接下去我们开始讲最关键的业务部分的开发,这里也是大家最期待的。这里我们将会把系统分成几个部分来讲解,包括整个系统基本模块、基础信息、销售管理、采购管理、生产管理、仓库管理。今天的话…

微信小游戏投放量增长五倍,下半年买量能怎么玩?

App中运行小游戏的技术价值和业务价值都是显著的&#xff1a;通过小程序容器技术&#xff0c;承载多样化的小游戏运行在自有App内&#xff0c;实现跨平台的游戏资源共享&#xff0c;降低买量成本&#xff0c;此为「降本」。进一步的&#xff0c;在App内快速引入多小游戏应用&am…

载入内存,让程序运行起来

如果你的电脑上安装了QQ&#xff0c;你希望和好友聊天&#xff0c;会双击QQ图标&#xff0c;打开QQ软件&#xff0c;输入账号和密码&#xff0c;然后登录就可以了。 那么&#xff0c;QQ是怎么运行起来的呢&#xff1f; 首先&#xff0c;有一点你要明确&#xff0c;你安装的QQ软…

智能数据建模软件DTEmpower 2023R2新版本功能介绍

DTEmpower是由天洑软件自主研发的一款通用的智能数据建模软件&#xff0c;致力于帮助工程师及工科专业学生&#xff0c;利用工业领域中的仿真、试验、测量等各类数据进行挖掘分析&#xff0c;建立高质量的数据模型&#xff0c;实现快速设计评估、实时仿真预测、系统参数预警、设…