博客园 首页 私信博主 显示目录 隐藏目录 管理 动画

Codeforces.520B.Two Buttons(正难则反)

题目链接

\(Description\)

给定两个数\(n,m\),每次可以使\(n\)减一或使\(n\)乘2。求最少需要多少次可以使\(n\)等于\(m\)

\(Solution\)

暴力连边BFS或者DP都行,都是O(n)的。有更优的做法。
过程是可逆的,我们考虑m变成n,有两种操作:1是m+=1;2是当m为偶数时,m/=2。
要用最少的次数使得m<=n。
因为m加两次再除以二和先除以二再加一次得到的结果是一样的,即能除就不加。这样就O(logn)解决了。

正推n到m好像推不出性质啊。。正推不行就试试倒推吧。

#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() getchar()

inline int read()
{
	int now=0;register char c=gc();
	for(;!isdigit(c);c=gc());
	for(;isdigit(c);now=now*10+c-'0',c=gc());
	return now;
}

int main()
{
	int n=read(),m=read(),ans=0;
	for(; m>n; m>>=1, ++ans)
		if(m&1) ++m, ++ans;
	printf("%d\n",ans+n-m);

	return 0;
}
posted @ 2018-08-29 17:54  SovietPower  阅读(189)  评论(0编辑  收藏  举报