【每日一题】简单瞎搞题

简单瞎搞题

https://ac.nowcoder.com/acm/problem/17193

心态炸了 不小心关掉 又要重新写

Description

一共有 个数,第 个数是 可以取 中任意的一个值。设 ,求 S 种类数。

Solution

这个问题的难点在于如何统计出所有和可能出现的情况,并且不能重复。
很容易想到用桶去存储每一个数,即某个和能够组合出来则为1,否则为0
不妨令 表示为第 次选择时,和为 的情况是否出现过
但是内存方面需要 内存,显然是不可接受的
那么我们考虑用 优化一下,有递推方程
代表第 次选择的时候是否能从当前状态转移到和为 的状态

Code

#include<bits/stdc++.h>

typedef long long ll;

using namespace std;

const ll mod = 1e9 + 7;
const int N = 2e5 + 5;
bitset<100 * 100 * 100 + 5> dp[105];
int main() {
  int n;
  cin >> n;
  dp[0][0] = 1;
  for(int i = 1; i <= n; i++) {
    int l, r;
    cin >> l >> r;
    for(int j = l; j <= r; j++) {
      dp[i] |= (dp[i - 1] << (j * j));
    }
  }
  cout << dp[n].count() << "\n";
  return 0;
}

当然, 我们发现, 只会和 有关, 可以用滚动数组继续优化一些内存

Code

/*
  autor: Kurisu
*/
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

const long long inf = 1e18;
const int N = 1e5 + 5;
const double eps = 1e-10;
const ll mod = 1e9 + 7;

int a[N];
bitset<1000005> now, nxt;
int main() {
  ios::sync_with_stdio(false), cin.tie(nullptr);
  now[0] = 1;
  int n;
  cin >> n;
  for(int i = 1; i <= n; i++) {
    int l, r;
    cin >> l >> r;
    for(int j = l; j <= r; j++) {
      if(j == l) nxt = (now << (j * j));
      else nxt |= (now << (j * j));
    }
    now = nxt;
  }
  cout << now.count() << "\n";
}
// 2 3 1 5 4
Kurisu与牛客的每日一题 文章被收录于专栏

每日一题

全部评论
就算是1e8的bool内存它都不接受..
点赞 回复 分享
发布于 2021-01-12 16:20

相关推荐

_mos_:要不是看评论区我都不知道你要找的是数分
点赞 评论 收藏
分享
11-13 20:16
已编辑
厦门理工学院 软件测试
专业嗎喽:硕佬,把学校背景放后面几段,学校背景双非还学院,让人看了就不想往下看。 把实习经历和个人奖项放前面,用数字化简述自己实习的成果和掌握的技能,比如负责项目一次通过率90%,曾4次发现项目潜在问题风险为公司减少损失等等
点赞 评论 收藏
分享
评论
4
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务