0x06 倍增

目录

例1:求出最大K

题目

题解

代码

例2:Genius ACM

题目

题意

题解

代码

ST算法

题目

题解

代码


例1:求出最大K

题目

  给一个长度为N的数列A,然后进行若干次询问,每次给定一个整数T,求出最大的k,满足sum[k]<=T, 其中sum[i]=a[1]+a[2]+……a[k]。要求算法必须在线。

题解

  用倍增的思想。

  1. 令 p=1, k = 0, sum = 0;
  2. 求出a的前缀和,记在s数组中。如果sum+s[k+p]-s[k]<=T,那么sum+=s[k+p]-s[k], k += p, p*=2 ,否则的话 p /= 2;
  3. 重复第二步,直到k=0为止。

代码

int n,a[maxn],t,s[maxn]; 

int main()
{
	cin>>n>>t;
	for(int i=1;i<=n;i++) {
		read(a[i]);
		s[i] = s[i-1]+a[i];
	}
	int p = 1, k = 0, sum = 0;
	while(p) {
		if(k+p<=n&&sum+s[k+p]-s[k]<=t) {
			sum += s[k+p]-s[k];
			k+=p;
			p*=2;
		}else {
			p /= 2;
		}
	}
	cout<<k<<endl;
	return 0;
}

例2:Genius ACM

题目

  CH0601

题意

  给定一个整数 M,对于任意一个整数集合 S,定义“校验值”如下:

  从集合 S 中取出 M 对数(即 2∗M 个数,不能重复使用集合中的数,如果 S 中的整 数不够 M 对,则取到不能取为止),使得“每对数的差的平方”之和最大,这个最大值 就称为集合 S 的“校验值”。

    现在给定一个长度为 N 的数列 A 以及一个整数 T。我们要把 A 分成若干段,使得 每一段的“校验值”都不超过 T。求最少需要分成几段。

题解

  倍增扩展区间,每个区间先排序,求出校验值,然后依次倍增扩展当前区间。

代码

ll t,n,m,K,a[maxn],tot,b[maxn];

bool calc(ll l,ll k,ll p) { //左区间为l,右区间为r,扩展长度为p,最右端为k+p
	ll res = 0;
	for(ll i=l;i<=k+p;i++)
		b[i-l] = a[i];
	ll len = p+k-l+1;
	sort(b,b+len);
	if(len>=2*m) { //取出m个数
		for(ll i=0;i<m;i++) {
			res += (b[i]-b[len-i-1])*(b[i]-b[len-i-1]);		
		}
	}else {
		ll temp = len/2;	
		for(ll i=0;i<temp;i++) {
			res += (b[i]-b[len-i-1])*(b[i]-b[len-i-1]);
		}
	}
	if(res<=K)	return true;
	return false;
}

int main()
{
	read(t);
	while(t--) {
		tot = 0;
		read(n),read(m),read(K);
		for(ll i=0;i<n;i++) {
			read(a[i]);
		}
		ll l = 0;
		while(l<n) {
			ll k = l, p = 1;
//			cout<<k<<" "<<p<<endl;
			while(p) { //枚举当前区间能扩展到的最大值
				if(calc(l,k,p)&&k+p<n) {
					k += p;
					p *= 2;
				}else {
					p /= 2;
				}
			}
			tot++; //区间数+1
			l = k+1; //
		}
		cout<<tot<<endl;
	}
	return 0;
}

 


ST算法

题目

  求l-r区间内最大的数字。

题解

  用 f[i][j] 表示区间 [i,i+2^{j}-1] 里最大的数, 递推边界显然是f[i][0] = a[i], 首先预处理,利用公式 f[i][j] = max(f[i][j-1], f[i+2^{j-1}][j-1]);

  在查询的时候,求出一个k,使得 2^{k}<r-l+1<2^{k+1} , 这样从l开始的2^{k}个数和以r结尾的2^{k}个数必然覆盖了整个区间,那么[l,r]区间必然被包含了,所有最大值即为所求。

代码

int n,f[N][N],a[maxn],q;

void st_work() {
	for(int i=1;i<=n;i++)
		f[i][0] = a[i];
	int t = log(n)/log(2)+1;
	for(int j=1;j<t;j++) {
		for(int i=1;i<=n-(1<<j)+1;i++)
			f[i][j] = max(f[i][j-1],f[i+ (1<<(j-1))][j-1]);
	}	
}

int query(int l,int r) {
	int k = log(r-l+1)/log(2);
	return max(f[l][k],f[r-(1<<k)+1][k]);	
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
这段代码是关于数字时钟的程序,使用了51单片机。 1. `#include<reg51.h>`:包含了51单片机的头文件。 2. `unsigned char hour = 0; unsigned char min = 0; unsigned char sec = 0;`:定义了时、分、秒三个变量,用于记录当前时间。 3. `unsigned char table[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};`:定义了一个表,用于将数字转换为数码管上显示的对应数字。 4. `unsigned int num=0;`:定义了一个计数器,用于计时一秒。 5. `unsigned dat;`:定义了一个变量,用于存储串口通信中收到的数据。 6. `void delay(unsigned int t);`:声明了一个延时函数。 7. `void main(void)`:主函数开始。 8. `TMOD=0x21;`:设置定时器模式,使用定时器0为模式1,使用定时器1为模式2。 9. `SCON=0x50;`:设置串口工作模式。 10. `PCON=0x00;`:关闭波特率倍增功能。 11. `TH0=0xFC; TL0=0x18;`:设置定时器0的初值,用于实现1秒的定时。 12. `TH1=0xf4; TL1=0xf4;`:设置定时器1的初值,用于串口通信时的波特率控制。 13. `TR1=1; EA=1; ES=1; PS=1; ET0=1; TR0=1;`:开启定时器1、总中断、串口中断、定时器0中断、定时器0。 14. `while(1)`:进入无限循环。 15. `P2=table[sec%10]&0x7f; //秒的个位 P1=0xFE; delay(200); P1=0xFF;`:将秒的个位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第0位置低,使对应数码管亮起来,延时200毫秒后将P1口的第0位置高,消除闪烁。 16. `P2=table[sec/10]; //秒的十位 P1=0xFD; delay(200); P1=0xFF;`:将秒的十位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第1位置低,使对应数码管亮起来,延时200毫秒后将P1口的第1位置高,消除闪烁。 17. `P2=table[min%10]; //分的个位 P1=0xFB; delay(200); P1=0xFF;`:将分的个位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第2位置低,使对应数码管亮起来,延时200毫秒后将P1口的第2位置高,消除闪烁。 18. `P2=table[min/10]; //分的十位 P1=0xF7; delay(200); P1=0xFF;`:将分的十位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第3位置低,使对应数码管亮起来,延时200毫秒后将P1口的第3位置高,消除闪烁。 19. `P2=table[hour%10]; //时的个位 P1=0xEF; delay(200); P1=0xFF;`:将时的个位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第4位置低,使对应数码管亮起来,延时200毫秒后将P1口的第4位置高,消除闪烁。 20. `P2=table[hour/10]; //时的十位 P1=0xDF; delay(200); P1=0xFF;`:将时的十位数转换为数码管上的数字,并输出到P2口上,同时将P1口的第5位置低,使对应数码管亮起来,延时200毫秒后将P1口的第5位置高,消除闪烁。 21. `void delay(unsigned int t) //延时函数 { unsigned int i; for(i=0;i<t;i++); }`:延时函数,用于实现闪烁效果。 22. `void Time() interrupt 1`:定时器0中断处理函数,用于实现1秒的计时和更新时间。 23. `TH0=0xFC; TL0=0x18;`:设置定时器0的初值,用于实现1秒的定时。 24. `num++;`:计数器加1。 25. `if(num==1000)`:计数器达到1000时,说明已经计时1秒。 26. `sec++;`:秒数加1。 27. `if(sec>=60)`:如果秒数超过60,说明已经过了1分钟。 28. `sec=0; min++;`:将秒数清零,并将分钟数加1。 29. `if(min>=60)`:如果分钟数超过60,说明已经过了1小时。 30. `min=0; hour++;`:将分钟数清零,并将小时数加1。 31. `if(hour>=24)`:如果小时数超过24,说明已经过了1天。 32. `hour=0;`:将小时数清零。 33. `void show(void) interrupt 4`:串口中断处理函数,用于接收上位机发送的指令并执行相应操作。 34. `if(RI==1)`:如果收到了数据。 35. `dat=SBUF; RI=0;`:将收到的数据存储在变量dat中,并将接收标志RI清零。 36. `switch(dat)`:根据收到的数据进行相应的操作。 37. `case 1: TR0=!TR0; break;`:收到数字1,将定时器0的运行状态取反,实现暂停和继续计时的功能。 38. `case 2: hour++;if(hour==24) hour=0;break;`:收到数字2,将小时数加1,如果超过24,则将小时数重置为0。 39. `case 3: min++;if(min==60) min=0;break;`:收到数字3,将分钟数加1,如果超过60,则将分钟数重置为0。 40. `case 4: sec++;if(sec==60) sec=0;break;`:收到数字4,将秒数加1,如果超过60,则将秒数重置为0。 41. `case 5: hour=0;min=0;sec=0;P2=table[0];P1=0xc0;TR0=!TR0;break;`:收到数字5,将时间重置为0,并将数码管显示为0,实现清零功能。 42. `default: break;`:如果收到其他数字,则不进行任何操作。 这就是这段代码的逐行解释。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

总想玩世不恭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值