算法课作业之删数问题

问题描述:

通过键盘输入一个高精度的正整数n(n的有效位数≤240),去掉其中任意s个数字后,剩下的数字按原左右次序将组成一个新的正整数。编程对给定的n和s,寻找一种方案,使得剩下的数字组成的新数最小。

问题分析:

这个问题是最优子结构问题,即局部最优能决定全局最优解,可以使用贪心算法进行解决。n个正整数去掉s个数字,求使得到的新的正整数最大的删除方案可以等价为:对于n个正整数组成的数字,一个一个地依次去掉s个数字,要求每删除一个数时,都使删除后的新的正整数最小。因此问题转化为求解删除一个数字时是新的数字最小的方案,求得这个方案后只需要对其执行s次即可。

因为删数后剩下的数字原左右次序不变,所以要尽可能在左边删除,我们从左往右进行考虑,假设n个整数为a(1),a(2)....a(n),当由左往右第一次出现a(k)>a(k+1),可以发现,①删除任何一个a(k)之前的元素得到的新数都比删除a(k)得到的新数大,所以可以明确a(k)之前的元素可以排除;②倘若删除a(k),则新的数字第k位变成a(k+1),记这个新的数字为c,倘若删除a(k)之后的任何一位,得到的新的数字第k位仍然是a(k),记这个新的数字为d。比较c和d,可以发现,c(1)=d(1),......,c(k-1)=d(k-1),因为a(k)>a(k+1),所以c(k)<d(k),那么有c<d。综上可知,对于数字a,应该删除由左往右第一次出现a(k)>a(k+1)时的a(k),如果a的数字序列一直是非减的,那么删除最后一位即可。

本算法的完整C++代码如下:

#include "stdafx.h"
#include "vector"
#include "cmath"
#include "iostream"

using namespace std;

int digit_to_int(char d)//将char数组中的元素转换为整数以便于进行比较
{
    char str[2];

    str[0] = d;
    str[1] = ‘/0‘;
    return (int)strtol(str, NULL, 10);
}

void DeleteOneBit(char * a,int _size){//每次只删除一位的函数
    int i = 0;
    bool _is = false;//用于判断是否需要将部分右边的元素左移一位
    while (i < _size - 1){
        if (digit_to_int(a[i])>digit_to_int(a[i+1])){
            _is = true;
            break;}
        i++;
    }
    if (_is){
        for (; i < _size - 1;i++)
        {
            a[i] = a[i + 1];
        }
    }
    a[_size - 1] = ‘\0‘;//写终止符
}
void GetNewNum(char* b,int _num_d){
    int _lenght = strlen(b);
    if (_num_d>=_lenght){
        cout << "删除的位数过多" << endl;
        return;}
    for (int i = 0; i < _num_d;i++){
        DeleteOneBit(b, _lenght);
        _lenght--;
    }

}
int _tmain(int argc, _TCHAR* argv[])
{
    cout << "请输入要进行删数的正整数(小于240位)"<<endl;
    char a[241];
    cin >> a;
    int _num_d;
    cout << "请输入要删除的位数(小于输入的整数的位数)" << endl;
    cin >> _num_d;
    GetNewNum(a, _num_d);
    cout << a;
    system("pause");
    return 0;
}
时间: 2024-11-05 08:12:14

算法课作业之删数问题的相关文章

1144: 零起点学算法51——数组中删数

1144: 零起点学算法51--数组中删数 Time Limit: 1 Sec  Memory Limit: 64 MB   64bit IO Format: %lldSubmitted: 3304  Accepted: 933[Submit][Status][Web Board] Description 在给定的数组中删除一个数 Input 多组测试,每组第一行输入1个整数n(n<20),然后是n个整数 第二行输入1个整数m Output 删除在第一行的n个整数中第一次出现数字m并删除,然后按

1145: 零起点学算法52——数组中删数II (有问题!)

1145: 零起点学算法52--数组中删数II Time Limit: 1 Sec  Memory Limit: 64 MB   64bit IO Format: %lldSubmitted: 2935  Accepted: 793[Submit][Status][Web Board] Description 在给定的数组中删除数 Input 多组测试,每组第一行输入1个整数n(n<20),然后是n个整数 第二行输入1个整数m Output 删除在第一行的n个整数中的数字m(多个的话都要删除),

解题报告—— 2018级2016第二学期第五周作业 删数问题

解题报告--  2018级2016第二学期第五周作业 删数问题 描述 键盘输入一个高精度的正整数n(<=240位),去掉其中任意s个数字后剩下的数字按原左右次序将组成一个新的正整数.编程对给定的n和s,寻找一种方案,使得剩下的数字组成的新数最小. 输入ns输出最后剩下的最小数样例输入 178543 4 样例输出 13 分析: 这题题目上已表明是贪心算法:原本最容易产生的错误贪心准则是删去其中最大的数字:但通过简单举例便可得之,这种贪心准则要漏洞:通过简单的计算举例发现如果这个数是一位比一位大的话

基于案例SQL优化第九课作业分享

默认统计信息收集: 1. 11g默认启动了统计信息收集的任务,默认运行时间是周一到周五晚上10点和周6,周天的早上6点 2. 你也可以关闭自动统计新收集任务,选择手工收集的方式,但是一般不建议这样操作. 动态统计信息: 1. 统计信息默认情况下是每天晚上10点半后收集,如果新建对象还没来得级收集统计信息,就采用动态采样的方式. 2. 具体在set autotrace 跟踪的执行计划中,可以看到类似:- dynamic sampling used for this statement (level

删数问题--贪心

删数问题 Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 给定n 位(n≤100)正整数a,去掉其中任意k≤n 个数字后,剩下的数字按原次序排列组成一个新的正整数.对于给定的n 位正整数a和正整数k,设计一个算法找出剩下数字组成的新数最小的删数方案. 对于给定的正整数a,计算删去k 个数字后得到的最小数. 输入 输入数据的第1 行是1 个正整数a.第2 行是正整数k. 输出 将计算出的最小数输出. 示例输入 178543 4

1748: 删数问题

题目描述 给定n 位(n≤100)正整数a,去掉其中任意k≤n 个数字后,剩下的数字按原次序排列组成一个新的正整数.对于给定的n 位正整数a和正整数k,设计一个算法找出剩下数字组成的新数最小的删数方案. 对于给定的正整数a,计算删去k 个数字后得到的最小数. 输入 输入数据的第1 行是1 个正整数a.第2 行是正整数k. 输出 将计算出的最小数输出. 样例输入 178543 4 样例输出 13 #include <iostream> #include <string> using

数据结构课作业系列

数据结构课作业-----顺序表 1 /**************************** 2 3 程序名称:简单电话号码查询系统 V.0.1 4 主要功能:简单联系人录入,查找,排序..... 5 主要结构:静态数组类链表 6 7 **************************/ 8 9 #include <cstdio> 10 #include <cstring> 11 #include <iostream> 12 #include <algorit

codevs4096 删数问题

题目描述 Description 键盘输入一个高精度的正整数N,去掉其中任意S个数字后剩下的数字按原左右次序将组成一个新的正整数.编程对给定的N 和S,寻找一种方案使得剩下的数字组成的新数最小. 输入数据均不需要判错. 输出组成的新的正整数.(N不超过240位) 输入描述 Input Description 第一行,输入一正整数N(N<=10240),表示要删的数: 第二行,输入一正整数S,表示删去的个数,当然S小于N的位数. 输出描述 Output Description 仅一行,输出删数后形

洛谷P1106 删数问题

题目描述 键盘输入一个高精度的正整数N,去掉其中任意k个数字后剩下的数字按原左右次序将组成一个新的正整数.编程对给定的N和k,寻找一种方案使得剩下的数字组成的新数最小. 输出应包括所去掉的数字的位置和组成的新的整数.(N不超过250位) 输入数据均不需判错. 输入输出格式 输入格式: n (高精度的正整数) k (需要删除的数字个数) 输出格式: 最后剩下的最小数. 输入输出样例 输入样例#1: 175438 4 输出样例#1: 13分析:贪心思想,我们每次删数肯定是要找前面的大数给删掉,并且这