Leetcode 17 电话号码的字母组合 &22 括号生成

思路:用队列,相当于树的层次遍历

17.电话号码的字母组合

{a,b,c}

a出队列,ad,ae,af如、入队列{b,c,ad,ae,af}

以此类推

循环退出条件:电话数字遍历完毕

答案:当前在队列中的所有字符串

class Solution {
private:
	string map[10] = { "","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz" };
public:
	vector<string> letterCombinations(string digits) {
		vector<string> ans;
        if(digits.size()==0) return ans;
        
		vector<int> num;//剥离出每一个电话数字
		for (int i = 0; i < digits.size(); i++)
			num.push_back(digits[i]-'0');//千万注意char型到int型的转换

		queue<string> q;
		int k = 1;//记录匹配到了第几个数字
		for (int i = 0; i < map[num[0]].size(); i++)
		{
			string tmp;
			tmp.push_back(map[num[0]][i]);//注意string的元素map[i][j]是char型,要转换
			q.push(tmp);
		}
		int size = q.size();
		while (!q.empty() && k <num.size())
		{
			for (int i = 0; i < map[num[k]].size(); i++)
			{
				string s = q.front();//s一定要在里面定义,等于每一轮循环都初始化,否则影响后续
				s += map[num[k]][i];
				q.push(s);
			}
			q.pop();
			size--;
			if (size == 0)
			{
				size = q.size();
				k++;
			}
		}
		while (!q.empty())
		{
			ans.push_back(q.front());
			q.pop();
		}
		return ans;
	}

};

 

 

22 括号生成

class Solution {
private:
    struct Tnode{
        int value,lnum;
        string s;
        Tnode(int _value,int _lnum,string _s)
        {
            value=_value;//定义"("为1,")"为-1,value为括号字符串等价值
            lnum=_lnum;//左括号的个数
            s=_s;//括号字符串
        }
    };
public:
    vector<string> generateParenthesis(int n) {
        vector<string> ans;
        queue<Tnode> q;
        if(n==0) return ans;
        if(n==1) 
        {
            ans.push_back("()");
            return ans;
        }
        Tnode node(1,1,"(");
        q.push(node);
        int level=1;//二叉树的层次
        while(!q.empty()&&level<n*2)//level这里不取等号,因为在倒数第二层这里已经把所有根节点入队列了
        {
            int size=q.size();//计录该层节点个数
            Tnode node=q.front();
            q.pop();
            if(node.lnum<n)
            {//左括号
                Tnode tmp(node.value+1,node.lnum+1,node.s+"(");//左括号使得value+1
                q.push(tmp);
            }
            if(node.value>0) 
            {//右括号
                Tnode tmp(node.value-1,node.lnum,node.s+")");//右括号使得value-1
                q.push(tmp);
            }
            if(node.lnum==n&&node.value==0)//左括号达到n,且值value中和完毕
            {//一种情况组合完毕
                ans.push_back(node.s);
            }
            if((--size)==0){
                size=q.size();
                level++;
            }
        }
        return ans;
    }
};

 

看到一位大神用的递归,很简洁:

class Solution {
    public List<String> generateParenthesis(int n) {
        List<String> res = new ArrayList<String>();
        generate(res, "", 0, 0, n);
        
        return res;
    }
        //count1统计“(”的个数,count2统计“)”的个数
    public void generate(List<String> res , String ans, int count1, int count2, int n){
        
        if(count1 > n || count2 > n) return;
        
        if(count1 == n && count2 == n)  res.add(ans);
 
        
        if(count1 >= count2){//保证了在整个递归过程中"("始终>=")"
            String ans1 = new String(ans);//记录原始ans
            generate(res, ans+"(", count1+1, count2, n);
            generate(res, ans1+")", count1, count2+1, n);//如果对于()的情况再加),下一次递归就会被pass
            
        }
    }

}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是电话号码字母组合问题的 C++ 代码实现: ``` class Solution { public: vector<string> letterCombinations(string digits) { unordered_map<char, string> mapping = { {'2', "abc"}, {'3', "def"}, {'4', "ghi"}, {'5', "jkl"}, {'6', "mno"}, {'7', "pqrs"}, {'8', "tuv"}, {'9', "wxyz"} }; vector<string> res; if (digits.empty()) { return res; } string combination; backtrack(res, combination, digits, 0, mapping); return res; } void backtrack(vector<string>& res, string& combination, string& digits, int index, unordered_map<char, string>& mapping) { if (index == digits.size()) { res.push_back(combination); return; } for (char c : mapping[digits[index]]) { combination.push_back(c); backtrack(res, combination, digits, index + 1, mapping); combination.pop_back(); } } }; ``` 其中 `letterCombinations` 函数用来生成所有的字母组合,`backtrack` 函数用来进行回溯操作。在 `letterCombinations` 函数中,首先根据数字字符和字母的映射关系创建了一个 `unordered_map` 对象 `mapping`。然后定义了一个空字符串 `combination` 和一个空 vector `res` 来保存最终结果。最后调用了 `backtrack` 函数来生成所有的字母组合。在 `backtrack` 函数中,首先判断是否达到了数字字符串的末尾,如果是,则将当前的 `combination` 字符串保存到 `res` 中。否则,遍历当前数字字符所能表示的所有字母,依次加入到 `combination` 字符串中,然后递归调用 `backtrack` 函数,添加下一个数字字符所能表示的字母。递归完成后,需要将 `combination` 字符串还原到上一个状态,以便进行下一次回溯。最终返回 `res` 数组即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值