每日OJ_牛客_Pre-Post(英文题树的遍历_排列组合)

目录

牛客_Pre-Post(英文题树的遍历_排列组合)

解析代码


牛客_Pre-Post(英文题树的遍历_排列组合)


解析代码

        这道题本质上其实是一个排列组合问题。通过前序和后序我们虽然还原不出来树,但是谁是谁的子树我们还是知道的。假设我们的前序是abejkcfghid,后序是jkebfghicda,那么我们根据前序,就能知道:

  1. 最多可以有13颗子树,也就是每一层都有13个可能位置。
  2. a是根,第一棵子树的根是b。
  3. 通过后树我们能知道,b的子树有j、k、e、b共四个结点。
  4. 再回到前序,向前走4个结点,下一棵子树的根是 c。
  5. 以此类推,最终得到 a 为根的下一层共有 3 棵子树。

        三颗子树长这样:前序 bejk cfghi d 后序 jkeb fghic d 则这一层一共的可能性就是13个空位随便挑3个摆这3颗子树,那么有13*12*11 / 3 * 2种可能。

#include <iostream>
#include <string>
#include <vector>
using namespace std;

struct SubTree // 保存子树的前序和后序遍历结果
{
	SubTree(const string& pre, const string& post)
		: _pre(pre)
		, _post(post)
	{}
	string _pre;
	string _post;
};
long long Fac(int n) // 求n的阶乘
{
	long long f = 1;
	for (int i = 1; i <= n; ++i)
	{
		f *= i;
	}
	return f;
}
long long CalcCom(int n, int m) // 求:C(n,m) = C(n, n-m)
{
	m = m < (n - m) ? m : (n - m);
	long long r = 1;
	for (int i = n; i >= n - m + 1; i--)
	{
		r *= i;
	}
	return r / Fac(m);
}
// 找根节点的所有子树
vector<SubTree> CalcSubTree(const string& pre, const string& post)
{
	size_t subRootPreIdx = 1;
	size_t postFirst = 0; // 子树在后序遍历结果中第一个元素的位置
	vector<SubTree> v;
	while (subRootPreIdx < pre.size())
	{
		// 确定子树的根节点
		char subRoot = pre[subRootPreIdx];
		// 确定根在后序遍历结果中的位置
		char subRootPostIdx = post.find(subRoot);
		// 计算该子树中节点的个数
		size_t subTreeNodeCount = subRootPostIdx - postFirst + 1;
		// 找到该棵子树前序遍历结果 - 找到该棵子树后序遍历结果
		SubTree subTree(pre.substr(subRootPreIdx, subTreeNodeCount), post.substr(postFirst, subTreeNodeCount));
		v.push_back(subTree);
		// 根新下一课子树根在前序遍历结果中的下标
		subRootPreIdx += subTreeNodeCount;
		// 更新下一棵子树中第一个节点在后序遍历结果中的下标
		postFirst += subTreeNodeCount;
	}
	return v;
}

long long CalcTreePossible(int m, const string& pre, const string& post)
{
	// 如果树中只有1个节点---树的可能性就是唯一的
	if (1 == pre.size())
		return 1;
	// 先找出根节点的所有子树
	vector<SubTree> v = CalcSubTree(pre, post);
	// 计算根节点子树的可能性---组合
	long long result = CalcCom(m, v.size());
	for (auto& e : v)
		result *= CalcTreePossible(m, e._pre, e._post);
	return result;
}

int main()
{
	int m = 0;
	string pre, post;
	while (cin >> m >> pre >> post)
	{
		if (m == 0)
			break;
		cout << CalcTreePossible(m, pre, post) << endl;
	}
	return 0;
}
  • 8
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
引用是一段关于利用pwntools库对jarvisoj_level0进行攻击的代码。代码中使用了remote()方法建立了远程连接,并构造了一个payload来利用栈溢出漏洞,最终执行callsystem()函数来获取shell权限。引用也是类似的代码,只是连接的地址不同。引用是一篇关于pwntools基本用法的文章,并提供了一些常用的函数和方法。根据这些引用内容,可以得出结论,jarvisoj_level0是一个存在栈溢出漏洞的程序,可以通过构造特定的payload来执行系统调用函数,从而获取shell权限。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [jarvisoj_level0](https://blog.csdn.net/weixin_56301399/article/details/125919313)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [jarvisOJ-level0](https://blog.csdn.net/qq_35661990/article/details/82889103)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [jarvis oj---level0解题方法](https://blog.csdn.net/weixin_45427676/article/details/97272924)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

GR鲸鱼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值