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

十大排序算法(冒泡排序、插入排序、选择排序、希尔排序、堆排序、快排、归并排序、桶排序、计数排序、基数排序)

目录

一、冒泡排序:

二、插入排序:

三、选择排序:

四、希尔排序:

五、堆排序:

六、快速排序:

6.1挖坑法:

6.2左右指针法

6.3前后指针法:

七、归并排序:

八、桶排序:

九、计数排序:

9.1绝对映射:

9.2现对映射:

十、基数排序: 


一、冒泡排序:

1、思路:通过对待排序序列从前向后(从下标较小的元素开始),依次对相邻两个元素的值进行两两比较,若发现前一个数大于后一个数则交换,使值较大的元素逐渐从前移向后部,就如果水底下的气泡一样逐渐向上冒。

2、先以一个数组讲解一下,然后再写代码:
      待排序数组:3,9,-1,10,20
       第一轮排序:
        (1)3,9,-1,10,20      ----3跟9比较,不交换

        (2)3,-1,9,10,20      ----9比 -1大,所以9跟 -1交换

        (3)3,-1,9,10,20      ----9跟10比较,不交换

        (4)3,-1,9,10,20      ----10跟20比较,不交换

           第一轮过后,将20这个最大的元素固定到了最后的位置。

           在第二轮的时候20不参与冒泡。

       第二轮排序:
            因为20的位置已经固定,所以只对前4个进行排序即可:

        (1)-1,3,9,10,20      ----3比 -1大,进行交换

        (2)-1,3,9,10,20      ----3跟9比较,不交换

        (3)-1,3,9,10,20      ----9跟10比较,不交换

            第二轮过后,将第二大的元素固定到了倒数第二个位置

       第三轮排序:
            10和20的位置已经确定,只需对前三个进行排序

        (1)-1,3,9,10,20      ----3和-1比较,不交换

        (2)-1,3,9,10,20      ----3和9比较,不交换

            第三轮过后,将第三大的元素位置确定

       第四轮排序:
            只对前两个元素进行排序

        (1)-1,3,9,10,20      ----3和-1比较,不交换

       第四轮过后,将第四大的元素位置确定,此时总共5个元素,已经排序好4个,从而最后一个自然而然就是排好序的了

小结:
设总的元素个数为n,那么由上边的排序过程可以看出:

(1)总计需要进行(n-1)轮排序,也就是(n-1)次大循环

(2)每轮排序比较的次数逐轮减少

(3)如果发现在某趟排序中,没有发生一次交换, 可以提前结束冒泡排序

(4)时间复杂度是O(N^2)      在有序的时候,很快,因为有exchange变量优化了代码

         在乱序的时候很慢很慢。

#include<stdio.h>
void swap(int* a, int* b)
{int tmp = *a;*a = *b;*b = tmp;
}
//冒泡排序
void BubbleSort(int* a, int n)
{int end = n - 1;//不能是n,不然会越界while(end){int exchange = 0;//优化,比较之后没有交换,说明已经排好了,就break循环for (int i = 0; i < end; i++){if (a[i] < a[i + 1]){swap(&a[i], &a[i + 1]);exchange++;}}if (exchange == 0) break;end--;}
}


二、插入排序:

1、思路:

        在待排序的元素中,假设前n-1个元素已有序,现将第n个元素插入到前面已经排好的序列中,使得前n个元素有序。按照此法对所有元素进行插入,直到整个序列有序。
  但我们并不能确定待排元素中究竟哪一部分是有序的,所以我们一开始只能认为第一个元素是有序的,依次将其后面的元素插入到这个有序序列中来,直到整个序列有序为止。

2、举例:

        如下图的插入扑克牌,当摸到7的时候,会不自觉的与前面的数比较,如果比7大,把大的数向后挪动(swap),然后在第一个小于7的后面插入7

 

//插入排序
void InsertSort(int* a, int n)
{for (int i = 1; i < n; i++){if (a[i] < a[i - 1])//先判断,如果i下标的值大于前面的数,就不进入{int tmp = a[i];int j;for (j = i - 1; j >= 0 && a[j] >tmp; j--){a[j+1] = a[j];}a[j+1] = tmp;}}
}
//两次循环就可以实现
//内部循环完成一趟的插入
//外层循环完成插入排序

三、选择排序:

思路:

1.内层循环一趟找出最小值的下标,与第一个数交换。重复找小,交换的两个操作。
2.实际上,我们可以一趟选出两个值,一个最大值一个最小值,然后将其放在序列开头和末尾,这样可以使选择排序的效率快一倍。

但时间复杂度还是O(N^2),效率还是不高

//选择排序
void SelectSort(int* a, int n)
{for (int i = 0; i < n-1; i++)//i<n-1当它是最后一个数的时候不需要进行交换排序{int min = i;int j;for (j = i; j < n; j++){if (a[j] < a[min]){min=j;}}swap(&a[i], &a[min]);//交换函数,前面的代码中有出现,我就不重复写了}
}

四、希尔排序:

思路:

1.插入排序的优化版,有一个预排序的过程。让大的数快速的跳到后面,小的数快速的跳到前面。

2.使待排序列接近有序,然后再对该序列进行一次插入排序。

3.相当于把直接插入排序中的1换成gap而已。 

//希尔排序
/*步骤:
1.先选定一个小于N的整数gap作为第一增量,然后将所有距离为gap的元素分在同一组,并对每一组的元素进行直接插入排序。然后再gap--,重复上述操作。
2.当gap==1时就是直接插入排序,就相当于整个序列被分到一组,进行一次直接插入排序,排序完成。*/void ShellSort(int* a, int n)
{//这里相当于把插入排序的1换成gapint gap = n;while (gap>1){gap = gap / 3 + 1;for (int i = gap; i < n; i++){if (a[i] < a[i - gap]){int tmp = a[i];int j;for (j = i - gap; j >= 0 && a[j] > tmp; j-=gap)//这里是j-=gap{a[j + gap] = a[j];}a[j + gap] = tmp;}}}
}

五、堆排序:

先来认识堆:

   1.什么是堆?

          大堆:父亲大于儿子          小堆:父亲小于儿子(父亲,儿子是二叉树的概念)

   2.堆的物理结构和逻辑结构?

          物理结构:数组         逻辑结构:完全二叉树

堆排序包括建堆(向下调整+循环)  堆排序(交换+向下调整)

  1.建堆:

        要建大堆,堆顶的元素和最后一个数交换,然后把size--,就不会破坏堆的结构

  2.向下调整算法:

        比较两个孩子的大小,选出大的孩子,与父亲比较,如果孩子大于父亲,交换。然后把parent=child,child=parent*2+1;向下调整算法一共会调整h-1次

//向下调整算法(要满足它下面的都满足堆,才能用)
void AdjustDown(int* a, int n, int root)
{int parent = root;int child = parent * 2 + 1;while (child < n){if (child + 1 < n && a[child] < a[child + 1]) child+=1;//把他移到右孩子那里if (a[child] > a[parent]){swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else break;}
}堆排序
void HeapSort(int* arr, int n)
{//建大堆//从最后一个根开始,就相当于它下面的都满足堆,就可以用向下调整算法for (int i = (n-1-1)/2; i >= 0; i--)//n-1-1是因为数组的最后一个元素下标是n-1{AdjustDown(arr, n, i);}//排序for (int i = n; i > 1; i--){swap(&arr[0],&arr[i - 1]);AdjustDown(arr, i-1, 0);}
}

六、快速排序:

三种快排方法:(一定要自己尝试着去写,会有一些坑,自己写才可以体会)

1.挖坑法

2.左右指针法

3.前后指针法


6.1挖坑法:

1.思想:

        记第一个数为key,要调整key的位置,使得左边的都要比key的小,右边的数都比key的大。

2.步骤:

        选出一个数据(一般是最左边或是最右边的)存放在key变量中,在该数据位置形成一个坑
        还是定义一个left和一个right,left从左向右走(当遇到大于key的值时停下来)。right从右向左走(当遇到小于key的值时停下来)。(若在最左边挖坑,则需要right先走;若在最右边挖坑,则需要left先走) 

        把right的那个小的数放在坑中,在把left那个位置的值放在right那个位置中

        重复操作,直到left>right时结束,完成一趟,把key放在了正确的位置

        最后用分治思想,分成左边和右边,递归。

 

//1.挖坑法的快速排序
void QuickSort(int* a,int left,int right)
{if (left >= right)//不能写成pivot==left,pivot-1与left不匹配,会报错{return;}int begin = left,end = right;int key = a[begin];//挖了一个关键字int pivot = begin;//挖了一个坑while (begin < end){//右边找小,一定要先右边找小,放在pivotwhile (begin < end&&a[end] >= key)//在这里也要判断begin < end,因为这里面end--{end--;}//小的放在左边的坑里,然后形成新的坑位a[pivot] = a[end];pivot = end;//左边找大while (begin < end && a[begin] <= key){begin++;}a[pivot] = a[begin];pivot = begin;	}//begin==end	a[pivot] = key;//[left,right]//[left,pivot-1]  pivot  [pivot+1,right]//如果左子区间和右子区间都有序,就全部有序。那就分治递归。QuickSort(a, left, pivot - 1);QuickSort(a, pivot+1, right);
}

6.2左右指针法

思路:
1、选出一个key,一般是最左边或是最右边的。
2、定义一个begin和一个end,begin从左向右走,end从右向左走。(需要注意的是:若选择最左边的数据作为key,则需要end先走;若选择最右边的数据作为key,则需要bengin先走)(考虑到最后的时候相遇点的和key交换)。
3、在走的过程中,若end遇到小于key的数,则停下,begin开始走,直到begin遇到一个大于key的数时,将begin和right的内容交换,end再次开始走,如此进行下去,直到begin和end最终相遇,此时将相遇点的内容与key交换即可。(选取最左边的值作为key)
4.此时key的左边都是小于key的数,key的右边都是大于key的数
5.将key的左序列和右序列再次进行这种单趟排序,如此反复操作下去,直到左右序列只有一个数据,或是左右序列不存在时

void QuickSort(int* a, int left, int right)
{if (left >= right){return;}int begin = left, end = right;int key = begin;//这里与挖坑法不同的地方,因为要交换key的那个数组中那个位置的数,而不是值while (begin < end){while (begin < end && a[end] >= a[key]){end--;}while (begin < end && a[begin] <= a[key]){begin++;}Swap(&a[begin], &a[end]);}Swap(&a[begin], &a[key]);QuickSort(a, left, begin - 1);QuickSort(a, begin + 1, right);
}

6.3前后指针法:

思路:
1、选出一个key,一般是最左边。
2、起始时,prev指针指向序列开头,cur指针指向prev+1。
3、让cur一直向前走,当遇到小于a[key]时,让prev向前走一格(这个值一定大于a[key],因为是cur走过的),然后a[cur]和a[prev]交换。

经过一次单趟排序,最终也能使得key左边的数据全部都小于key,key右边的数据全部都大于key。

然后也还是将key的左序列和右序列再次进行这种单趟排序,如此反复操作下去

用if(left>right)

{
        reurn;

}//跳出递归

void QuickSort(int* a, int left,int right)
{if (left >= right){return;}int index=GetMidIndex(a,left, right);swap(&a[left], &a[index]);int key = left;int prev = left;int cur = left+1;while (cur <= right){if (a[cur] < a[key]){prev++;swap(&a[cur], &a[prev]);}/*可以简写成cur++,   但是当时一定要注意不要放在if语句的前面,因为if语句里面有让cur与prev交换的,cur==right跳出循环,但是a[cur]超过数组的范围,会越界范围。while (cur<=right&&a[cur] >= a[key])   {cur++;}*/cur++;}swap(&a[prev], &a[key]);QuickSort(a, left, prev - 1);QuickSort(a, prev+1,right);
}

 6.4优化快排

三数取中法:取左端、中间、右端三个数,然后进行比较,将中值数当做key

否则有序时时间复杂度为O(N^2)

三数取中法可以套入三种方法中,这里我就写一种

//三数取中
int GetMidIndex(int* a, int left, int right)
{int mid = (left + right) / 2;if (a[mid] >= a[left]){if (a[mid] <= a[right]){return mid;}else{if (a[right] >= a[left]){return right;}else{return left;}}}else//a[left]>a[mid]{if (a[right] >= a[left]){return left;}else{if (a[right] >= a[mid]){return right;}else{return mid;}}}
}//交换
void swap(int* a, int* b)
{int tmp = *a;*a = *b;*b = tmp;
}//前后指针法
void QuickSort(int* a, int left,int right)
{if (left >= right){return;}int index=GetMidIndex(a,left, right);swap(&a[left], &a[index]);int key = left;int prev = left;int cur = left+1;while (cur <= right){if (a[cur] < a[key]){prev++;swap(&a[cur], &a[prev]);}cur++;}swap(&a[prev], &a[key]);QuickSort(a, left, prev - 1);QuickSort(a, prev+1,right);
}

七、归并排序:

思路:

1.不断的分割数据,让数据的每一段都有序(一个数据相当于有序)

2.当所有子序列有序的时候,在把子序列归并,形成更大的子序列,最终整个数组有序。

 !!!需要开一个_MergeSort,而不是直接在MergeSort中直接递归,是因为MergeSort中有一个malloc

归并排序很像二叉树中的后序思想,先递归,递归到最后的时候再合并。!!!

//归并排序
void _MergeSort(int* a, int left, int right, int* tmp)//在这个函数中调用递归{if (left >= right){return;}int mid = (left + right) >> 1;_MergeSort(a, left, mid, tmp);_MergeSort(a, mid+1, right, tmp);//合并int begin1 = left, end1 = mid;int begin2 = mid + 1, end2 = right;int i = left;while (begin1 <= end1 && begin2 <= end2){if (a[begin1] <= a[begin2]){tmp[i++] = a[begin1++];}else{tmp[i++] = a[begin2++];}}while (begin1 <= end1){tmp[i++] = a[begin1++];}while (begin2 <= end2){tmp[i++] = a[begin2++];}for (int j = left; j <= right; j++){a[j] = tmp[j];}
}
void MergeSort(int* a, int n)
{int* tmp = (int*)malloc(sizeof(int) * n);_MergeSort(a, 0, n - 1, tmp);free(tmp);
}

八、桶排序:

思路:大问题化小

桶排序 (Bucket sort)或所谓的箱排序,是一种分块的排序算法,工作的原理是将数组分到有限数量的桶里,每个桶的大小都相等。每个桶再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)

把待排序序列(数组)中的数据根据函数映射方法分配到若干个桶中,在分别对各个桶进行排序,最后依次按顺序取出桶中的数据。
适用于数据分配均匀,数据比较大,相对集中的情况。

//桶排序
void bucket_sort(int a[],int size,int bucket_size) {int i,j;        //数组,数组长度,桶的大小//定义动态的指针数组KeyNode **bucket_num = (KeyNode **)malloc(bucket_size * sizeof(KeyNode*));for(i = 0;i < bucket_size;i++) {bucket_num[i] = (KeyNode*)malloc(sizeof(KeyNode));//为每个链表定义头结点 bucket_num[i]->num = 0;   bucket_num[i]->next = NULL;   //指针变量初始化为空}for(j = 0;j < size;j++) //准备插入{KeyNode *node = (KeyNode *)malloc(sizeof(KeyNode));//定义一个节点 node->num = a[j];    //数据域存数据 node->next = NULL;	//指向空int index = a[j]/100;  //映射函数 计算桶号KeyNode *p = bucket_num[index];//p指向链表的头//链表结构的插入排序while(p->next != NULL && p->next->num <= node->num){p = p->next;	//1.链表为空,p->next==NULL,进入不了循环 }					//2.链表不为空,因为链表从无开始按顺序插入,数据为有序的,//可以找到    前一个节点 <= node <=后一个节点//节点插入链表 node->next = p->next;p->next = node;(bucket_num[index]->num)++;	//记录一下该链表中有几个有效节点 }//打印结果KeyNode * k = NULL;  //定义一个空的结构体指针用于储存输出结果for(i = 0;i < bucket_size;i++){//for(k = bucket_num[i]->next;k!=NULL;k=k->next)//通过最后一个指针指向空k = bucket_num[i]->next;for(int m=0;m<bucket_num[i]->num;m++)   //通过头指针记录节点数{printf("%d ",k->num);k=k->next;}		printf("\n");
}

九、计数排序:

一种特殊的排序,唯一种没有比较的排序(指没有前后比较,还是有交换的)

以数组的下标当做数值,有这个数的时候a[i]++;

 局限:适用于整数。数要求集中(否则空间的浪费大)

9.1绝对映射:

int * countingSort1(int arr[],int count,int max) {int index = 0;int *tmpArr = (int *)malloc(max*sizeof(int));int *result = (int *)malloc(max*sizeof(int));for(int k = 0;k<max;k++) {tmpArr[k] = 0;}for (int i = 0; i<count; i++) {tmpArr[arr[i]]++;}for (int j = 0; j<max; j++) {while (tmpArr[j]) {result[index++] = j;tmpArr[j]--;}}free(tmpArr);tmpArr = NULL;return result;
}

9.2现对映射:

void CountSort(int* a, int n)
{int max = a[0], min = a[0];for (int i = 0; i < n; i++){if (a[i] > max) max = a[i];if (a[i] < min) min = a[i];}int range = max - min + 1;int* count = (int*)malloc(sizeof(int) * range);memset(count, 0, sizeof(int) * range);for (int i = 0; i < n; i++){count[a[i] - min]++;}int i = 0;for (int j = 0; j < range; j++){while (count[j]--){a[i++] = j + min;}}free(count);
}

十、基数排序: 

原理:是将整数按位数切割成不同的数字,然后按每个位数分别比较。

#include<math.h>
testBS()
{inta[] = {2, 343, 342, 1, 123, 43, 4343, 433, 687, 654, 3};int *a_p = a;//计算数组长度intsize = sizeof(a) / sizeof(int);//基数排序bucketSort3(a_p, size);//打印排序后结果inti;for(i = 0; i < size; i++){printf("%d\n", a[i]);}intt;scanf("%d", t);
}
//基数排序
voidbucketSort3(int *p, intn)
{//获取数组中的最大数intmaxNum = findMaxNum(p, n);//获取最大数的位数,次数也是再分配的次数。intloopTimes = getLoopTimes(maxNum);inti;//对每一位进行桶分配for(i = 1; i <= loopTimes; i++){sort2(p, n, i);}
}
//获取数字的位数
intgetLoopTimes(intnum)
{intcount = 1;inttemp = num / 10;while(temp != 0){count++;temp = temp / 10;}returncount;
}
//查询数组中的最大数
intfindMaxNum(int *p, intn)
{inti;intmax = 0;for(i = 0; i < n; i++){if(*(p + i) > max){max = *(p + i);}}returnmax;
}
//将数字分配到各自的桶中,然后按照桶的顺序输出排序结果
voidsort2(int *p, intn, intloop)
{//建立一组桶此处的20是预设的根据实际数情况修改intbuckets[10][20] = {};//求桶的index的除数//如798个位桶index=(798/1)%10=8//十位桶index=(798/10)%10=9//百位桶index=(798/100)%10=7//tempNum为上式中的1、10、100inttempNum = (int)pow(10, loop - 1);inti, j;for(i = 0; i < n; i++){introw_index = (*(p + i) / tempNum) % 10;for(j = 0; j < 20; j++){if(buckets[row_index][j] == NULL){buckets[row_index][j] = *(p + i);break;}}}//将桶中的数,倒回到原有数组中intk = 0;for(i = 0; i < 10; i++){for(j = 0; j < 20; j++){if(buckets[i][j] != NULL){*(p + k) = buckets[i][j];buckets[i][j] = NULL;k++;}}}
}

相关文章:

十大排序算法(冒泡排序、插入排序、选择排序、希尔排序、堆排序、快排、归并排序、桶排序、计数排序、基数排序)

目录 一、冒泡排序&#xff1a; 二、插入排序&#xff1a; 三、选择排序&#xff1a; 四、希尔排序&#xff1a; 五、堆排序&#xff1a; 六、快速排序&#xff1a; 6.1挖坑法&#xff1a; 6.2左右指针法 6.3前后指针法&#xff1a; 七、归并排序&#xff1a; 八、桶…...

matplotlib 画多子图的时候添加图例/legend

一开始搞不懂图例是什么意思&#xff0c;以为是整个图&#xff0c;最后发现原来图例就是代码中的legend&#xff1a; 子图的图例&#xff08;legend&#xff09;用于解释图表中各条线、点或其他元素所代表的含义。图例通常位于图表的一角&#xff0c;以帮助观众理解图表中展示的…...

手写一个线程池

自己手动写一个线程池的必要条件需要先了解我们使用的线程池的功能。为什么会有线程池&#xff1f;这是为了减少线程创建和销毁的开销。复用线程的目的。为了达到这个目的。预计方案是&#xff1a;需要一个存放任务的队列&#xff0c;主线程相当于生产者&#xff0c;在这个队列…...

Spring Boot 多环境配置

Spring Boot 多环境配置 在现代的软件开发中&#xff0c;通常需要将应用程序部署到不同的环境中&#xff0c;如开发环境、生产环境和测试环境等。每个环境可能需要不同的配置参数&#xff0c;例如数据库连接信息、日志级别等。在 Spring Boot 中&#xff0c;我们可以通过简单的…...

【Python】一文带你详解sys.executable函数的作用

【Python】一文带你详解sys.executable函数的作用 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望得到您的订阅和支…...

thingsboard如何自定义udp-transport

0、参考netty实现udp的文章 https://github.com/narkhedesam/Netty-Simple-UDP-TCP-server-client/blob/master/netty-udp/src/com/sam/netty_udp/server/MessageDecoder.java 调试工具使用的是:卓岚TCP&UDP调试工具 1、在common\transport下面创建udp模块,仿照mqtt的创…...

【汇编】#3 8086与数据有关的寻址方式

文章目录 操作码与操作数1. 8086处理器的与数据有关的寻址方式1.1 立即数寻址方式1.2 寄存器寻址方式 2. 有效&#xff08;偏移&#xff09;地址&#xff08;effective address&#xff0c;EA&#xff09;与缺省段寄存器选择tips:段跨越前缀2.1 直接寻址tips:直接寻址与立即寻址…...

【数据结构】单链表的层层实现!! !

关注小庄 顿顿解馋(●’◡’●) 上篇回顾 我们上篇学习了本质为数组的数据结构—顺序表&#xff0c;顺序表支持下标随机访问而且高速缓存命中率高&#xff0c;然而可能造成空间的浪费&#xff0c;同时增加数据时多次移动会造成效率低下&#xff0c;那有什么解决之法呢&#xff…...

丰田研究所(TRI)最新成果——可实现全身操控的软体机器人Punyo

文 | BFT机器人 人形机器人在近年的科技浪潮中迅速崛起&#xff0c;成为了各界瞩目的焦点&#xff0c;众多企业纷纷推出自家的机器人模型&#xff0c;但仔细观察&#xff0c;不难发现它们中的许多在操作方式上仍显得颇为相似。这些典型的人形机器人&#xff0c;以其机械臂和抓…...

【PyTorch实战演练】深入剖析MTCNN(多任务级联卷积神经网络)并使用30行代码实现人脸识别

文章目录 0. 前言1. 级联神经网络介绍2. MTCNN介绍2.1 MTCNN提出背景2.2 MTCNN结构 3. MTCNN PyTorch实战3.1 facenet_pytorch库中的MTCNN3.2 识别图像数据3.3 人脸识别3.4 关键点定位 0. 前言 按照国际惯例&#xff0c;首先声明&#xff1a;本文只是我自己学习的理解&#xff…...

MFC中字符串string类型和CString类型互转方法

在Microsoft Foundation Classes (MFC)中&#xff0c;CString是一个非常方便的类&#xff0c;用于处理C风格的字符串。有时&#xff0c;你可能需要在MFC的CString和C标准库中的std::string之间进行转换。下面是如何在两者之间进行转换的方法&#xff1a; CString转std::string…...

Jmeter-使用http proxy代理录制脚本

Jmeter-使用http proxy代理录制脚本 第1步&#xff1a;打卡jmeter工具新增1个线程组 第2步&#xff1a;给线程组添加1个HTTP请求默认值 第3步&#xff1a;设置下HTTP请求默认值第4步&#xff1a;在工作台中新增1个----HTTP代理服务器 第5步&#xff1a;设置HTTP代理服务器 …...

C++训练营:new 运算符

大家好&#xff1a; 衷心希望各位点赞。 您的问题请留在评论区&#xff0c;我会及时回答。 一、new 运算符 new 运算符用于动态分配一片内存空间&#xff0c;并返回这片内存空间的首地址&#xff0c;可将该首地址存入一个指针变量中&#xff0c;主要有以下三种格式。 二、格…...

C# 用Trace.WriteLine输出调试信息无法查看

写程序就会遇见BUG&#xff0c;这时候在代码不同部位输出一些标记的信息对查找错误非常有必要&#xff0c;一般情况下我们都是使用Console.WriteLine()方法来打印信息到控制台窗口&#xff0c;但有时候使用Console.WriteLine()方法会存在不方便的情况&#xff0c;比如鄙人遇到的…...

【Echarts】柱状图上方显示数字以及自定义值,标题和副标题居中,鼠标上显示信息以及自定义信息

欢迎来到《小5讲堂》 大家好&#xff0c;我是全栈小5。 这是《前端》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解&#xff0c; 特别是针对知识点的概念进行叙说&#xff0c;大部分文章将会对这些概念进行实际例子验证&#xff0c;以此达到加深对知识点的理解和掌握…...

HTML 语义化:构建优质网页的关键

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…...

Flutter入门学习——Flutter和Dart

因为工作的需要&#xff0c;也为了个人发展&#xff0c;现在的话&#xff0c;转战Flutter跨端开发了&#xff0c;虽然目前的项目只发了android端&#xff0c;但是那天尝试了一下Ios的打包流程&#xff0c;也能运行&#xff0c;只是IOS那边的打包稍微复杂一些。 差不多学习了一…...

C++中的内存管理方式

一、C内存管理方式简介 C语言中的内存管理方式在C中可以继续使用&#xff0c;但是在有些地方就无能为力&#xff0c;而且使用起来比较麻烦。因此C中引入了自己的内存管理方式&#xff0c;通过new和delete操作符进行动态内存管理。 二、new语法 new可以申请1个或多个空间&…...

macos m1 arm芯片 使用jpype报错 FileNotFoundError: [Errno 2] JVM DLL not found

startJVM(jpype.getDefaultJVMPath()) 报错 Traceback (most recent call last):File "/Users/thomas990p/PycharmProjects/tuya/volcano-biz-scripts/WenKongFa/FinalCode/java2python/CallJavaAPI.py", line 12, in <module>startJVM(jpype.getDefaultJVMPa…...

Hive中UNION ALL和UNION的区别

1.概述 Hive官方提供了一种联合查询的语法&#xff0c;原名为Union Syntax&#xff0c;用于联合两个表的记录进行查询&#xff0c;此处的联合和join是不同的&#xff0c;join是将两个表的字段拼接到一起&#xff0c;而union是将两个表的记录拼接在一起。 换言之&#xff0c; jo…...

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周&#xff0c;有很多同学在写期末Java web作业时&#xff0c;运行tomcat出现乱码问题&#xff0c;经过多次解决与研究&#xff0c;我做了如下整理&#xff1a; 原因&#xff1a; IDEA本身编码与tomcat的编码与Windows编码不同导致&#xff0c;Windows 系统控制台…...

树莓派超全系列教程文档--(62)使用rpicam-app通过网络流式传输视频

使用rpicam-app通过网络流式传输视频 使用 rpicam-app 通过网络流式传输视频UDPTCPRTSPlibavGStreamerRTPlibcamerasrc GStreamer 元素 文章来源&#xff1a; http://raspberry.dns8844.cn/documentation 原文网址 使用 rpicam-app 通过网络流式传输视频 本节介绍来自 rpica…...

Java 8 Stream API 入门到实践详解

一、告别 for 循环&#xff01; 传统痛点&#xff1a; Java 8 之前&#xff0c;集合操作离不开冗长的 for 循环和匿名类。例如&#xff0c;过滤列表中的偶数&#xff1a; List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

Oracle查询表空间大小

1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...

dify打造数据可视化图表

一、概述 在日常工作和学习中&#xff0c;我们经常需要和数据打交道。无论是分析报告、项目展示&#xff0c;还是简单的数据洞察&#xff0c;一个清晰直观的图表&#xff0c;往往能胜过千言万语。 一款能让数据可视化变得超级简单的 MCP Server&#xff0c;由蚂蚁集团 AntV 团队…...

今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存

文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...

HarmonyOS运动开发:如何用mpchart绘制运动配速图表

##鸿蒙核心技术##运动开发##Sensor Service Kit&#xff08;传感器服务&#xff09;# 前言 在运动类应用中&#xff0c;运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据&#xff0c;如配速、距离、卡路里消耗等&#xff0c;用户可以更清晰…...

在Ubuntu24上采用Wine打开SourceInsight

1. 安装wine sudo apt install wine 2. 安装32位库支持,SourceInsight是32位程序 sudo dpkg --add-architecture i386 sudo apt update sudo apt install wine32:i386 3. 验证安装 wine --version 4. 安装必要的字体和库(解决显示问题) sudo apt install fonts-wqy…...

安宝特案例丨Vuzix AR智能眼镜集成专业软件,助力卢森堡医院药房转型,赢得辉瑞创新奖

在Vuzix M400 AR智能眼镜的助力下&#xff0c;卢森堡罗伯特舒曼医院&#xff08;the Robert Schuman Hospitals, HRS&#xff09;凭借在无菌制剂生产流程中引入增强现实技术&#xff08;AR&#xff09;创新项目&#xff0c;荣获了2024年6月7日由卢森堡医院药剂师协会&#xff0…...

【Go语言基础【12】】指针:声明、取地址、解引用

文章目录 零、概述&#xff1a;指针 vs. 引用&#xff08;类比其他语言&#xff09;一、指针基础概念二、指针声明与初始化三、指针操作符1. &&#xff1a;取地址&#xff08;拿到内存地址&#xff09;2. *&#xff1a;解引用&#xff08;拿到值&#xff09; 四、空指针&am…...