zoukankan      html  css  js  c++  java
  • 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/ 希望所写的文章对您有帮助。
  • 相关阅读:
    常用正则表达式应用
    iOS 通知推送APNS
    IOS生活圈开发练习记录
    [转]定时器
    JavaScript如何在webView跳转到指点的位置
    用imageNamed加载图片产生的问题
    如何再分类中添加属性
    CADisplayLink定时器
    自定义UIBtton类后,在@implementation重写方法的作用
    CAAnimationGroup(组合动画)创建
  • 原文地址:https://www.cnblogs.com/qieqiemin/p/11033007.html
Copyright © 2011-2022 走看看