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++;
}
}
}
二分代码有时候较为复杂,但要记住固定的模板,套进交换即可
目的主要还是空间换时间,避免超时