解题思路:
借鉴了大佬的思想,在此表示崇敬和感谢。
https://blog.dotcpp.com/a/95964
站在大佬的肩膀上,进一步发现了更容易编程的规律:每个x对应的数组由三个序列构成。
整体思想和注意事项:
1、特殊情况
即:以n开头的递减序列,其对应的x为: n*(n-1)//2
只要遇到 x,反推得到n,恰好满足:n*(n-1)//2==x,那么序列就是:n n-1 n-2 n-3....1
2、其他情况
若遇到x,不能恰好满足特殊情况,就先找到一个满足 n*(n-1)//2<x 的最大的n。
下面是最笨的穷举法:
def f2(n): # return n*(n-1)//2
return n*(n-1)//2
def f(x):# n*(n-1)<=2*x,返回最大的n
for n in range(2,100000):
y=f2(n)
if y==x:
return n
elif y>x:
return n-1
比如:
x=22,得到n=7,那么只需要在序列 7654321 前面加上 a=x-n*(n-1)//2+1=2 的数字即可得到一个解:27654321
依次类推。
但这个解不是字典序最小的。还需要想办法调整。
观察下面表中规律。【此数据来自大佬页面】
当在长度为n的递减序列前面添加a后,对应的最终序列有下面规律:
1、应为n+1长度
2、最终序列可看成是三递减序列的组合:数字a、a-1递减序列和b递减序列。且a+b==n+1。
3、三个序列的组合规律是:先把a-1序列和b递减序列组合,然后降序排序,然后把a插入到头部。
计算得到n
y=x-n*(n-1)//2
a=x-y+1
b=n+1-a
L=[]
for i in range(1,a):
L.append(i)
for j in range(1,b+1):
L.append(j)
L.sort(reverse=True)
L.insert(0,a)
参考代码:
import sys,math def f2(n): # return n*(n-1)//2 return n*(n-1)//2 def f(x):# n*(n-1)x: return n-1 x=int(input()) n=f(x) y=f2(n) #print("1:",n) L=[] if y==x:#满足特殊情况的序列 a=0 b=n for i in range(b,0,-1): L.append(i) else:#一般序列 #最终的序列由两个序列组合而成, #a开始的递减序列和b开始的递减序列,a+b==n+1,a是要添加的数字 #如果a>=b,直接合并,降序即可 #如果a<b,则是a-1递减序列和b递减序列合并、降序,再在结果前面添加a #可统一为:a-1递减序列和b递减序列合并、降序,再在结果前面添加a a=x-y+1 #a是最高位要添加的数 b=n+1-a #b是另一个序列的开始值 #print(n,a,b) for i in range(1,a): L.append(i) for j in range(1,b+1): L.append(j) L.sort(reverse=True) L.insert(0,a) print(len(L)) for x in L: print(x,end=" ")
0.0分
2 人评分
C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:
一点编程也不会写的:零基础C语言学练课程
解决困扰你多年的C语言疑难杂症特性的C语言进阶课程
从零到写出一个爬虫的Python编程课程
只会语法写不出代码?手把手带你写100个编程真题的编程百练课程
信息学奥赛或C++选手的 必学C++课程
蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程
手把手讲解近五年真题的蓝桥杯辅导课程
发表评论 取消回复