首页 > 其他分享 >Acw 170.加成序列

Acw 170.加成序列

时间:2023-01-30 21:35:46浏览次数:46  
标签:le int qquad dep 加成 Acw 序列 path 170

题目描述

满足如下条件的序列 \(X\)(序列中元素被标号为 \(1、2、3…m\))被称为“加成序列”:

  1. \(X[1]=1\)
  2. \(X[m]=n\)
  3. \(X[1]<X[2]<…<X[m-1]<X[m]\)
  4. 对于每个 \(k\)(\(2 \le k \le m\))都存在两个整数 \(i\) 和 \(j\) (\(1 \le i,j\le k-1\),\(i\) 和 \(j\) 可相等),使得 \(X[k]=X[i]+X[j]\)。

你的任务是:给定一个整数 \(n\),找出符合上述条件的长度 \(m\) 最小的“加成序列”。

如果有多个满足要求的答案,只需要找出任意一个可行解。

解题思路

\(\qquad\) 看到 \(n\le 100\),考虑爆搜,但是这个一般的爆搜过不去,原因如下:递归的层数可能很深(因为序列的长度可以很长),但是答案序列的长度会很短,举个例子,在这个序列中:

\[\large 1\ \ 2\ \ 3\ \ 5\ \ 8\ \ 13\ \ 21\ \ 34\ \ 55\ \ 89\ \ 144 \]

\(\qquad\)长度只有 \(11\),但是很容易构造出一个递增的序列,可以达到 \(144\) 项。为了避免这样的不必要搜索,我们可以采用迭代加深搜索

\[\]

\(\qquad\qquad\qquad\)迭代加深搜索适合搜索规模不定,但是答案在较浅的层数的问题

\[\]

\(\qquad\)具体就是我们先假定一个最大层数,在搜索的时候超过最大层数就不继续,知道搜索成功后,这个最大层数就是我们的答案。

剪枝操作

\(\qquad1.\)优化搜索顺序:因为这是严格递增顺序,所以为了让答案尽快趋近于 \(n\),我们应该从后面的项开始,并且 \(a[i]\) 必定可以由 \(a[i - 1]\) 得到,这里可以采用反证法:因为如果 \(a[i]\) 的组成中不包含 \(a[i - 1]\),那我们把 \(a[i - 1]\) 这一项删去不改变序列的性质,而且答案更优。所以 \(a[i]\) 必定可以由 \(a[i - 1]\) 得到,只要枚举另一个数就可以了。

\(\qquad2.\)排除等效冗余:由于两个数的和可能出现重复,所以我们要排除这种情况,搞个判重数组

\(\qquad3.\)估价函数:由于后一项至多是前一项的两倍,所以当当前数 \(a[i]\times 2^{m - i + 1} < n\) 的时候,代表无论如何也无法达到目标,所以也是不行的。

代码实现

#include <iostream>

using namespace std;

const int N = 110;
int n, path[N];

bool dfs(int u, int dep) 
{
	if (u == dep + 1) return path[dep] == n;
	if (path[u - 1] * (1 << dep - u + 1) < n) return 0;

	int st[N] = {0};
	for (int j = u - 1; j >= 1; j -- ) 
	{
		int sum = path[u - 1] + path[j];
		if (sum <= path[u - 1]) continue ;
		if (st[sum] || sum > n) continue ;

		path[u] = sum, st[sum] = 1;
		if (dfs(u + 1, dep)) return 1;
	} 

	return 0;
}

int main() 
{
	ios::sync_with_stdio(false);
	cin.tie(0), cout.tie(0);

	path[1] = 1;
	while (cin >> n, n) 
	{
		int dep = 2;
		while (!dfs(2, dep)) dep ++ ;

		for (int i = 1; i <= dep; i ++ )
			cout << path[i] << ' ';
		cout << '\n';
	}

	return 0;
}

这些全部搞起来我们的代码就可以稳定 \(20ms\) 以内了 qwq

标签:le,int,qquad,dep,加成,Acw,序列,path,170
From: https://www.cnblogs.com/StkOvflow/p/17077286.html

相关文章

  • Acwing73
    哈希表时空复杂度O(n)classSolution{public:vector<int>findNumsAppearOnce(vector<int>&nums){unordered_map<int,int>m;vector<in......
  • CF1702F
    EquateMultisets-洛谷|计算机科学教育新生态(luogu.com.cn)一道很妙的思维题:此题的关于在认识到:奇数只能被除出来,不能被乘出来认识到这点之后,我们把A,B中的所有偶......
  • CF1703G
    GoodKey,BadKey-洛谷|计算机科学教育新生态(luogu.com.cn)根据题意:我们可以设DP[i][j]表示前i个箱子用了j把钥匙由于 1≤n≤105,所以第二位不可能这......
  • AcWing 01背包问题
    有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。第 i 件物品的体积是 vi,价值是 wi。求解将哪些物品装入背包,可使这些物品的总体积不超过背包容量,且总......
  • AcWing 第 88 场周赛 ABC
    今天T2卡的我有点久(一开始的思考路线错了,手又冻的哆哆嗦嗦的,手速慢了哈哈(越来越菜了https://www.acwing.com/activity/content/competition/problem_list/2840/AcWing......
  • AcWing 第 87 场周赛 ABC
    https://www.acwing.com/activity/content/competition/problem_list/2814/4797.移动棋子#include<bits/stdc++.h>usingnamespacestd;typedeflonglongLL;typede......
  • AcWing 第87场周赛题解
    T1移动棋子算出为\(1\)的点离\((3,3)\)的距离即可。#include<iostream>#include<cstring>#include<algorithm>usingnamespacestd;intmain(){int......
  • AcWing1081. 度的数量
    题目描述求给定区间\([X,Y]\)中满足下列条件的整数个数:这个数恰好等于\(K\)个互不相等的\(B\)的整数次幂之和。例如,设\(X=15,Y=20,K=2,B=2\),则有且仅......
  • AcWing 1077. 皇宫看守
    解题思路\(\qquad\)题目就不再复述了,我们这题和上一题类似,可以采用树形DP+状态机状态表示\[f[i][j],j\in[0,2]表示的是第i个点,第j种状态\]对于三种状态,有如下......
  • AcWing. 323. 战略游戏
    题意简述\(\qquad\)给定一棵树,要求树中任意一边至少选中一点,求最少满足题意的选点数解题思路\(\qquad\)我们可以先画出示意图来橙色点表示选,灰色点表示不选。\(\qqu......