【数据结构入门精讲 | 第十篇】考研408排序算法专项练习(二)

本文主要是介绍【数据结构入门精讲 | 第十篇】考研408排序算法专项练习(二),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上文中我们进行了排序算法的判断题、选择题的专项练习,在这一篇中我们将进行排序算法中编程题的练习。

在这里插入图片描述

目录

    • 编程题
      • R7-1 字符串的冒泡排序
      • R7-1 抢红包
      • R7-1 PAT排名汇总
      • R7-2 统计工龄
      • R7-1 插入排序还是堆排序
      • R7-2 龙龙送外卖
      • R7-3 家谱处理

编程题

R7-1 字符串的冒泡排序

我们已经知道了将N个整数按从小到大排序的冒泡排序法。本题要求将此方法用于字符串序列,并对任意给定的K(<N),输出扫描完第K遍后的中间结果序列。

输入格式:

输入在第1行中给出N和K(1≤K<N≤100),此后N行,每行包含一个长度不超过10的、仅由小写英文字母组成的非空字符串。

输出格式:

输出冒泡排序法扫描完第K遍后的中间结果序列,每行包含一个字符串。

输入样例:

6 2
best
cat
east
a
free
day

输出样例:

best
a
cat
day
east
free

这里要知道一个知识点:先比较两个字符串的第一个字符的ASCII码值),如果第一个字符串的第一个字符大于第二个字符串的第一个字符,则返回1;反之,则返回-1;若两个字符相等,则进行下一对字符的比较,(下面的比较均是ASCII码值的比较)直到出现有一对字符不同则返回相应的值(1或-1)。若两个字符串所有的字符都相同,则返回数字0。

#include <stdio.h>
#include <string.h>
int main()
{int n,k;scanf("%d%d",&n,&k);char a[100][11]={0},x[11];for(int i=0;i<n;i++)//用二维数组存储{scanf("%s",&a[i]);}for(int i=0;i<k;i++)//对于每个元素而言{for(int j=0;j<n-i-1;j++)//每个元素都比较n-i-1次{if(strcmp(a[j],a[j+1])>0)//交换{strcpy(x,a[j]);strcpy(a[j],a[j+1]);strcpy(a[j+1],x);}}}for(int i=0;i<n;i++)printf("%s\n",a[i]);}

R7-1 抢红包

没有人没抢过红包吧…… 这里给出N个人之间互相发红包、抢红包的记录,请你统计一下他们抢红包的收获。

输入格式:

输入第一行给出一个正整数N(≤104),即参与发红包和抢红包的总人数,则这些人从1到N编号。随后N行,第i行给出编号为i的人发红包的记录,格式如下:

K N1 P1 ⋯ NK PK

其中K(0≤K≤20)是发出去的红包个数,Ni是抢到红包的人的编号,Pi(>0)是其抢到的红包金额(以分为单位)。注意:对于同一个人发出的红包,每人最多只能抢1次,不能重复抢。

输出格式:

按照收入金额从高到低的递减顺序输出每个人的编号和收入金额(以元为单位,输出小数点后2位)。每个人的信息占一行,两数字间有1个空格。如果收入金额有并列,则按抢到红包的个数递减输出;如果还有并列,则按个人编号递增输出。

输入样例:

10
3 2 22 10 58 8 125
5 1 345 3 211 5 233 7 13 8 101
1 7 8800
2 1 1000 2 1000
2 4 250 10 320
6 5 11 9 22 8 33 7 44 10 55 4 2
1 3 8800
2 1 23 2 123
1 8 250
4 2 121 4 516 7 112 9 10

输出样例:

1 11.63
2 3.63
8 3.63
3 2.11
7 1.69
6 -1.67
9 -2.18
10 -3.26
5 -3.26
4 -12.32
#include <iostream>
#include <algorithm>
#include <cstdio>using namespace std;// 定义结构体redbag,用于存储每个人的红包信息
struct redbag{int id; // 编号int count = 0; // 收到红包的个数int money = 0; // 最终收到的金额
};// 自定义比较函数cmp,用于sort排序时指定排序规则
bool cmp(redbag &r1, redbag &r2){if(r1.money != r2.money){ // 按照最终收到的金额从大到小排序return r1.money > r2.money;}else if(r1.count != r2.count){ // 如果最终收到金额相等,则按照收到红包的个数从大到小排序return r1.count > r2.count;}else{ // 如果最终收到金额和收到红包的个数都相等,则按照编号从小到大排序return r1.id < r2.id;}
}int main(){int n; // 总人数cin >> n;redbag red[n+1]; // 定义数组red,用于存储每个人的信息int k, card, mon; // k表示某个人发出的红包数,card表示发红包的人的编号,mon表示红包金额for(int i = 1; i <= n; i++){cin >> k;red[i].id = i;for(int j = 1; j <= k; j++){cin >> card >> mon;red[i].money -= mon; // 发出红包的人需要减掉红包金额red[card].money += mon; // 收到红包的人需要加上红包金额red[card].count++; // 收到红包的人收到红包数加1}}sort(red+1, red+n+1, cmp); // 使用sort排序,按照指定的cmp规则排序for(int i = 1; i <= n; i++){printf("%d %.2lf\n", red[i].id, red[i].money / 100.0); // 输出每个人最终获得的金额,需要将单位从分转换为元,并保留两位小数}return 0;
}

R7-1 PAT排名汇总

计算机程序设计能力考试(Programming Ability Test,简称PAT)旨在通过统一组织的在线考试及自动评测方法客观地评判考生的算法设计与程序设计实现能力,科学的评价计算机程序设计人才,为企业选拔人才提供参考标准。

每次考试会在若干个不同的考点同时举行,每个考点用局域网,产生本考点的成绩。考试结束后,各个考点的成绩将即刻汇总成一张总的排名表。

现在就请你写一个程序自动归并各个考点的成绩并生成总排名表。

输入格式:

输入的第一行给出一个正整数N(≤100),代表考点总数。随后给出N个考点的成绩,格式为:首先一行给出正整数K(≤300),代表该考点的考生总数;随后K行,每行给出1个考生的信息,包括考号(由13位整数字组成)和得分(为[0,100]区间内的整数),中间用空格分隔。

输出格式:

首先在第一行里输出考生总数。随后输出汇总的排名表,每个考生的信息占一行,顺序为:考号、最终排名、考点编号、在该考点的排名。其中考点按输入给出的顺序从1到N编号。考生的输出须按最终排名的非递减顺序输出,获得相同分数的考生应有相同名次,并按考号的递增顺序输出。

输入样例:

2
5
1234567890001 95
1234567890005 100
1234567890003 95
1234567890002 77
1234567890004 85
4
1234567890013 65
1234567890011 25
1234567890014 100
1234567890012 85

输出样例:

9
1234567890005 1 1 1
1234567890014 1 2 1
1234567890001 3 1 2
1234567890003 3 1 2
1234567890004 5 1 4
1234567890012 5 2 2
1234567890002 7 1 5
1234567890013 8 2 3
1234567890011 9 2 4
#include<bits/stdc++.h>
using namespace std;struct student {string s; // 考生姓名int score; // 考生成绩int id; // 考场编号int rank; // 考场内排名int zrank; // 总体排名} a[30000]; // 存储考生信息的数组bool cmp(student x,student y)
{if(x.score != y.score) return x.score > y.score; // 根据成绩从高到低排序elsereturn x.s < y.s; // 成绩相同时,按照姓名的字典序排序}int main() {int n, cnt = 0, k; // n为考生人数,cnt为考生总数,k为每个考场的考生人数cin >> n; // 输入考生人数for(int i = 0; i < n; i++){cin >> k; // 输入每个考场的考生人数for(int j = 0; j < k; j++) {cin >> a[j + cnt].s >> a[j + cnt].score; // 输入考生姓名和成绩a[j + cnt].id = i + 1; // 设置考场编号}sort(a + cnt, a + cnt + k, cmp); // 对当前考场的考生按照成绩从高到低排序a[cnt].rank = 1; // 第一名考生的排名为1for(int j = 1; j < k; j++) {if(a[cnt + j - 1].score == a[cnt + j].score)a[cnt + j].rank = a[cnt + j - 1].rank; // 成绩相同的考生排名相同elsea[cnt + j].rank = j + 1; // 不同成绩的考生排名递增}cnt = cnt + k; // 更新考生总数}sort(a, a + cnt, cmp); // 对所有考生按照成绩从高到低排序a[0].zrank = 1; // 总体排名第一名考生的排名为1for(int j = 1; j < cnt; j++){if(a[j - 1].score == a[j].score)a[j].zrank = a[j - 1].zrank; // 成绩相同的考生总体排名相同elsea[j].zrank = j + 1; // 不同成绩的考生总体排名递增}cout << cnt << endl; // 输出考生总数for(int i = 0; i < cnt; i++) cout << a[i].s << " " << a[i].zrank << " " << a[i].id << " " << a[i].rank << endl; // 输出每个考生的姓名、总体排名、考场编号和考场排名return 0;
}

R7-2 统计工龄

给定公司N名员工的工龄,要求按工龄增序输出每个工龄段有多少员工。

输入格式:

输入首先给出正整数N(≤105),即员工总人数;随后给出N个整数,即每个员工的工龄,范围在[0, 50]。

输出格式:

按工龄的递增顺序输出每个工龄的员工个数,格式为:“工龄:人数”。每项占一行。如果人数为0则不输出该项。

输入样例:

8
10 2 0 5 7 2 5 2

输出样例:

0:1
2:3
5:2
7:1
10:1
#include <stdio.h>  // 包含标准输入输出头文件int main()
{int n;scanf("%d", &n);  // 输入待排序数字的个数int a[51] = {0};  // 初始化一个长度为 51 的整型数组,并初始化所有元素为 0// 循环读入待排序的数字,统计各个年龄出现的次数for (int i = 0; i < n; i++){int age;scanf("%d", &age);  // 输入当前数字a[age]++;  // 对应年龄出现次数加一}// 输出统计结果for (int i = 0; i <= 50; i++){if (a[i] != 0)  // 如果该年龄出现次数不为 0printf("%d:%d\n", i, a[i]);  // 输出该年龄及其出现次数}
}

R7-1 插入排序还是堆排序

根据维基百科的定义:

插入排序是迭代算法,逐一获得输入数据,逐步产生有序的输出序列。每步迭代中,算法从输入序列中取出一元素,将之插入有序序列中正确的位置。如此迭代直到全部元素有序。

堆排序也是将输入分为有序和无序两部分,迭代地从无序部分找出最大元素放入有序部分。它利用了大根堆的堆顶元素最大这一特征,使得在当前无序区中选取最大元素变得简单。

现给定原始序列和由某排序算法产生的中间序列,请你判断该算法究竟是哪种排序算法?

输入格式:

输入在第一行给出正整数 N (≤100);随后一行给出原始序列的 N 个整数;最后一行给出由某排序算法产生的中间序列。这里假设排序的目标序列是升序。数字间以空格分隔。

输出格式:

首先在第 1 行中输出Insertion Sort表示插入排序、或Heap Sort表示堆排序;然后在第 2 行中输出用该排序算法再迭代一轮的结果序列。题目保证每组测试的结果是唯一的。数字间以空格分隔,且行首尾不得有多余空格。

输入样例 1:

10
3 1 2 8 7 5 9 4 6 0
1 2 3 7 8 5 9 4 6 0

输出样例 1:

Insertion Sort
1 2 3 5 7 8 9 4 6 0

输入样例 2:

10
3 1 2 8 7 5 9 4 6 0
6 4 5 1 0 3 2 7 8 9

输出样例 2:

Heap Sort
5 4 3 1 0 2 6 7 8 9
#include<stdio.h>// 插入排序函数
void insert_sort(int *am,int n)
{int i,j;// 找到第一个逆序的元素位置for(i=2; i<=n&&am[i-1]<=am[i]; i++);// 将该元素插入到合适的位置int key=am[i];for(j=i-1;j>=1;j--){if(am[j]>key) am[j+1]=am[j];else break;}am[j+1]=key;
}// 交换两个数的值
void swap(int *a,int *b)
{int t=*a;*a=*b;*b=t;
}// 最大堆调整函数
void maxheapify(int *am,int i,int n)
{int l,r,max;while(1){l=i<<1;r=i<<1|1;// 找到左右子节点中的最大值if(l<=n&&am[l]>am[i]) max=l;else max=i;if(r<=n&&am[r]>am[max]) max=r;// 如果最大值是当前节点,则满足最大堆性质,结束循环if(max==i) return;// 否则,交换当前节点与最大值节点的值,并继续调整swap(&am[i],&am[max]);i=max;}
}// 堆排序函数
void heap_sort(int *am,int n)
{int i,size;// 找到第一个逆序的元素位置for(i=n;i>1&&am[i]>am[i-1];i--);// 交换第一个元素与最后一个元素,并调整堆size=i-1;swap(&am[1],&am[i]);maxheapify(am,1,size);
}int main(void)
{int n;scanf("%d",&n);int num[120]={0};int am[120]={0};// 输入数字序列for(int i=1;i<=n;i++)scanf("%d",num+i);// 输入比较的数字序列for(int i=1;i<=n;i++)scanf("%d",am+i);// 根据第一个数字的大小选择排序算法if(am[1]<am[2]){printf("Insertion Sort\n");// 使用插入排序算法进行排序insert_sort(am,n);// 输出排序结果for(int i=1;i<=n;i++){if(i!=n) printf("%d ",am[i]);else printf("%d\n",am[i]);}}else{printf("Heap Sort\n");// 使用堆排序算法进行排序heap_sort(am,n);// 输出排序结果for(int i=1;i<=n;i++){if(i!=n) printf("%d ",am[i]);else printf("%d\n",am[i]);}}return 0;
}

R7-2 龙龙送外卖

龙龙是“饱了呀”外卖软件的注册骑手,负责送帕特小区的外卖。帕特小区的构造非常特别,都是双向道路且没有构成环 —— 你可以简单地认为小区的路构成了一棵树,根结点是外卖站,树上的结点就是要送餐的地址。

每到中午 12 点,帕特小区就进入了点餐高峰。一开始,只有一两个地方点外卖,龙龙简单就送好了;但随着大数据的分析,龙龙被派了更多的单子,也就送得越来越累……

看着一大堆订单,龙龙想知道,从外卖站出发,访问所有点了外卖的地方至少一次(这样才能把外卖送到)所需的最短路程的距离到底是多少?每次新增一个点外卖的地址,他就想估算一遍整体工作量,这样他就可以搞明白新增一个地址给他带来了多少负担。

输入格式:

输入第一行是两个数 N 和 M (2≤N≤105, 1≤M≤105),分别对应树上节点的个数(包括外卖站),以及新增的送餐地址的个数。

接下来首先是一行 N 个数,第 i 个数表示第 i 个点的双亲节点的编号。节点编号从 1 到 N,外卖站的双亲编号定义为 −1。

接下来有 M 行,每行给出一个新增的送餐地点的编号 Xi。保证送餐地点中不会有外卖站,但地点有可能会重复。

为了方便计算,我们可以假设龙龙一开始一个地址的外卖都不用送,两个相邻的地点之间的路径长度统一设为 1,且从外卖站出发可以访问到所有地点。

注意:所有送餐地址可以按任意顺序访问,且完成送餐后无需返回外卖站。

输出格式:

对于每个新增的地点,在一行内输出题目需要求的最短路程的距离。

输入样例:

7 4
-1 1 1 1 2 2 3
5
6
2
4

输出样例:

2
4
4
6
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+5;
int n,m,t,a[N],maxx,dis[N],sum,k,vis[N],s;
vector<int>e[N];// DFS函数,用于计算每个节点到根节点的距离
void dfs(int x,int d){dis[x]=d; // 记录节点x到根节点的距离为dfor(int i=0;i<e[x].size();i++) dfs(e[x][i],d+1); // 递归计算子节点到根节点的距离
}int main(){cin>>n>>m;for(int i=1;i<=n;i++){scanf("%d",&a[i]); // 输入每个节点的父节点编号if(a[i]==-1) s=i; // 找到根节点的编号else e[a[i]].push_back(i); // 将节点i加入到其父节点a[i]的子节点列表中}dfs(s,0); // 从根节点开始进行DFS,计算每个节点到根节点的距离vis[s]=1; // 标记根节点已经访问过for(int i=0;i<m;i++){scanf("%d",&t),k=t;while(!vis[k]) vis[k]=1,k=a[k]; // 从节点t一直向上找,直到找到已经访问过的节点maxx=max(maxx,dis[t]); // 更新最大距禝sum+=dis[t]-dis[k]; // 累加每次找到的路径长度printf("%d\n",sum*2-maxx); // 输出结果}return 0;
}

R7-3 家谱处理

人类学研究对于家族很感兴趣,于是研究人员搜集了一些家族的家谱进行研究。实验中,使用计算机处理家谱。为了实现这个目的,研究人员将家谱转换为文本文件。下面为家谱文本文件的实例:

JohnRobertFrankAndrewNancyDavid

家谱文本文件中,每一行包含一个人的名字。第一行中的名字是这个家族最早的祖先。家谱仅包含最早祖先的后代,而他们的丈夫或妻子不出现在家谱中。每个人的子女比父母多缩进2个空格。以上述家谱文本文件为例,John这个家族最早的祖先,他有两个子女RobertNancyRobert有两个子女FrankAndrewNancy只有一个子女David

在实验中,研究人员还收集了家庭文件,并提取了家谱中有关两个人关系的陈述语句。下面为家谱中关系的陈述语句实例:

John is the parent of Robert
Robert is a sibling of Nancy
David is a descendant of Robert

研究人员需要判断每个陈述语句是真还是假,请编写程序帮助研究人员判断。

输入格式:

输入首先给出2个正整数N(2≤N≤100)和M(≤100),其中N为家谱中名字的数量,M为家谱中陈述语句的数量,输入的每行不超过70个字符。

名字的字符串由不超过10个英文字母组成。在家谱中的第一行给出的名字前没有缩进空格。家谱中的其他名字至少缩进2个空格,即他们是家谱中最早祖先(第一行给出的名字)的后代,且如果家谱中一个名字前缩进k个空格,则下一行中名字至多缩进k+2个空格。

在一个家谱中同样的名字不会出现两次,且家谱中没有出现的名字不会出现在陈述语句中。每句陈述语句格式如下,其中XY为家谱中的不同名字:

X is a child of Y
X is the parent of Y
X is a sibling of Y
X is a descendant of Y
X is an ancestor of Y

输出格式:

对于测试用例中的每句陈述语句,在一行中输出True,如果陈述为真,或False,如果陈述为假。

输入样例:

6 5
JohnRobertFrankAndrewNancyDavid
Robert is a child of John
Robert is an ancestor of Andrew
Robert is a sibling of Nancy
Nancy is the parent of Frank
John is a descendant of Andrew

输出样例:

True
True
True
False
False
#include<iostream>
#include<vector>
#include<map>
#include<algorithm>
using namespace std;
const int MAX=105;
vector<string>name(MAX); // 存储每个人的姓名
map<string,string>parent; // 存储每个人与其父母的关系
string anc="anc"; // 用于表示祖先节点的字符串int main(){int N,M;cin>>N>>M;getchar(); // 读取换行符string s;while(N--){getline(cin,s); // 逐行读取输入int count1=count(s.begin(),s.end(),' '); // 统计该行中空格的个数s=s.substr(count1); // 截取出姓名if(count1==0){parent[s]=anc; // 如果没有空格,说明是祖先节点name[0]=s; // 将祖先节点的姓名存入name[0]}else{parent[s]=name[count1/2-1]; // 否则,将该节点与其父母的关系存入parent中name[count1/2]=s; // 将该节点的姓名存入name中}}while(M--){string a,b,c,d;cin>>a>>d>>d>>b>>d>>c; // 依次读取查询的条件if(b=="child"){ // 如果查询条件是childif(parent[a]==c)cout<<"True"<<endl; // 如果a的父亲是c,则输出Trueelse cout<<"False"<<endl; // 否则输出False}if(b=="ancestor"){ // 如果查询条件是ancestorwhile(parent[c]!=a&&parent[c]!=anc)c=parent[c]; // 从c开始向上查找,直到找到a或者祖先节点if(parent[c]==anc)cout<<"False"<<endl; // 如果找到祖先节点,则输出Falseelse cout<<"True"<<endl; // 否则输出True}if(b=="sibling"){ // 如果查询条件是siblingif(parent[a]==parent[c])cout<<"True"<<endl; // 如果a和c的父母相同,则输出Trueelse cout<<"False"<<endl; // 否则输出False}if(b=="parent"){ // 如果查询条件是parentif(parent[c]==a)cout<<"True"<<endl; // 如果c的父亲是a,则输出Trueelse cout<<"False"<<endl; // 否则输出False}if(b=="descendant"){ // 如果查询条件是descendantwhile(parent[a]!=c&&parent[a]!=anc)a=parent[a]; // 从a开始向上查找,直到找到c或者祖先节点if(parent[a]==anc)cout<<"False"<<endl; // 如果找到祖先节点,则输出Falseelse cout<<"True"<<endl; // 否则输出True}} return 0;
}

以上就是排序算法编程题的专项练习,在下一篇文章中我们将介绍树的相关知识点。

这篇关于【数据结构入门精讲 | 第十篇】考研408排序算法专项练习(二)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/531754

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费

hdu 1285(拓扑排序)

题意: 给各个队间的胜负关系,让排名次,名词相同按从小到大排。 解析: 拓扑排序是应用于有向无回路图(Direct Acyclic Graph,简称DAG)上的一种排序方式,对一个有向无回路图进行拓扑排序后,所有的顶点形成一个序列,对所有边(u,v),满足u 在v 的前面。该序列说明了顶点表示的事件或状态发生的整体顺序。比较经典的是在工程活动上,某些工程完成后,另一些工程才能继续,此时