2021-5-5 信息总结 (分治与二分)

14 篇文章 0 订阅
12 篇文章 0 订阅

1.分治
顾名思义,即“分而治之”,将一个或多个大问题拆解成多个小问题,并
通过一定的方法将其连接起来,寻找特殊解加入

例如比赛安排这道题
如图所示
看到图片,我们可以找寻规律
将其分为四个象限,以此类推
在这里插入图片描述
每一个大区域,都可以理解为左上角,右上角,左下角,右下角
找寻对应关系即可
最后用一个大的for循环for(int i=1;i<=n;i++)
来排列次数,内部镶嵌小循环,二维数组即可
还有一个易错点
每个字符占三个字符
所以我们可以使用setw函数

这就是典型的分治思想

2.二分
个人认为二分是分治的一个分支
将一个问题,每次求解一半,最后交融

具体体现在逆序对排列这道题

我们通过在主要函数代码中加注释进行理解

void work(int l,int r) //即left,right,左右边界值
{
	int mid,tmp,i,j;
	
	/**这里是典型的二分**
	其中mid即左右分完的中间值*/
	
	if(r>l+1) //如果右边大于左边,即边界值
	{
		mid=(l+r)/2;
		//寻找中间值
		
		work(l,mid-1);
		work(mid,r);
		//得到左右的数,递归思想
		
		tmp=l;//记录左边的值
		for(i=l,j=mid; i<=mid-1&&j<=r;) //这里循环有两个条件,要开始左右交换比了
		{
			if(a[i]>a[j])
			{
				c[tmp++]=a[j++];//交换
//				tmp; j;
				cnt+=mid-i;//一次逆序对出现并记录,注意理解mid-i
			}
			
			else c[tmp++]=a[i++];//交换
		}
		//全部赋值回来
		if(j<=r) for(;j<=r;j++) c[tmp++]=a[j];
		else for(;i<=mid-1;i++) c[tmp++]=a[i];
		for(i=l;i<=r;i++) a[i]=c[i];
	}
	//另一种情况
	else
	{
	//一种只有一的特殊情况
	
		if(l+1==r)//这个其实应该可以省略
		if(a[l]>a[r])
		{
			swap(a[l],a[r]);
			cnt++;
		}
	}
}

二分代码有时候较为复杂,但要记住固定的模板,套进交换即可

目的主要还是空间换时间,避免超时

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值