c++回溯法解決1到9之間插入加減或空使運(yùn)算結(jié)果為100
問題分析
這時(shí)我最近偶然看到的一道題目,發(fā)現(xiàn)實(shí)現(xiàn)起來還確實(shí)有些麻煩,所以把實(shí)現(xiàn)的過程記錄下來。
這種要羅列出所有結(jié)果的問題,我一般是采用回溯法解決的,說的通俗一點(diǎn)就是暴力解法,去遍歷所有的情況。
這個(gè)問題有一點(diǎn)比較難處理的地方就在于有這個(gè)“什么都不插入”這個(gè)選項(xiàng),所以干脆單獨(dú)拎出來解決。也就是先把1-9這9個(gè)數(shù)組相互組合,形成一個(gè)數(shù)組,比如:
{1,2,3,4,5,6,7,8,9}
{1,2,3,4,5,6,7,89}
{1,2,3,4,5,6,78,9}
{1,2,3,4,5,6,789}
...
在分組的過程當(dāng)中,由于問題的特殊性(要求結(jié)果為100),我們會(huì)發(fā)現(xiàn)像
{123456,789}
這樣位數(shù)特別大的是不可能得到100這樣的結(jié)果的,一個(gè)最小的6位數(shù)和一個(gè)最大的3位數(shù)的差都有
100000−999=99001
所以本問題中不用考慮把1-9劃分成6位數(shù)及以上的情況。
將1-9劃分好之后,接下來要做的就是把”+”和”-“填到劃分的數(shù)字之間了,比如
劃分成{1,2,3,4,5,6,7,8,9}時(shí)有:
1+2+3+4+5+6+7+8+9
1+2+3+4+5+6+7+8-9
1+2+3+4+5+6+7-8+9
...
劃分成{1,2,3,4,5,6,7,89}時(shí)有:
1+2+3+4+5+6+7+89
1+2+3+4+5+6+7-89
...
其他情況就不列舉了,相信應(yīng)該看明白了
基于這樣的思路,用C++對(duì)該想法進(jìn)行了實(shí)現(xiàn)。
代碼展示
下面程序可以將結(jié)果100改成其他的整數(shù),都是適用的。
#include <iostream> #include <math.h> #include <vector> #include <string> using namespace std; class Solution{ private: vector<string> res; vector<int> nums; vector<int> eles; private: void _compute(vector<int> vec, int index, int target, string &s){ if (index == vec.size()){ if (0 == target) res.push_back(s + "=100"); return; } //分“+”和“-”兩種情況討論 for (int i = 0; i < 2; i++){ if (i == 0){ string tempStr = s + "+" + to_string(vec[index]); _compute(vec, index + 1, target - vec[index], tempStr); } else if (i == 1){ string tempStr = s + "-" + to_string(vec[index]); _compute(vec, index + 1, target + vec[index], tempStr); } } return; } //用來得到1-9的不同整數(shù)組合,比如{123, 456, 789},本質(zhì)是將“”這個(gè)空符號(hào)加入到數(shù)之間 void _recursion(int index, int target){ if (index == 9){ string s = to_string(eles[0]); _compute(eles, 1, target - eles[0], s); return; } //為了問題的泛化采用i <= 9,如果針對(duì)結(jié)果為100的可以改成i <= 5 for (int i = 1; i <= 9; i++){ if (index + i > 9) break; int temp = 0; //求得分解出來的每個(gè)元素的值 for (int j = 0; j < i; j++){ temp += nums[index + j] * pow(10, i - j - 1); } eles.push_back(temp); _recursion(index + i, target); eles.pop_back(); } return; } public: Solution(){ nums = { 1, 2, 3, 4, 5, 6, 7, 8, 9 }; } vector<string> recursion(int index, int target){ _recursion(index, target); return res; } }; int main() { Solution s; vector<string> res = s.recursion(0, 100); cout << "共有" << res.size() << "種情況" << endl; for (string s : res){ cout << s << endl; } return 0; }
以上就是c++回溯法解決1-9之間插入加減或空使運(yùn)算結(jié)果為100的詳細(xì)內(nèi)容,更多關(guān)于c++回溯法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
統(tǒng)計(jì)C語言二叉樹中葉子結(jié)點(diǎn)個(gè)數(shù)
這篇文章主要介紹的是統(tǒng)計(jì)C語言二叉樹中葉子結(jié)點(diǎn)個(gè)數(shù),文章以C語言二叉樹中葉子結(jié)點(diǎn)為基礎(chǔ)分享一個(gè)簡(jiǎn)單小栗子講解,具有一定的知識(shí)參考價(jià)值,需要的小伙伴可以參考一下2022-02-02C++11中條件標(biāo)量和互斥鎖應(yīng)用出現(xiàn)死鎖問題
這篇文章主要介紹了C++11中條件標(biāo)量和互斥鎖應(yīng)用出現(xiàn)死鎖思考,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06