Sol

首先,考虑这个要怎么搞...让总和最大的方法就是选出当前集合中最大的两个数相加放入集合中就可以了,证明非常简单,当前集合的和为x,它的和只会一直往后增加,所以只需要找到最大的两个数的和加入便是最佳答案.知道了这个以后,手动递推一下就是一个斐波拉契数列.

然后斐波拉契数列数列自然可以矩乘,但是矩阵乘法不能解决负数斐波拉契问题.而且,一正一负就不是斐波拉契数列的递推了.递推应该是这个样子.

\(F_2=F_1+F_0,F_3=F_2+F_1=F_1+F_1+F_0...F_n=F_1*(n-1)+F_0\)

所以我们就可以直接 \(O(1)\) 得到需要进行多少次操作使得出现两个正数,原来的 \(F_1\) 就已经是一个正数了,所以操作数为 \(\left \lfloor -F_0/F_1 \right \rfloor +1\) 求和用等差数列求和公式,剩下的直接矩乘.

剩下的问题就是斐波拉契数列求和的问题了,我们知道斐波拉契数列前n项和 \(\sum_{i=0}^{n}F_i=F_{n+2}-1\) 虽然这个前两项并不是1,但是可以得到相似的结论 \(\sum_{i=0}^{n}F_i=F_{n+2}-F_1\) .

在这里我给出证明,证明也很简单,数学归纳法.

证明: \(n=0\) 时显然成立, \(F_0=F_2-F_1\) 移项一下就是递推式

当 \(n>0\) 时,假设有 \(\sum_{i=0}^{n}F_i=F_{n+2}-F_1\)

只需要证明在 \(n+1\) 时同样成立即可.

\(\sum_{i=0}^{n+1}F_i=\sum_{i=0}^{n}F_i +F_{n+1}=F_{n+2}-F_1+F_{n+1}=F_{n+3}-F_1\)

即在 \(n+1\) 处依然成立,证毕.

PS:当然这个也可以构造一个矩阵同时记录前缀和即可.

PS:我代码码风改了改...应该可以还算可阅吧...QAQ

Code

  1. /**************************************************************
  2. Problem: 4547
  3. User: BeiYu
  4. Language: C++
  5. Result: Accepted
  6. Time:324 ms
  7. Memory:2076 kb
  8. ****************************************************************/
  9.  
  10. #include<cstdio>
  11. #include<vector>
  12. #include<algorithm>
  13. #include<iostream>
  14. using namespace std;
  15.  
  16. #define debug(a) cout<<#a<<"="<<a<<endl
  17. typedef long long LL;
  18. typedef vector<LL> Vec;
  19. typedef vector<Vec> Mat;
  20. const LL Mo = 10000007;
  21. const int N = 100005;
  22.  
  23. Mat operator * (const Mat &A,const Mat &B){
  24. Mat C(2,Vec(2));
  25. for(int i=0;i<2;i++) for(int j=0;j<2;j++) for(int k=0;k<2;k++)
  26. C[i][j]=(C[i][j]+A[i][k]*B[k][j])%Mo;
  27. return C;
  28. }
  29. Mat operator ^ (Mat A,LL b){
  30. Mat res(2,Vec(2));
  31. res[0][0]=1,res[0][1]=0,res[1][0]=0,res[1][1]=1;
  32. for(;b;b>>=1,A=A*A){
  33. if(b&1) res=res*A;
  34. }return res;
  35. }
  36.  
  37. LL n,k;LL a[N];
  38. inline LL in(LL x=0,char ch=getchar(),int v=1){
  39. while(ch>'9'||ch<'0') v=ch=='-'?-1:1,ch=getchar();
  40. while(ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
  41. return x*v;
  42. }
  43. int main(){
  44. n=in(),k=in();
  45. for(int i=1;i<=n;i++) a[i]=in();
  46. sort(a+1,a+n+1);
  47. LL tmp,f0=a[n-1],f1=a[n],ans=0;
  48. for(int i=1;i<n-1;i++) ans=(ans+a[i]+Mo)%Mo;
  49. if(f0<0){
  50. tmp=-f0/f1+1;
  51. if(tmp<=k){
  52. k-=tmp,ans=(ans+(f0+f0+f1*(tmp-1))*(tmp)/2%Mo)%Mo;
  53. f0=(f0+f1*tmp%Mo)%Mo;
  54. if(f0>f1) swap(f0,f1);
  55. }
  56. }
  57. Mat F(2,Vec(2));F[0][0]=0,F[0][1]=1,F[1][0]=1,F[1][1]=1;
  58. Mat Fn=F^(k+2);
  59. ans=(ans+f0*Fn[0][1]%Mo+f1*Fn[1][1]%Mo)%Mo;
  60. ans=(ans-f1+Mo)%Mo;
  61. printf("%lld\n",ans);
  62. return 0;
  63. }

  

BZOJ 4547: Hdu5171 小奇的集合的更多相关文章

  1. bzoj4547: Hdu5171 小奇的集合(矩阵乘法)

    4547: Hdu5171 小奇的集合 题目:传送门 题解: 做一波大佬们的坑...ORZ 不得不说,我觉得矩阵很简单啊,就一个3*3的(直接看代码吧) 给个递推柿纸:f[i]=f[i-1]+max1 ...

  2. BZOJ4547 Hdu5171 小奇的集合 【矩阵快速幂优化递推】

    BZOJ4547 Hdu5171 小奇的集合 Description 有一个大小为n的可重集S,小奇每次操作可以加入一个数a+b(a,b均属于S),求k次操作后它可获得的S的和的最大值.(数据保证这个 ...

  3. 【BZOJ4547】Hdu5171 小奇的集合 矩阵乘法

    [BZOJ4547]Hdu5171 小奇的集合 Description 有一个大小为n的可重集S,小奇每次操作可以加入一个数a+b(a,b均属于S),求k次操作后它可获得的S的和的最大值.(数据保证这 ...

  4. BZOJ4547 Hdu5171 小奇的集合

    题意 有一个大小为n的可重集S,小奇每次操作可以加入一个数a+b(a,b均属于S),求k次操作后它可获得的S的和的最大值.(数据保证这个值为非负数) 对于100%的数据,有 n<=10^5,k& ...

  5. 【BZOJ-4547】小奇的集合 矩阵乘法 + 递推

    4547: Hdu5171 小奇的集合 Time Limit: 2 Sec  Memory Limit: 256 MBSubmit: 175  Solved: 85[Submit][Status][D ...

  6. bzoj 4547 小奇的集合

    Description 有一个大小为n的可重集S,小奇每次操作可以加入一个数a+b(a,b均属于S),求k次操作后它可获得的S的和的最大 值.(数据保证这个值为非负数) Input 第一行有两个整数n ...

  7. 【BZOJ 4547】【HDU 5157】小奇的集合

    http://www.lydsy.com/JudgeOnline/problem.php?id=4547 本蒟蒻并不会矩乘求Fibonacci数列前缀和,所以果断分块打表,常数竟然比矩乘要小! PS: ...

  8. bzoj4547 小奇的集合

    当序列中最大和次大都是负数的时候,其相加会是一个更小的负数,因此答案为(Σai)+(m1+m2)*k,如果最大是正数次大是负数,那么一直相加直到两个数都为正数,当最大和次大都是正数时,做一下矩阵乘法即 ...

  9. [HDU517] 小奇的集合

    题目链接 显然有贪心每次选择最大的两个数来做. 于是暴力地把最大的两个数调整到非负(暴力次数不超过1e5),接下来使用矩阵乘法即可. \[ \begin{pmatrix} B'\\S'\\T' \en ...

随机推荐

  1. UVa 11988 Broken Keyboard(链表->数组实现)

    /*数组形式描述链表:链表不一定要用指针. 题目链接:UVa 11988 Broken Keyboard 题目大意: 小明没有开屏幕输入一个字符串,电脑键盘出现了问题会不定时的录入 home end ...

  2. php----显示中文乱码的问题

    条件: 在显示页面设置页面编码格式为<?php header('Content-Type: text/html; charset=utf-8');?>: 在写入数据库时设置:mysql_q ...

  3. C++ 以费波纳茨数列为权重的加权均值计算方法 wMA

    #pragma once #include <iostream> using namespace std; template <typename T> double *wMA( ...

  4. C++ 函数返回数组指针的问题

    提醒一下:int *func(){int a[10] = {5};return a;}是非法的,因为 a 是局部变量,被申请在系统栈中,局部变量在函数返回后会被销毁,于是你返回的指针指向的是一段已经被 ...

  5. Unity Particle System Sorting Order

    http://answers.unity3d.com/questions/577288/particle-system-rendering-behind-sprites.html

  6. CentOS_PHP_NGINX_FastCGI

    yum安装nginx,它会默认作为一个服务加到系统中,启动nginx: service nginx start/nginx -s start 他有4个参数(start|stop|restart|rel ...

  7. Python开发【第二篇】:初识Python

    Python开发[第二篇]:初识Python   Python简介 Python前世今生 python的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏 ...

  8. System.exit(0)和System.exit(1)区别

    System.exit(0)是将你的整个虚拟机里的内容都停掉了 ,而dispose()只是关闭这个窗口,但是并没有停止整个application exit() .无论如何,内存都释放了!也就是说连JV ...

  9. scala入门教程:scala中的面向对象定义类,构造函数,继承

    我们知道scala中一切皆为对象,函数也是对象,数字也是对象,它是一个比java还要面向对象的语言. 定义scala的简单类 class Point (val x:Int, val y:Int) 上面 ...

  10. 关于JS的几点TIPS

    作为前端基本工作每天都会用到JS...但是我们对JS真的都了解吗,或者说有什么tips是我们不知道的呢.. So..此文关于JS的几点tips..... 一:定时器(可传多个参数) 首先是一个一般的定 ...