AtCoder beginner contest #159 题解(EF)

比赛地址

E:
题意:
给你一个高为h宽为w的矩阵,里面由01构成,现在问你最少切几刀能使每块1的和不超过k。

思路:
一开始没看到h<=10, 就觉得枚举肯定做不了,后来看到<=10就豁然开朗,用状压来枚举嘛。所以题目条件对于思路的提示也是很重要的,以后看到数据范围很小的,又感觉可以枚举每个条件做的,就可以考虑一下状压了。

ac代码:

/**
* Think twice, code once.
* 1.integer overflow(maybe even long long overflow : (a+b >= c) -> (a >= c-b)
* 2.runtime error
* 3.boundary condition
* ---------------------------------------------------------------------------------------
* Author : zzy
* Date : 2020-03-22-20.33.12 Sunday
*/
#include <bits/stdc++.h>

#define eb emplace_back
#define mp make_pair
#define mt make_tuple
#define fi first
#define se second
#define pb push_back
#define all(x) (x).begin(), (x).end()
#define rall(x) (x).rbegin(), (x).rend()
#define forn(i, n) for (int i = 0; i < (int)(n); ++i)
#define for1(i, n) for (int i = 1; i <= (int)(n); ++i)
#define ford(i, a, b) for (int i = (int)(a); i >= (int)b; --i)
#define fore(i, a, b) for (int i = (int)(a); i <= (int)(b); ++i)
#define rep(i, l, r) for (int i = (l); i <= (r); i++)
#define per(i, r, l) for (int i = (r); i >= (l); i--)
#define ms(x, y) memset(x, y, sizeof(x))
#define SZ(x) ((int)(x).size())

using namespace std;

typedef pair<int, int> pii;
typedef vector<int> vi;
typedef vector<pii> vpi;
typedef vector<vi> vvi;
typedef long long i64;
typedef vector<i64> vi64;
typedef vector<vi64> vvi64;
typedef pair<i64, i64> pi64;
typedef double ld;

template<class T> bool uin(T &a, T b) { return a > b ? (a = b, true) : false; }
template<class T> bool uax(T &a, T b) { return a < b ? (a = b, true) : false; }

int h, w, k, team[50], val[50][1100];
char mat[50][1100];

int makeTeam(int x) {
    vi a;
    for (int i = 0; i <= h-2; ++i) {
        if (x&(1<<i)) {
            a.eb(i);
        }
    }
    int tot = 1;
    if (!SZ(a)) {
        forn(i, h) team[i] = 1;
    } else {
        int len = SZ(a);
        for (int i = 0; i <= a[0]; ++i) team[i] = tot;
        ++tot;
        for (int i = 0; i < len-1; ++i) {
            for (int j = a[i]+1; j <= a[i+1]; ++j) {
                team[j] = tot;
            }
            ++tot;
        }
        for (int i = a[len-1]+1; i < h; ++i) team[i] = tot;
    }
    return tot;
}

int main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cout.precision(10);
    cout << fixed;
#ifdef LOCAL_DEFINE
    freopen("input.txt", "r", stdin);
#endif

    cin >> h >> w >> k;
    for (int i = 0; i < h; ++i){
        for (int j = 0; j < w; ++j) {
            cin >> mat[i][j];
            val[i][j] = int(mat[i][j]-'0');
        }
    }
    int ans = h-1+w-1;
    for (int i = 0; i < (1<<(h-1)); ++i) {
        ms(team, -1);
        int cnt = makeTeam(i)-1;
        int now[50];
        ms(now, 0);
        bool impossible = false;
        int pref = 0;
        for (int j = 0; j < w; ++j) {
            for (int o = 0; o < h; ++o) {
                now[team[o]] += val[o][j];
                if (now[team[o]] > k) {
                    ms(now, 0);
                    ++cnt;
                    if (pref == j) impossible = true;
                    pref = j;
                    --j;
                    break;
                }
            }
            if (impossible) break;
        }
        if (!impossible) ans = min(ans, cnt);
    }
    cout << ans << '\n';

#ifdef LOCAL_DEFINE
    cerr << "Time elapsed: " << 1.0 * clock() / CLOCKS_PER_SEC << " s.\n";
#endif
    return 0;
}

F
题意:
输入n,s, 接下来输入n个数ai,现在定义f(l,r)为[l,r]区间内一段子序列(可以不连续)的数和为s的个数,(比如有三个数2,2, 4, s为4,因为2+2=4, 4=4, 所以f(1,3)= 2),因为[l,r]有n*(n-1)/2种可能,现在问你计算这全部情况然后加起来的答案是多少。

思路:
感觉思路很巧妙,比赛时没做出来,是道好题。
比较明显的可以想到从每个区间等于s的贡献度出发,一个区间的贡献度等于左边的长度*右边的长度,然而如果把每个区间都这么搞的话肯定又很多重复的,所以这题可以用dp[i][j]来表示处理到前i数的所有区间内和为j的个数(这样可以保证不重复),开始转移:
1.dp[i][j]=dp[i-1][j]+dp[i-1][j-a[i]] (a[i] <= j <= s)
2.dp[i][a[i]]=dp[i][a[i]]+i

ac代码:

/**
* Think twice, code once.
* 1.integer overflow(maybe even long long overflow : (a+b >= c) -> (a >= c-b)
* 2.runtime error
* 3.boundary condition
* ---------------------------------------------------------------------------------------
* Author : zzy
* Date : 2020-03-23-03.25.36 Sunday
*/
#include <bits/stdc++.h>

#define eb emplace_back
#define mp make_pair
#define mt make_tuple
#define fi first
#define se second
#define pb push_back
#define all(x) (x).begin(), (x).end()
#define rall(x) (x).rbegin(), (x).rend()
#define forn(i, n) for (int i = 0; i < (int)(n); ++i)
#define for1(i, n) for (int i = 1; i <= (int)(n); ++i)
#define ford(i, a, b) for (int i = (int)(a); i >= (int)b; --i)
#define fore(i, a, b) for (int i = (int)(a); i <= (int)(b); ++i)
#define rep(i, l, r) for (int i = (l); i <= (r); i++)
#define per(i, r, l) for (int i = (r); i >= (l); i--)
#define ms(x, y) memset(x, y, sizeof(x))
#define SZ(x) ((int)(x).size())

using namespace std;

typedef pair<int, int> pii;
typedef vector<int> vi;
typedef vector<pii> vpi;
typedef vector<vi> vvi;
typedef long long i64;
typedef vector<i64> vi64;
typedef vector<vi64> vvi64;
typedef pair<i64, i64> pi64;
typedef double ld;

template<class T> bool uin(T &a, T b) { return a > b ? (a = b, true) : false; }
template<class T> bool uax(T &a, T b) { return a < b ? (a = b, true) : false; }

const i64 maxn = 3100;
const i64 mod = 998244353;
i64 dp[maxn][maxn], n, s, a[maxn], ans=0;

int main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cout.precision(10);
    cout << fixed;
#ifdef LOCAL_DEFINE
    freopen("input.txt", "r", stdin);
#endif

    cin >> n >> s;
    for1(i, n) cin >> a[i];
    ms(dp, 0);
    for (int i = 1; i <= n; ++i) {
        for (int j = 0; j <= s; ++j) {
            dp[i][j] = dp[i-1][j];
        }
        for (int j = a[i]; j <= s; ++j) {
            dp[i][j] = (dp[i-1][j]+dp[i-1][j-a[i]])%mod;
        }
        dp[i][a[i]] = (dp[i][a[i]]+i)%mod;
        ans = (ans+dp[i][s])%mod;
    }
    cout << ans << '\n';

#ifdef LOCAL_DEFINE
    cerr << "Time elapsed: " << 1.0 * clock() / CLOCKS_PER_SEC << " s.\n";
#endif
    return 0;
}

滚动数组优化:

/**
* Think twice, code once.
* 1.integer overflow(maybe even long long overflow : (a+b >= c) -> (a >= c-b)
* 2.runtime error
* 3.boundary condition
* ---------------------------------------------------------------------------------------
* Author : zzy
* Date : 2020-03-23-03.25.36 Sunday
*/
#include <bits/stdc++.h>

#define eb emplace_back
#define mp make_pair
#define mt make_tuple
#define fi first
#define se second
#define pb push_back
#define all(x) (x).begin(), (x).end()
#define rall(x) (x).rbegin(), (x).rend()
#define forn(i, n) for (int i = 0; i < (int)(n); ++i)
#define for1(i, n) for (int i = 1; i <= (int)(n); ++i)
#define ford(i, a, b) for (int i = (int)(a); i >= (int)b; --i)
#define fore(i, a, b) for (int i = (int)(a); i <= (int)(b); ++i)
#define rep(i, l, r) for (int i = (l); i <= (r); i++)
#define per(i, r, l) for (int i = (r); i >= (l); i--)
#define ms(x, y) memset(x, y, sizeof(x))
#define SZ(x) ((int)(x).size())

using namespace std;

typedef pair<int, int> pii;
typedef vector<int> vi;
typedef vector<pii> vpi;
typedef vector<vi> vvi;
typedef long long i64;
typedef vector<i64> vi64;
typedef vector<vi64> vvi64;
typedef pair<i64, i64> pi64;
typedef double ld;

template<class T> bool uin(T &a, T b) { return a > b ? (a = b, true) : false; }
template<class T> bool uax(T &a, T b) { return a < b ? (a = b, true) : false; }

const i64 maxn = 3100;
const i64 mod = 998244353;
i64 dp[maxn], n, s, a[maxn], ans=0;

int main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cout.precision(10);
    cout << fixed;
#ifdef LOCAL_DEFINE
    freopen("input.txt", "r", stdin);
#endif

    cin >> n >> s;
    for1(i, n) cin >> a[i];
    ms(dp, 0);
    for (int i = 1; i <= n; ++i) {
        for (int j = s; j >= a[i]; --j) dp[j] = (dp[j]+dp[j-a[i]])%mod;
        dp[a[i]] = (dp[a[i]]+i)%mod;
        ans = (ans+dp[s])%mod;
    }
    cout << ans << '\n';

#ifdef LOCAL_DEFINE
    cerr << "Time elapsed: " << 1.0 * clock() / CLOCKS_PER_SEC << " s.\n";
#endif
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值