zoukankan      html  css  js  c++  java
  • [Offer收割]编程练习赛34

    共同富裕

    显然每次选最大的数字,其余的加一。也可以理解为每次选一个最大的数字减一,直到所有数字都变成最小的数字为止。

    #include<stdio.h>
    #include<string.h>
    #include<stdlib.h>
    int cmp(const void * x, const void * y) {
        //x < y
        return (*((int *)(x))) > (*((int *)(y))) ? 1 : -1;
    }
    int a[100005];
    int main() {
    #ifndef ONLINE_JUDGE
        freopen("input.txt", "r", stdin);
    #endif
        int n;
        scanf("%d", &n);
        for (int i = 0; i < n; i++) {
            scanf("%d", &a[i]);
        }
        qsort(a, n, sizeof(int), cmp);
        long long ans = 0;
        for (int i = 1; i < n; i++) {
            ans += a[i] - a[0];
        }
        printf("%lld
    ", ans);
        return 0;
    }
    View Code

    股票价格3

    维护一个数组,按时间顺序排列,表示当前还未被超过的价格。显然,这个数组一定是降序的。因为如果出现i<j且a[i]>a[j],则第i天的价格已经被超过,不应出现在数组中。

    从左到右遍历每天的价格,当处理第i天时,首先将a[i]插入维护的数组,再删除由于a[i]的插入产生的逆序对。由于a[i]插入前数组中得价格均未被超过,所以被删除的价格都是被a[i]首次超过,记录这些价格的日期与i之差。

    #include<stdio.h>
    #include<string.h>
    #include<stdlib.h>
    int cmp(const void * x, const void * y) {
        //x < y
        return (*((int *)(x))) > (*((int *)(y))) ? 1 : -1;
    }
    struct node {
        int price, id;
    };
    int a[100005], ans[100005], size;
    node stack[100005];
    int main() {
    #ifndef ONLINE_JUDGE
        freopen("input.txt", "r", stdin);
    #endif
        int n;
        scanf("%d", &n);
        for (int i = 0; i < n; i++) {
            scanf("%d", &a[i]);
        }
        stack[0].price = a[0], stack[0].id = 0, size = 1;
        memset(ans, -1, sizeof(ans));
        for (int i = 1; i < n; i++) {
            stack[size].price = a[i];
            stack[size].id = i;
            size++;
            while (size > 1) {
                if (stack[size - 2].price >= stack[size - 1].price) {
                    break;
                }
                ans[stack[size - 2].id] = i - stack[size - 2].id;
                stack[size - 2] = stack[size - 1];
                size--;
            }
        }
        for (int i = 0; i < n; i++) {
            printf("%d
    ", ans[i]);
        }
        return 0;
    }
    View Code

     超市规划

    区间动态规划在遇到时间问题时想一想四边形不等式。

    首先,最优的分配肯定满足每个超市控制的小区是连续的一段;其次,一段连续的几个小区由一个超市控制,则最优的点坐标为各个超市坐标的平均数。

    dp[i][j]表示前i个小区由j个超市控制,w[l][r]表示从第l个小区到第r个小区由一个超市控制的最小不方便程度,则dp[i][j]=min{dp[k-1][j-1]+w[k][i] | 0<=k<=r}。

    nk的范围都是2000,时间肯定是不够的。可以用四边形不等式来优化,证明是不会证的,输出了一下各个k点选择的值,发现是满足的,于是就直接上了。

    #include<stdio.h>
    #include<string.h>
    #include<stdlib.h>
    int cmp(const void * x, const void * y) {
        //x < y
        return (*((double *)(x))) > (*((double *)(y))) ? 1 : -1;
    }
    int p[2005][2005];
    double x[2005], sum[2005], c[2005];
    double dp[2005][2005], f[2005][2005];
    double qwe(int l, int r) {
        double sum_ = l == 0 ? sum[r] : sum[r] - sum[l - 1], c_ = l == 0 ? c[r] : c[r] - c[l - 1], ave;
        ave = sum_ / (r - l + 1.0);
        return (r - l + 1.0) * ave * ave - 2 * sum_ * ave + c_;
    }
    
    int main() {
    #ifndef ONLINE_JUDGE
        freopen("input.txt", "r", stdin);
    #endif
        int n, k;
        scanf("%d%d", &n, &k);
        for (int i = 0; i < n; i++) {
            scanf("%lf", &x[i]);
        }
        qsort(x, n, sizeof(double), cmp);
        if (k >= n) {
            printf("0.000
    ");
            return 0;
        }
        sum[0] = x[0], c[0] = x[0] * x[0];
        for (int i = 1; i < n; i++) {
            sum[i] = sum[i - 1] + x[i];
            c[i] = c[i - 1] + x[i] * x[i];
        }
        for (int i = 0; i <= n; i++) {
            for (int j = 0; j <= k; j++) {
                if (i < j) {
                    dp[i][j] = 0;
                } else {
                    dp[i][j] = 1e10;
                }
            }
        }
        for (int i = 1; i < n; i++) {
            for (int j = 1; j <= k; j++) {
                if (i < j) {
                    dp[i][j] = 0;
                    continue;
                }
                dp[i][j] = qwe(0, i);
                p[i][j] = 0;
                int st = p[i - 1][j] > p[i][j - 1] ? p[i][j - 1] : p[i - 1][j];
                for (int l = st; l <= i; l++) {
                    double tmp = qwe(l, i);
                    if (dp[i][j] > dp[l - 1][j - 1] + tmp) {
                        dp[i][j] = dp[l - 1][j - 1] + tmp;
                        p[i][j] = l;
                    }
                }
            }
        }
        printf("%.3lf
    ", dp[n - 1][k]);
        return 0;
    }
    View Code

     有趣的子区间

    首先构造出1-1e9中所有的回文数,然后依次计算包含0个、2个......回文数的区间个数。计算时提出公因子,可以减少计算时间。

    没调通,懒的再搞了。

    #include<stdio.h>
    #include<string.h>
    #include<stdlib.h>
    #include<vector>
    #include<algorithm>
    using namespace std;
    vector<long long>  v;
    const long long maxn = 1000000000;
    const long long p[10] = {1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000};
    long long d[120000], s[120000];
    int cmp(const void * x, const void * y) {
        //x < y
        return (*((double *)(x))) > (*((double *)(y))) ? 1 : -1;
    }
    int main() {
    #ifndef ONLINE_JUDGE
        freopen("input.txt", "r", stdin);
    #endif
        v.clear();
        for (int i = 1; i < 10; i++) v.push_back(i), v.push_back(i * 10 + i);
        for (int i = 2; i < 6; i++) {
            int tmp;
            long long ttmp;
            for (int j = p[i - 1]; j < p[i]; j++) {
                tmp = j;
                for (int k = 0; k < i; k++) {
                    d[k] = tmp % 10;
                    tmp /= 10;
                }
                if (i < 5) {
                    for (int k = 0; k < i; k++) d[2 * i - 1 - k] = d[k];
                    ttmp = 0;
                    for (int k = 2 * i - 1; k >= 0; k--)ttmp = ttmp * 10 + d[k];
                    v.push_back(ttmp);
                }
                for (int k = 0; k < i - 1; k++) d[2 * k - 2 - k] = d[k];
                ttmp = 0;
                for (int k = 2 * i - 2; k >= 0; k--)ttmp = ttmp * 10 + d[k];
                v.push_back(ttmp);
            }
        }
        v.push_back(0), v.push_back(1000000001);
        sort(v.begin(), v.end());
        long long a, b;
        scanf("%lld%lld", &a, &b);
        int l, r;
        for (int i = 0; i < v.size(); i++) {
            if (v[i] >= a) {
                l = i;
                break;
            }
        }
        for (int i = v.size() - 1; i >= 0; i--) {
            if (v[i] <= b) {
                r = i;
                break;
            }
        }
        for (int i = l; i < r; i++) d[i] = v[i + 1] - v[i];
        d[r] = b > v[r] ? b - v[r] : 1;
        s[r] = d[r], s[r - 1] = d[r - 1];
        for (int i = r - 2; i >= l; i--) s[i] = s[i + 2] + d[i];
        long long ans = 0;
        for (int i = l; i <= r; i++) {
            long long pl, pr;
            if (i == l) pl = v[l] - a + 1;
            else pl = d[i - 1];
            pr = s[i + 1];
            ans += pl * pr;
        }
        ans += (v[l] - a) * (v[l] - a + 1) / 2;
        ans += (b - v[r]) * (b - v[r] + 1) / 2;
        for (int i = l; i < r; i++) ans += (d[i] - 1) * d[i] / 2;
        printf("%lld
    ", ans);
        return 0;
    }
    View Code
  • 相关阅读:
    穷举和迭代
    for循环练习题
    case when then else end 用法
    如何将数据库账号(用户)解锁
    比赛安排
    How to spend you day ?
    异常-问题型
    重载和重写的区别
    new关键字的理解-问题型
    源辰项目-1
  • 原文地址:https://www.cnblogs.com/dramstadt/p/7794021.html
Copyright © 2011-2022 走看看