• Codeforces Round #567 (Div. 2)B. Split a Number (字符串,贪心)


    B. Split a Number
    time limit per test2 seconds
    memory limit per test512 megabytes
    inputstandard input
    outputstandard output
    Dima worked all day and wrote down on a long paper strip his favorite number n consisting of l digits. Unfortunately, the strip turned out to be so long that it didn't fit in the Dima's bookshelf.

    To solve the issue, Dima decided to split the strip into two non-empty parts so that each of them contains a positive integer without leading zeros. After that he will compute the sum of the two integers and write it down on a new strip.

    Dima wants the resulting integer to be as small as possible, because it increases the chances that the sum will fit it in the bookshelf. Help Dima decide what is the minimum sum he can obtain.

    Input
    The first line contains a single integer l (2≤l≤100000) — the length of the Dima's favorite number.

    The second line contains the positive integer n initially written on the strip: the Dima's favorite number.

    The integer n consists of exactly l digits and it does not contain leading zeros. Dima guarantees, that there is at least one valid way to split the strip.

    Output
    Print a single integer — the smallest number Dima can obtain.

    Examples
    inputCopy
    7
    1234567
    outputCopy
    1801
    inputCopy
    3
    101
    outputCopy
    11
    Note
    In the first example Dima can split the number 1234567 into integers 1234 and 567. Their sum is 1801.

    In the second example Dima can split the number 101 into integers 10 and 1. Their sum is 11. Note that it is impossible to split the strip into "1" and "01" since the numbers can't start with zeros.

    题意:
    给你一个字符串表示一个整数,让你把整数分成两个分部,两个非空的部分,而且不能有一个部分有前导0,例如分成a,b, 两个部分,然后求哪种分开的方法可以让a+b 最小,输出a+b的数值。
    思路:
    首先我们处理出所有不是0数字的位置,加入到一个vector里,然后我们最vector 进行二分查找距离 len/2 最近的位置 (因为最靠中间分,答案最优),然后枚举二分得到的位置附件的几个位置,求a+b 中的最小值 。输出即可。

    细节见代码:

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    #include <cmath>
    #include <queue>
    #include <stack>
    #include <map>
    #include <set>
    #include <vector>
    #include <bits/stdc++.h>
    #include <iomanip>
    #define ALL(x) (x).begin(), (x).end()
    #define rt return
    #define sz(a) int(a.size())
    #define all(a) a.begin(), a.end()
    #define rep(i,x,n) for(int i=x;i<n;i++)
    #define repd(i,x,n) for(int i=x;i<=n;i++)
    #define pii pair<int,int>
    #define pll pair<long long ,long long>
    #define gbtb ios::sync_with_stdio(false),cin.tie(0),cout.tie(0)
    #define MS0(X) memset((X), 0, sizeof((X)))
    #define MSC0(X) memset((X), '', sizeof((X)))
    #define pb push_back
    #define mp make_pair
    #define fi first
    #define se second
    #define eps 1e-6
    #define gg(x) getInt(&x)
    #define db(x) cout<<"== [ "<<x<<" ] =="<<endl;
    using namespace std;
    typedef long long ll;
    ll gcd(ll a,ll b){return b?gcd(b,a%b):a;}
    ll lcm(ll a,ll b){return a/gcd(a,b)*b;}
    ll powmod(ll a,ll b,ll MOD){ll ans=1;while(b){if(b%2)ans=ans*a%MOD;a=a*a%MOD;b/=2;}return ans;}
    inline void getInt(int* p);
    const int maxn=1000010;
    const int inf=0x3f3f3f3f;
    /*** TEMPLATE CODE * * STARTS HERE ***/
    string S(ll n){stringstream ss;string s;ss<<n;ss>>s;return s;}
    ll N(string s){stringstream ss;ll n;ss<<s;ss>>n;return n;}
    string rm0(string s){//  去除前导0函数
    	int i;
    	for(i=0;i<s.size()-1;i++)
    		if(s[i]!='0')
    			break;
    	return s.substr(i);
    }
    string ADD(string s,string t) {// 字符串整数相加,返回一个字符串
      if(s.size()<t.size())swap(s,t);s='0'+s;
      reverse(t.begin(), t.end());while(s.size()>t.size())t+='0';reverse(t.begin(), t.end());int c=0;
      for(int i=s.size();i>=0;i--)
      {
        if(c){
        	if(s[i]=='9'){
        		s[i]='0';c=1;
        	}else{
        		s[i]=(char)(s[i]+1);c=0;
        	}
        }
        int sum=(int)s[i]+(int)t[i]-'0'*2;
        if(sum>=10){
        	s[i]=(char)(sum-10+'0');c=1;
        }
        else 
        	s[i]=(char)(sum+'0');
      }
      return rm0(s);
    }
    bool cmp(string s,string t) { 
    	// s>=t 返回1
    	// s<t 返回0
    	if(s.size()!=t.size())
      		return s.size()>t.size();
    	for(int i=0;i<s.size();i++)
    		if(s[i]!=t[i])
    			return s[i]>t[i];
      return 1;
    }
    int main()
    {
        //freopen("D:\code\text\input.txt","r",stdin);
    	//freopen("D:\code\text\output.txt","w",stdout);
    	
    	int len;
    	string str;
    	cin>>len>>str;
    	std::vector<int> v;
    	v.clear();
    	rep(i,0,len)
    	{
    		if(str[i]!='0')
    		{
    			v.pb(i);
    		}
    	}
    	int k=lower_bound(ALL(v),len/2)-v.begin();
    	string ans=str;
    	repd(j,-2,3)
    	{
    		int id=max(1,min(sz(v)-1,k+j));
    		string s1=str.substr(0,v[id]);
    		string s2=str.substr(v[id]);
    		string w=ADD(s1,s2);
    		if(cmp(ans,w))
    		{
    			ans=w;
    		}
    	}
    	cout<<ans<<endl;
        return 0;
    }
    
    inline void getInt(int* p) {
        char ch;
        do {
            ch = getchar();
        } while (ch == ' ' || ch == '
    ');
        if (ch == '-') {
            *p = -(getchar() - '0');
            while ((ch = getchar()) >= '0' && ch <= '9') {
                *p = *p * 10 - ch + '0';
            }
        }
        else {
            *p = ch - '0';
            while ((ch = getchar()) >= '0' && ch <= '9') {
                *p = *p * 10 + ch - '0';
            }
        }
    }
    
    
    
    本博客为本人原创,如需转载,请必须声明博客的源地址。 本人博客地址为:www.cnblogs.com/qieqiemin/ 希望所写的文章对您有帮助。
  • 相关阅读:
    从浏览器输入URL到页面渲染的过程
    安全分析的几个好的工具网站的使用
    从一次渗透谈到linux如何反弹shell
    python 进行抓包嗅探
    MYSQL的索引和常见函数
    一篇博客搞定redis基础
    新型横向移动工具原理分析、代码分析、优缺点以及检测方案
    Java反序列化漏洞的挖掘、攻击与防御
    关于Memcached反射型DRDoS攻击分析
    spark未授权RCE漏洞
  • 原文地址:https://www.cnblogs.com/qieqiemin/p/11033007.html
Copyright © 2020-2023  润新知