【数论】【分类讨论】【C++算法】1611使整数变为 0 的最少操作次数

作者推荐

【动态规划】【字符串】【行程码】1531. 压缩字符串

涉及知识点

数论 数学 分类讨论

LeetCoce1611. 使整数变为 0 的最少操作次数

给你一个整数 n,你需要重复执行多次下述操作将其转换为 0 :
翻转 n 的二进制表示中最右侧位(第 0 位)。
如果第 (i-1) 位为 1 且从第 (i-2) 位到第 0 位都为 0,则翻转 n 的二进制表示中的第 i 位。
返回将 n 转换为 0 的最小操作次数。
示例 1:
输入:n = 3
输出:2
解释:3 的二进制表示为 “11”
“11” -> “01” ,执行的是第 2 种操作,因为第 0 位为 1 。
“01” -> “00” ,执行的是第 1 种操作。
示例 2:
输入:n = 6
输出:4
解释:6 的二进制表示为 “110”.
“110” -> “010” ,执行的是第 2 种操作,因为第 1 位为 1 ,第 0 到 0 位为 0 。
“010” -> “011” ,执行的是第 1 种操作。
“011” -> “001” ,执行的是第 2 种操作,因为第 0 位为 1 。
“001” -> “000” ,执行的是第 1 种操作。
提示:
0 <= n <= 109

数论

两种方式其实是一种:
如果是最低位,可以翻转。
如果有一位比它低,后边必须是1。
如果有两位比它低,后边必须是10。
如果有三位比它低,后边必须是100。

消除高位时,比它低的位必须是指定值。先低位后高位没有好处。所以先处理最高位。
分以下情况:
一,0,1。直接n。
二,n只有一个二进制位是1。这1后面的0转成1…0…。在将这个1消掉。
三,只有2个1,且挨着一起,消掉高位的1。
四,2个1,不挨着。低位1的前一位变成1,自身变成0。
五,2个以上1,除1的最高位和次高位,其它全为0。

翻转是可逆的,所以a到b和b到a的操作次数完全一样。

代码

核心代码

class Solution {
public:int minimumOneBitOperations(int n) {return Rec(n);}int Rec(int n){if (n <= 1){return n;}if (m_data.count(n)){return m_data[n];}int iLowBit = n & (-n);int iOther = n - iLowBit;if (0 == iOther){//只有一位是1: 0变成n/2后 ,将iLowBit消掉,变成n/2,变成0。 n1 变成n2和n2变成需要的步骤完全一样,逆运算,成立。return m_data[n] = Rec(n / 2) + 1 + Rec(n / 2);}if (iLowBit * 2== iOther){//只有两个1,且挨着一起,方式二消除return	m_data[n] = 1 + Rec(n - 2*iLowBit);}while (iOther & (iOther - 1)){iLowBit = iOther & (-iOther);iOther -= iLowBit;}const int iNew = iOther + iLowBit;if (iNew == n){return m_data[n] = 1 + Rec(iOther + iLowBit * 2) + Rec(iLowBit);}return m_data[n] = Rec(n - iNew) + Rec(iNew);}unordered_map<int, int> m_data;
};

## 测试用例

template<class T>
void Assert(const T& t1, const T& t2)
{assert(t1 == t2);
}template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{if (v1.size() != v2.size()){assert(false);return;}for (int i = 0; i < v1.size(); i++){Assert(v1[i], v2[i]);}}int main()
{	vector<int> balls;{Solution sln;vector<int> nums = {3,6,2,4,5,7};vector<int> ans = {2,4,3,7,6,5};vector<int> ret;for (const auto& n : nums ){ret.emplace_back(sln.minimumOneBitOperations(n));}Assert(ret, ans);}}

2023年2月

class Solution {
public:
int minimumOneBitOperations(int n) {
if (n < 2)
{
return n;
}
if (m_n.count(n))
{
return m_n[n];
}
int iMaxBitValue = GetMaxBitValue(n);
const int iDest = iMaxBitValue >> 1;
const int iSrc = n - iMaxBitValue;
return m_n[n] = Change2(iSrc, iDest) + 1 + minimumOneBitOperations(iDest);
}
int Change2(int iSrc, int iDest)
{
if (iSrc == iDest)
{
return 0;
}
if (1 == iDest)
{
return 1;
}
if (m_change2[iSrc].count(iDest))
{
return m_change2[iSrc][iDest];
}
if (iSrc & iDest)
{
return minimumOneBitOperations(iSrc - iDest);
}
const int iRet = 1 + Change2(iSrc, iDest >> 1) + minimumOneBitOperations(iDest >> 1);
return m_change2[iSrc][iDest] =iRet;
}
inline int GetMaxBitValue(int n )
{
int iMaxBitValue = n;
while (iMaxBitValue&(iMaxBitValue - 1))
{
iMaxBitValue = iMaxBitValue&(iMaxBitValue - 1);
}
return iMaxBitValue;
}
std::unordered_map<int, int> m_n;
std::unordered_map<int, std::unordered_map<int, int>> m_change2;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关

下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/441328.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Ubuntu 22.04 中文乱码解决方案

sudo apkg-reconfigure locales 按空格键选中

[ project.config.json 文件内容错误] project.config.json: libVersion 字段需为

解决方法&#xff1a; 在manifest文件的mp-weixin中添加 "libVersion": "latest" 即可

如何使用postman进行接口自动化测试?

1、什么是自动化测试&#xff1f; 把人对软件的测试行为转化为由机器执行测试行为的一种实践。 例如GUI自动化测试&#xff0c;模拟人去操作软件界面&#xff0c;把人从简单重复的劳动中解放出来&#xff0c;本质是用代码去测试另一段代码&#xff0c;属于一种软件开发工作&a…

idea报错 :(java: 找不到符号)

java: 找不到符号 符号: 变量 adminService 位置: 类 com.example.controller.WebController 查到网上一个办法&#xff1a;因为项目是maven&#xff1a;先点clean在点package

283. Move Zeroes(移动零)

题目描述 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 请注意 &#xff0c;必须在不复制数组的情况下原地对数组进行操作。 问题分析 从后向前&#xff0c;记录第一个非0的数字的下标&#xff0c;然后从…

Uniapp小程序端打包优化实践

背景描述&#xff1a; 在我们最近开发的一款基于uniapp的小程序项目中&#xff0c;随着功能的不断丰富和完善&#xff0c;发现小程序包体积逐渐增大&#xff0c;加载速度也受到了明显影响。为了提升用户体验&#xff0c;团队决定对小程序进行一系列打包优化。 项目优化点&…

产品原型图设计规范大全

目前&#xff0c;市场上许多产品经理或设计师都在使用一些优秀的原型设计规范&#xff0c;这些规范几乎涵盖了原型设计的许多方面。一套好的、完整的原型设计规范可以统一产品设计风格&#xff0c;检验产品的可用性&#xff0c;有效提高产品经理绘制原型图的效率&#xff0c;更…

Python编辑开发 --- pycharm pro 中文

PyCharm Pro是一款专业的Python集成开发环境&#xff08;IDE&#xff09;&#xff0c;由JetBrains公司开发。它为Python开发者提供了丰富的功能和工具&#xff0c;使得Python编程变得更加高效和便捷。PyCharm Pro具有智能代码编辑功能&#xff0c;能够自动完成代码、快速导航至…

【图文详解】阿里云服务器放行高防IP加入安全组

打开阿里云的云服务器配置面板&#xff0c;在要操作实例的操作列找到更多 > 网络和安全组 > 安全组配置。 对已有安全组配置规则&#xff0c;或者直接添加安全组规则。 根据需要放通高防IP在内的IP段相应协议类型的端口访问。

《Docker技术革命:从虚拟机到容器化,全面解析Docker的原理与应用-上篇》

文章目录 Docker为什么会出现总结 Docker的思想Docker历史总结 Docker能干嘛虚拟机技术虚拟机技术的缺点 容器化技术Docker和虚拟机技术的区别 Docker概念Docker的基本组成镜像&#xff08;image)容器&#xff08;container&#xff09;仓科&#xff08;repository&#xff09;…

【DC-DC】AP5125 降压恒流驱动器 60W LED电源驱动方案PCB+BOM表

这是一款60WLED驱动方案,线路图如下 ​ 祥单表&#xff1a; 实物图&#xff1a; 产品描述 特点应用领域应用原理图AP5125 是一款外围电路简单的 Buck 型平均电流检测模式的 LED 恒流驱动器&#xff0c;适用于 8-100V 电压范围的非隔离式大功率恒流 LED 驱动领域。芯片采用固定…

机器学习——绪论总结

目录 一、引入 二、基本术语 三、假设空间与归纳偏 四、模型选择 一、引入 机器学习&#xff1a;通过计算手段&#xff0c;得出具有能够自我修改、完善能力的模型&#xff0c;利用经验改善系统自身性能。算法使用数据得到模型的过程即称为学习&#xff0c;或训练 流程&…