题目描述
小明这些天一直在思考这样一个奇怪而有趣的问题:
在 1∼N 的某个排列中有多少个连号区间呢?
这里所说的连号区间的定义是:
如果区间 [L,R] 里的所有元素(即此排列的第 L 个到第 R 个元素)递增排序后能得到一个长度为 R−L+1 的“连续”数列,则称这个区间连号区间。
当 N 很小的时候,小明可以很快地算出答案,但是当 N 变大的时候,问题就不是那么简单了,现在小明需要你的帮助。
输入格式
第一行是一个正整数 N,表示排列的规模。
第二行是 N 个不同的数字 Pi,表示这 N 个数字的某一排列。
输出格式
输出一个整数,表示不同连号区间的数目。
数据范围
1≤N≤10000,
1≤Pi≤N
输入样例1:
4
3 2 4 1
输出样例1:
7
输入样例2:
5
3 4 2 5 1
输出样例2:
9
样例解释
第一个用例中,有 7 个连号区间分别是:[1,1],[1,2],[1,3],[1,4],[2,2],[3,3],[4,4]
第二个用例中,有 9 个连号区间分别是: [1,1],[1,2],[1,3],[1,4],[1,5],[2,2],[3,3],[4,4],[5,5]
解题思路
暴力的话:直接枚举所有的可能解,用两层for循环,再将得到的每组解sort一下,然后再用一层for循环判断这组解是否连号,这样的话,时间复杂度是(N^3) 的,N=10000的话,会超时。
所以优化:由于N是10000,所以时间复杂度应该控制在N^2或者NlogN,那就要减少一层循环,判断是否连号有没有可能优化成O(1)的?由题目可知,给的n个数是1~n的排列,所以一定没有重复的数,那如果一个序列连号,一定存在max-min==r-l
,max是这个序列的最大值,min是这个序列的最小值,r是这个序列右端点的下标,l是这个序列左端点的下标
比如对应样例1 的连号区间[1,4],[3 2 4 1]对应左端点下标是l=1,右端点下标是r=4,max是4,min是1 ——> max-min=r-l
代码实现+注释 C++
#include
#include
#include
using namespace std;
const int N=10010,INF=1e5;
int a[N];
int n,res=0;
int main()
{
cin>>n;
for(int i=1;i<=n;i++) cin>>a[i];
for(int i=1;i<=n;i++){//枚举 l
int maxv=-INF,minv=INF;//先将min和max初始化成不可能的最大值和最小值
for(int j=i;j<=n;j++){//枚举 r
maxv=max(maxv,a[j]);
minv=min(minv,a[j]);
if((maxv-minv)==(j-i))//如果这个序列的最大值减去最小值等于这个序列的长度,那么一定连号
res++;
}
}
cout<<res<<endl;
}
题目描述
给定三个整数数组
A=[A1,A2,…AN],
B=[B1,B2,…BN],
C=[C1,C2,…CN],
请你统计有多少个三元组 (i,j,k) 满足:
1≤i,j,k≤N
Ai
第一行包含一个整数 N。
第二行包含 N 个整数 A1,A2,…AN。
第三行包含 N 个整数 B1,B2,…BN。
第四行包含 N 个整数 C1,C2,…CN。
输出格式
一个整数表示答案。
数据范围
1≤N≤105,
0≤Ai,Bi,Ci≤105
输入样例:
3
1 1 1
2 2 2
3 3 3
输出样例:
27
解题思路
暴力枚举
:如果直接枚举,那么第一个for循环枚举A,第二个for循环枚举B,第三个for循环枚举C,时间复杂度是O(N^3),N=105,数据太大的话可能是会超时的
优化
:根据N的最大值105,我们可以判断该算法最大应该控制在O(N^2logN),那就要考虑怎么把三重循环优化到两层或者一层,所以考虑先枚举哪一个数比较好,如果先枚举A,那么C就要收到B的限制,这样还是要枚举三层,枚举C也是一样,所以我们考虑先枚举B,如果Bi确定了,那么只要再A中所有小于Bi数的个数,在C中找实验大于Bi的数的个数,两者相乘,所有枚举的结果相加就是答案
1.前缀和:
时间复杂度O(1),整个题的时间复杂度就是O(N^2),用cnt[]数组记录a[]中每个数出现的次数,再对cnt[]求前缀和s[],则s[i]就表示a中小于等于i的元素个数,那么a中小于Bi的数就是s[Bi-1]了,同理可以求大于Bi的
2.二分
时间复杂度O(logN),整个题的时间复杂度就是O(NlogN),将数组a和c排序,枚举Bi的时候,用二分分别找到在a中第一个大于等于Bi的数的小标,在c中找到最后一个小于等于Bi的数的下标,二者相乘即可
代码实现+注释 C++
前缀和
#include
#include
#include
using namespace std;
const int N=100010;
int a[N],b[N],c[N];
int s[N],as[N],cs[N],cnt[N];//as[i] a中所有小于 b[i] 的元素的个数
//cs[i] c中所有大于 b[i] 的元素的个数
int main()
{
int n;
cin>>n;
for(int i=1;i<=n;i++) cin>>a[i],a[i]++;//这里是为了后面处理前缀和从1开始,不用判断下标越界,因为a[i]取值可能是0
for(int i=1;i<=n;i++) cin>>b[i],b[i]++;
for(int i=1;i<=n;i++) cin>>c[i],c[i]++;
//求as
for(int i=1;i<=n;i++) cnt[a[i]]++;
for(int i=1;i<N;i++) s[i]=s[i-1]+cnt[i];//s[i]就表示所有小于等于i的元素个数
for(int i=1;i<=n;i++) as[i]=s[b[i]-1];//所有小于b[i]的元素个数就是所有小于等于b[i-1]的元素个数
memset(cnt,0,sizeof(cnt));
memset(s,0,sizeof(s));
//求cs
for(int i=1;i<=n;i++) cnt[c[i]]++;
for(int i=1;i<N;i++) s[i]=s[i-1]+cnt[i];
for(int i=1;i<=n;i++) cs[i]=s[N-1]-s[b[i]];//所有大于b[i]的元素个数 就是集合c的元素个数减去小于等于b[i]的元素个数
long long res=0;
for(int i=1;i<=n;i++) res+=(long long)cs[i]*as[i];//将所有组合累加,但可能会爆int,所有强转
cout<<res<<endl;
return 0;
}
二分
#include
#include
#include
using namespace std;
const int N=100010;
int a[N],b[N],c[N];
int n;
int main()
{
cin>>n;
for(int i=0;i<n;i++) cin>>a[i];
for(int i=0;i<n;i++) cin>>b[i];
for(int i=0;i<n;i++) cin>>c[i];
sort(a,a+n);
sort(b,b+n);
sort(c,c+n);
long long res=0;
for(int i=0;i<n;i++){
long long x = (lower_bound(a , a + n, b[i])- a) ; //在数组a中找比b[i]小的数
long long y = n - (upper_bound(c , c+ n, b[i]) - c); //在数组c中找比b[i]大的数
res+=(long long) x*y;
}
cout<<res<<endl;
return 0;
}
题目描述
小明正在整理一批历史文献。这些历史文献中出现了很多日期。
小明知道这些日期都在1960年1月1日至2059年12月31日。
令小明头疼的是,这些日期采用的格式非常不统一,有采用年/月/日的,有采用月/日/年的,还有采用日/月/年的。
更加麻烦的是,年份也都省略了前两位,使得文献上的一个日期,存在很多可能的日期与其对应。
比如02/03/04,可能是2002年03月04日、2004年02月03日或2004年03月02日。
给出一个文献上的日期,你能帮助小明判断有哪些可能的日期对其对应吗?
输入格式
一个日期,格式是”AA/BB/CC”。
即每个’/’隔开的部分由两个 0-9 之间的数字(不一定相同)组成。
输出格式
输出若干个不相同的日期,每个日期一行,格式是”yyyy-MM-dd”。
多个日期按从早到晚排列。
数据范围
0≤A,B,C≤9
输入样例:
02/03/04
输出样例:
2002-03-04
2004-02-03
2004-03-02
解题思路
枚举:如果将日期看做一个八位数,那么日期的取值范围就是[19600101,20591231],我们可以枚举这个区间的所有数,先判断这个数是不是日期格式,然后判断能不能用所给的日期表示出来,时间复杂度大概在10^7。
代码实现+注释 C++
#include
#include
#include
#include
using namespace std;
int days[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};
char str[9];
bool check_month(int y,int m,int d){//月份要满足1~12 日要满足days 且是闰年时要满足二月29天
if(y<1960||y>2059) return false;
if(m<=0||m>12) return false;
if(m!=2&&days[m]<d||d<=0) return false;
if(m==2){
if(d>days[m]+(y%4==0&&y%100!=0||y%400==0)) return false;
}
return true;
}
int main()
{
int a,b,c;
scanf("%d/%d/%d",&a,&b,&c);
for(int i=19600101;i<=20591231;i++){//吧日期当做一个八位数,枚举所有在这两个数之间的数
int year=i/10000,month=i%10000/100,day=i%100;
if(check_month(year,month,day)){//检查这个数是否是合法日期
//年/月/日 月/日/年 日/月/年
if(year%100==a&&month==b&&day==c||month==a&&day==b&&year%100==c||day==a&&month==b&&year%100==c)//判断这个日期能否用所给的数表示
printf("%d-%02d-%02d\n",year,month,day);
}
}
return 0;
}
题目描述
在日常生活中,通过年、月、日这三个要素可以表示出一个唯一确定的日期。
牛牛习惯用 8 位数字表示一个日期,其中,前 4 位代表年份,接下来 2 位代表月份,最后 2 位代表日期。
显然:一个日期只有一种表示方法,而两个不同的日期的表示方法不会相同。
牛牛认为,一个日期是回文的,当且仅当表示这个日期的 8 位数字是回文的。
现在,牛牛想知道:在他指定的两个日期之间(包含这两个日期本身),有多少个真实存在的日期是回文的。
一个 8 位数字是回文的,当且仅当对于所有的 i(1≤i≤8) 从左向右数的第 i 个数字和第 9−i 个数字(即从右向左数的第 i 个数字)是相同的。
例如:
对于 2016 年 11 月 19 日,用 8 位数字 20161119 表示,它不是回文的。
对于 2010 年 1 月 2 日,用 8 位数字 20100102 表示,它是回文的。
对于 2010 年 10 月 2 日,用 8 位数字 20101002 表示,它不是回文的。
输入格式
输入包括两行,每行包括一个 8 位数字。
第一行表示牛牛指定的起始日期 date1,第二行表示牛牛指定的终止日期 date2。保证 date1 和 date2 都是真实存在的日期,且年份部分一定为 4 位数字,且首位数字不为 0。
保证 date1 一定不晚于 date2。
输出格式
输出共一行,包含一个整数,表示在 date1 和 date2 之间,有多少个日期是回文的。
输入样例:
20110101
20111231
输出样例:
1
解题思路
如果要枚举给出的两个日期之间的日期来判断是否回文,我们就要手写一个日历,为了不手写日历,我们可以枚举回文数,再判断该回文数是否在所给日期之间,再判断日期是否合法。枚举回文数只用枚举年份就行,也就是前四位,那么后四位由回文数的性质可知是前四位反过来。年份一定为四位且年份首位不为0,所以可以从1枚举到9999
代码实现+注释 C++
#include
#include
#include
using namespace std;
int date1,date2;
int days[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};
bool check_date(int date){
//date/100%100 得到的是月份
//date%100 得到的是天数
//date/10000 得到的是年份
if(date/100%100>12||date/100%100<=0) return false;//判断月份是否合法
if(date/100%100!=2&&days[date/100%100]<date%100) return false;//判断日期是否合法,二月除外
if(date/100%100==2){
int x=date/10000;
if(date%100>days[2]+(x%100!=0&&x%4==0||x%400==0)) return false;//判断二月日期是否合法,如果是闰年 最大是28+1天
}
return true;
}
int main()
{
cin>>date1>>date2;
int res=0;
for(int i=0;i<=9999;i++){
int date=i,x=i;
for(int j=0;j<4;j++) date=date*10+x%10,x/=10;//给出回文串
if(date>=date1&&date<=date2){//判断年份是否在给的区间之内
if(check_date(date))//判断年份是否合法
res++;
}
}
cout<<res<<endl;
return 0;
}
题目描述
小明对数位中含有 2、0、1、9 的数字很感兴趣(不包括前导 0),在 1 到 40 中这样的数包括 1、2、9、10 至 32、39 和 40,共 28 个,他们的和是 574。
请问,在 1 到 n 中,所有这样的数的和是多少?
输入格式
共一行,包含一个整数 n。
输出格式
共一行,包含一个整数,表示满足条件的数的和。
数据范围
1≤n≤10000
输入样例:
40
输出样例:
574
解题思路
如果按照题目描述直接模拟,第一层for循环i从1~n,然后对于i,将每一位拿出来判断是不是包含2、0、1、9,时间复杂度可以看成O(N^2),但其实不到,跟你N的最大值10000,所以是不会超时,可直接做
代码实现+注释 C++
#include
#include
using namespace std;
int n,res;
int main()
{
cin>>n;
for(int i=1;i<=n;i++){//判断从1~n的所有数
int x=i,y=i;
while(x){//这个while循环是用来取出每一位的
y=x%10;
if(y==0||y==1||y==2||y==9) {//只要取出来的数字包含0、1、2、9任意一个数字可以
res+=i;
break;//记得退出,因为这个数如果满足条件就不用继续判断其它位了,如果继续判断,很可能会重复累加
}
x/=10;
}
}
cout<<res<<endl;
return 0;
}
题目描述
某涉密单位下发了某种票据,并要在年终全部收回。
每张票据有唯一的ID号。
全年所有票据的ID号是连续的,但ID的开始数码是随机选定的。
因为工作人员疏忽,在录入ID号的时候发生了一处错误,造成了某个ID断号,另外一个ID重号。
你的任务是通过编程,找出断号的ID和重号的ID。
假设断号不可能发生在最大和最小号。
输入格式
第一行包含整数 N,表示后面共有 N 行数据。
接下来 N 行,每行包含空格分开的若干个(不大于100个)正整数(不大于100000),每个整数代表一个ID号。
输出格式
要求程序输出1行,含两个整数 m,n,用空格分隔。
其中,m表示断号ID,n表示重号ID。
数据范围
1≤N≤100
输入样例:
2
5 6 8 11 9
10 12 9
输出样例:
7 9
解题思路
模拟:可以先将所给的ID号排序,然后用一个for循环来判断,如果a[i]==a[i-1],说明a[i] 这个数重号ID,n=a[i]。如果a[i]-a[i-1]==2,说明a[i]-1 这个ID是短号ID m=a[i]-1
代码实现+注释 C++
#include
#include
#include
#include
using namespace std;
int a[100010];
int main()
{
int N,x,cnt=0;
int n,m;
cin>>N;
string line;
getline(cin,line);//读取第一行的回车
while(N--){
getline(cin,line);
stringstream ssin(line);//以空格为界限将字符串分开
while(ssin>>a[cnt]) cnt++;
}
sort(a,a+cnt);
for(int i=1;i<cnt;i++){
if(a[i]==a[i-1]) n=a[i];
if(a[i]-a[i-1]==2) m=a[i]-1;
}
cout<<m<<" "<<n<<endl;
return 0;
}
题目描述
X星球居民小区的楼房全是一样的,并且按矩阵样式排列。
其楼房的编号为 1,2,3…
当排满一行时,从下一行相邻的楼往反方向排号。
比如:当小区排号宽度为 6 时,开始情形如下:
1 2 3 4 5 6
12 11 10 9 8 7
13 14 15 …
我们的问题是:已知了两个楼号 m 和 n,需要求出它们之间的最短移动距离(不能斜线方向移动)。
输入格式
输入共一行,包含三个整数 w,m,n,w 为排号宽度,m,n 为待计算的楼号。
输出格式
输出一个整数,表示 m,n 两楼间最短移动距离。
数据范围
1≤w,m,n≤10000,
输入样例:
6 8 2
输出样例:
4
解题思路
由题意可知,两个数之间的移动距离就等于两个数在矩阵中的横坐标之差的绝对值加上纵坐标绝对值之差的绝对值,所以对于任意给出的两点算出在矩阵中的坐标即可。先将给出的m和n都减一 行号:n/w m/w,列号:n%w m%w 对于行号为奇数的 它的列号应该翻转 因为奇数二进制最后一位是1,所以用行号&1就可判断是不是奇数,翻转奇数行 :列号 w-1-n%w
代码实现+注释 C++
#include
#include
using namespace std;
int w,m,n;
int main()
{
cin>>w>>m>>n;
n--; m--;
int x1=m/w,x2=n/w;//计算行号
int y1=m%w,y2=n%w;//计算列号
if(x1&1) y1=w-1-y1;//翻转奇数行
if(x2&1) y2=w-1-y2;
cout<<abs(x1-x2)+abs(y1-y2)<<endl;//输出曼哈顿距离
return 0;
}
题目描述
小 h 前往美国参加了蓝桥杯国际赛。
小 h 的女朋友发现小 h 上午十点出发,上午十二点到达美国,于是感叹到“现在飞机飞得真快,两小时就能到美国了”。
小 h 对超音速飞行感到十分恐惧。
仔细观察后发现飞机的起降时间都是当地时间。
由于北京和美国东部有 12 小时时差,故飞机总共需要 14 小时的飞行时间。
不久后小 h 的女朋友去中东交换。
小 h 并不知道中东与北京的时差。
但是小 h 得到了女朋友来回航班的起降时间。
小 h 想知道女朋友的航班飞行时间是多少。
对于一个可能跨时区的航班,给定来回程的起降时间。
假设飞机来回飞行时间相同,求飞机的飞行时间。
输入格式
一个输入包含多组数据。
输入第一行为一个正整数 T,表示输入数据组数。
每组数据包含两行,第一行为去程的起降时间,第二行为回程的起降时间。
起降时间的格式如下:
h1:m1:s1 h2:m2:s2
h1:m1:s1 h3:m3:s3 (+1)
h1:m1:s1 h4:m4:s4 (+2)
第一种格式表示该航班在当地时间h1时m1分s1秒起飞,在当地时间当日 h2时m2分s2秒降落。
第二种格式表示该航班在当地时间h1时m1分s1秒起飞,在当地时间次日 h2时m2分s2秒降落。
第三种格式表示该航班在当地时间h1时m1分s1秒起飞,在当地时间第三日 h2时m2分s2秒降落。
输出格式
对于每一组数据输出一行一个时间hh:mm:ss,表示飞行时间为hh小时mm分ss秒。
注意,当时间为一位数时,要补齐前导零,如三小时四分五秒应写为03:04:05。
数据范围
保证输入时间合法(0≤h≤23,0≤m,s≤59),飞行时间不超过24小时。
输入样例:
3
17:48:19 21:57:24
11:05:18 15:14:23
17:21:07 00:31:46 (+1)
23:02:41 16:13:20 (+1)
10:19:19 20:41:24
22:19:04 16:41:09 (+1)
输出样例:
04:09:05
12:10:39
14:22:05
解题思路
因为两次所花费的时间是相同的,所有两次航班飞行时间/2就是一次飞行时间,两次花费总时间=去的时间-时差+回的时间+时差=去的时间+回的时间,故飞行时间=去的时间+回的时间
代码实现+注释 C++
#include
#include
#include
#include
using namespace std;
int get_second(int h,int m,int s){//由h:m:s得到这一时刻距离 00:00:00有多少秒
return h*3600+m*60+s;
}
int get_time(){//计算一次飞行时间,将时间转化成秒
string line;
getline(cin,line);
if(line.back()!=')') line+="(+0)";//将所有字符串统一,方便下面处理
int h1,m1,s1,h2,m2,s2,d;
sscanf(line.c_str(),"%d:%d:%d %d:%d:%d (+%d)",&h1,&m1,&s1,&h2,&m2,&s2,&d);
return get_second(h2,m2,s2)-get_second(h1,m1,s1)+24*d*3600;
}
int main()
{
int t;
cin>>t;
string line;
getline(cin,line);
while(t--){
int time=(get_time()+get_time())/2;
int h=time/3600,m=time%3600/60,s=time%60;//将得到的秒转换成小时 分钟 秒
printf("%02d:%02d:%02d\n",h,m,s);
}
}
题目描述
“饱了么”外卖系统中维护着 N 家外卖店,编号 1∼N。
每家外卖店都有一个优先级,初始时 (0 时刻) 优先级都为 0。
每经过 1 个时间单位,如果外卖店没有订单,则优先级会减少 1,最低减到 0;而如果外卖店有订单,则优先级不减反加,每有一单优先级加 2。
如果某家外卖店某时刻优先级大于 5,则会被系统加入优先缓存中;如果优先级小于等于 3,则会被清除出优先缓存。
给定 T 时刻以内的 M 条订单信息,请你计算 T 时刻时有多少外卖店在优先缓存中。
输入格式
第一行包含 3 个整数 N,M,T。
以下 M 行每行包含两个整数 ts 和 id,表示 ts 时刻编号 id 的外卖店收到一个订单。
输出格式
输出一个整数代表答案。
数据范围
1≤N,M,T≤105,
1≤ts≤T,
1≤id≤N
输入样例:
2 6 6
1 1
5 2
3 1
6 2
2 1
6 2
输出样例:
1
样例解释
6 时刻时,1 号店优先级降到 3,被移除出优先缓存;2 号店优先级升到 6,加入优先缓存。
所以是有 1 家店 (2 号) 在优先缓存中。
解题思路
如果直接模拟,将每一个时刻的所有没有订单的店铺优先级都减一,这样时间复杂度是O(N^2)的,会超时,所以要优化,由于不是每个时刻都有订单,而且可能一个时刻有多个订单,所以可以一次处理一批订单,而且每一次只处理一个店铺,在
t
时刻,如果某店铺有订单
,我们记录有多少订单,上一次有订单的时刻是多少,那么没有订单的总时间=这个时刻 t -上一次有订单的时刻-1
,那么在t 时刻之前店铺优先级=原来的优先级-没有订单的总时间
,而且,在这个时间内,要判断优先级是否小于0,是否要从缓存拿出等,然后在 t 时刻,店铺的优先级=t时刻之前的优先级+订单数*2
,最后,因为对于某些店铺最后一个有订单的时刻可能在T之前,所以为了计算到T时刻每一个店铺的的优先级都应该-=T-该店铺上一次有订单的时刻
,然后判断是否有店铺从优先缓存拿出。
代码实现+注释 C++
#include
#include
#include
#include
using namespace std;
const int N=100010;
int last[N],score[N];//score[i]表示id为i的店铺的优先级 last[i]表示id为i的店铺上一次有订单的时刻
int st[N];//为1表示在优先缓存里
pair<int,int> order[N];//订单时刻+id
int main()
{
int n,m,T;
cin>>n>>m>>T;
for(int i=0;i<m;i++) cin>>order[i].first>>order[i].second;
sort(order,order+m);//将所有订单按照时间排序
for(int i=0;i<m;){//枚举所有订单
int j=i;
while(j<m&&order[i]==order[j]) j++;//每次处理一批,因为在同一个时刻,一个店铺可能有多个订单
int t=order[i].first,id=order[i].second,cnt=j-i;//这个订单的时刻,订单的id,订单的数量
i=j;
score[id]-=t-last[id]-1;//此id是店铺每个没有订单的时刻优先级都减去1,所以现在有订单的时刻减去上一次有订单的时刻得到的就是应该减去的优先级
if(score[id]<0) score[id]=0;
if(score[id]>=3) st[id]=0;
score[id]+=cnt*2;//这个id的店铺在这个时刻的订单数是cnt,每个订单优先级加2
if(score[id]>5) st[id]=1;
last[id]=t;//要记录这次订单的时刻,为下一次做铺垫
}
for(int i=1;i<=n;i++){//如果有店铺最后一次没订单的时刻在T之前,那在T之前的时刻优先级都在减
if(last[i]<T){
score[i]-=T-last[i];
if(score[i]<=3) st[i]=0;
}
}
int res=0;
for(int i=1;i<=n;i++) res+=st[i];
cout<<res<<endl;
return 0;
}
题目描述
给定你一个长度为 n 的整数数列。
请你使用归并排序对这个数列按照从小到大进行排序。
并将排好序的数列按顺序输出。
输入格式
输入共两行,第一行包含整数 n。
第二行包含 n 个整数(所有整数均在 1∼109 范围内),表示整个数列。
输出格式
输出共一行,包含 n 个整数,表示排好序的数列。
数据范围
1≤n≤100000
输入样例:
5
3 1 2 4 5
输出样例:
1 2 3 4 5
解题思路
归并排序
(Merge Sort)是建立在归并操作上的一种有效,稳定的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并
代码实现+注释 C++
#include
using namespace std;
const int N=100010;
int a[N],tmp[N];
void merge_sort(int a[],int l,int r){
if(l>=r) return;
int mid=l+r>>1;//确定分界点
merge_sort(a,l,mid);//归并排序左边
merge_sort(a,mid+1,r);//归并排序右边
int k=0,i=l,j=mid+1;
while(i<=mid&&j<=r){//将左边和右边归并到tmp数组 成为整体有序
if(a[i]<a[j]) tmp[k++]=a[i++];
else tmp[k++]=a[j++];
}
while(i<=mid) tmp[k++]=a[i++];//以防左边或者右边没有全部放入tmp数组
while(j<=r) tmp[k++]=a[j++];
for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];//再将有序的数组拷贝回原数组
}
int main()
{
int n;
cin>>n;
for(int i=0;i<n;i++) cin>>a[i];
merge_sort(a,0,n-1);
for(int i=0;i<n;i++) cout<<a[i]<<" ";
return 0;
}
题目描述
给定一个长度为 n 的整数数列,请你计算数列中的逆序对的数量。
逆序对的定义如下:对于数列的第 i 个和第 j 个元素,如果满足 i
输入格式
第一行包含整数 n,表示数列的长度。
第二行包含 n 个整数,表示整个数列。
输出格式
输出一个整数,表示逆序对的个数。
数据范围
1≤n≤100000,
数列中的元素的取值范围 [1,109]。
输入样例:
6
2 3 4 5 6 1
输出样例:
5
解题思路
用归并排序来求逆序对数量:归并排序的时候,在第三步将分界点左右两边合并到一个数组的时候,会判断左右两边元素的大小,因为 i 一定小于 j ,且左右两边都是有序的了,如果a[i]>a[j],则a[i]~mid的元素都大于a[j],那么逆序对的数量就是 mid-i+1,对应每一次归并,只要满足a[i]>a[j],我们就可以求元素a[i]的逆序对数量,最后累加,就是整个序列的逆序对数量
代码实现+注释 C++
#include
using namespace std;
typedef long long ll;
const int N=100010;
int a[N],tmp[N];
ll ans=0;
ll msort(int l,int r){
if(l>=r) return 0;
int mid=l+r>>1;
msort(l,mid);
msort(mid+1,r);
int i=l,j=mid+1,k=0;
while(i<=mid&&j<=r){
if(a[i]<=a[j]) tmp[k++]=a[i++];
else tmp[k++]=a[j++],ans+=mid-i+1;//满足a[i]>a[j] 则a[i]~a[mid]的所有元素都大于a[j]
}
while(i<=mid) tmp[k++]=a[i++];
while(j<=r) tmp[k++]=a[j++];
for(int i=l,j=0;i<=r;i++,j++) a[i]=tmp[j];
return ans;
}
int main()
{
int n;
cin>>n;
for(int i=0;i<n;i++) cin>>a[i];
cout<<msort(0,n-1)<<endl;
}
对于枚举,每一次都先考虑暴力解法是什么,然后算一下时间复杂度是否能AC,如果超时就看看怎么优化,可能是优化一重循环或者优化将O(N)变为O(logN),这就考虑二分等算法,有时候不一定是枚举解,也可能是枚举所有的方案,看看是否满足所需的条件
对于模拟,每次直接按照题目描述来解题就可,但是,有时候可能要转化一下求解的问题,比如航班时间这个题,超时的话也是要优化,但这种优化很多可能也是转化计算方式,比如外卖店优先级,将处理一个订单转化为处理一批订单。
对于排序,学数据结构的时候,老师说学排序不是真的让我们手写排序,而是因为排序里面有很多的技巧值得我们学习,并且可用到多种算法里,很多题也都是基于排序或者排序的变形,所以明白怎么排序的很重要,模板最好背过
今日刷题结束,欢迎留言交流讨论,多看,多学,多练,upupup!!