说起这道题, 真是一把辛酸泪.

题意

将一个正整数 $n(\le 10^{1000})$ 分解成不超过50个回文数的和.

做法

构造.

队友UHC提出的一种构造方法, 写起来比较方便一些, 而且比较巧妙. 可惜我码力太弱, 现场没调出来.

大体的想法是:

将一个数"等"分成两半. 如果长度为奇数可以采取下面两种处理方法之一.

  1. 在开头补0.
  2. 后一半 (低位) 的长度向下取整.

我采取第2种处理方式.

设前一半的长度为$h_1$, 后一半的长度为 $h_2$.

然后将前一半长

(写不下去了..., 贴代码...)

Implementation

#include <bits/stdc++.h>
using namespace std; using num=vector<int>;
using iter=num::iterator;
using riter=num::reverse_iterator; bool Less(const num &a, const num &b){ // a and b are of same length
for(auto it1=a.rbegin(), it2=b.rbegin(); it1!=a.rend(); ++it1, ++it2)
if(*it1 != *it2){
return *it1 < *it2;
}
return false;
} void borrow(num &a){
for(int i=a.size()/2; i<a.size(); i++)
if(a[i]){
for(--a[i]; i; a[--i]+=9); //?
break;
}
while(a.size() && *a.rbegin()==0)
a.pop_back();
} void split(num &lo, num &hi, const num &a){
auto mid=a.begin()+a.size()/2;
lo={a.begin(), mid};
hi={mid, a.end()};
} num operator-(num a, num b){ // a and b are of same length && a >= b
num res;
for(int i=0; i<a.size(); i++){
if(a[i]<b[i]){
a[i]+=10;
a[i+1]-=1;
}
res.push_back(a[i]-b[i]);
} for(; res.size() && *res.rbegin()==0; )
res.pop_back(); if(res.size()){ //error-prone
for(int i=0; i<res.size()-1; i++){
res[i+1]+=res[i]/10;
res[i]%=10;
} for(; *res.rbegin()>9; ){
int t=*res.rbegin();
*res.rbegin()%=10;
res.push_back(t/10);
}
} for(auto x:res){
assert(x>=0 && x<10);
} return res;
} vector<num> res; void out(const num &a){
for(auto it=a.rbegin(); it!=a.rend(); ++it)
cout<<*it;
cout<<endl;
} void solve(num &a){
res.clear();
num lo, hi;
int ones=0; while(a.size()>1){
// out(a);
split(lo, hi, a);
num rhi(hi.rbegin(), hi.rbegin()+lo.size());
if(Less(lo, rhi)){
borrow(a);
++ones;
continue;
}
a=lo-rhi;
// There is no advantage to using {} initialization, and one trap,
// when using auto to get the type determined by the initializer.
// The trap is that if the initializer is a {}-list, we may not want its
// type deduced.
//initializer_list<num> xx={rhi};
//cout<<typeid(xx).name()<<endl;
auto tmp=rhi;
//cout<<typeid(tmp).name()<<endl;
for(auto &x: hi)
tmp.push_back(x);
res.push_back(tmp);
} if(a.size()) res.push_back(a); assert(res.size() + ones <= 50); printf("%d\n", res.size()+ones);
for(auto &x: res)
out(x);
while(ones--)
puts("1");
} char s[1<<10]; int main(){ // num x, y;
// in g++:
// direct-list-intialization of 'auto' requires exactly one element
// for deduction to 'std::initializer_list', use copy-list-intialization (i.e. add '=' before the '{')
// auto xx{x, y}; int T, cas=0;
num a;
for(cin>>T; T--; ){
scanf("%s", s);
a.clear();
int n=strlen(s);
for(int i=n-1; i>=0; --i)
a.push_back(s[i]-'0');
printf("Case #%d:\n", ++cas);
solve(a);
}
return 0;
}

HDU 5920 Ugly Problem的更多相关文章

  1. HDU 5920 Ugly Problem 【模拟】 (2016中国大学生程序设计竞赛(长春))

    Ugly Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Tota ...

  2. HDU - 5920 Ugly Problem 求解第一个小于n的回文数

    http://acm.hdu.edu.cn/showproblem.php?pid=5920 http://www.cnblogs.com/xudong-bupt/p/4015226.html 把前半 ...

  3. HDU 5920 Ugly Problem 高精度减法大模拟 ---2016CCPC长春区域现场赛

    题目链接 题意:给定一个很大的数,把他们分为数个回文数的和,分的个数不超过50个,输出个数并输出每个数,special judge. 题解:现场赛的时候很快想出来了思路,把这个数从中间分为两部分,当位 ...

  4. D - Ugly Problem HDU - 5920

    D - Ugly Problem HDU - 5920 Everyone hates ugly problems. You are given a positive integer. You must ...

  5. hdu-5920 Ugly Problem(贪心+高精度)

    题目链接: Ugly Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Other ...

  6. HDU 3549 Flow Problem(最大流)

    HDU 3549 Flow Problem(最大流) Time Limit: 5000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/ ...

  7. hdu 5106 Bits Problem(数位dp)

    题目链接:hdu 5106 Bits Problem 题目大意:给定n和r,要求算出[0,r)之间全部n-onebit数的和. 解题思路:数位dp,一个ct表示个数,dp表示和,然后就剩下普通的数位d ...

  8. Ugly Problem

    Ugly Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Spec ...

  9. HDU 3374 String Problem (KMP+最大最小表示)

    HDU 3374 String Problem (KMP+最大最小表示) String Problem Time Limit: 2000/1000 MS (Java/Others)    Memory ...

随机推荐

  1. 让你彻底理解 “==”与 Equals

    相信很多朋友在面对,对象判等时经常会犹豫是用“==”还是Equals呢?有时候发现两者得到的结果相同,但有时候有不同, 究竟在什么情况下"==" 会相等,什么情况下Equals会不 ...

  2. C#开发Windows服务

    Microsoft Windows 服务(即,以前的 NT 服务)使您能够创建在它们自己的 Windows 会话中可长时间运行的可执行应用程序. 服务可以在计算机启动时自动启动,可以暂停和重新启动而且 ...

  3. UWP 拉勾客户端

    前些天, 用 Xamarin.Forms (XF) 将就着写了个拉勾的 UWP 和 Android 的客户端. XF 对 Android  和 IOS 的支持做的很到位, 但是对 UWP 的支持目前仅 ...

  4. Hibernate之Annotation(注解的方式,非映射)

    在hibernate 3.0之后,可以建立一个符合JPA标准的Annotation,以hibernate3.3.2GA为例 Annotation 以 hibernate Annotation 3.3. ...

  5. jQuery能做些什么

    来源于: Learning jQuery, 4th Edition What jQuery does: 1. Access elements in a document; $('div.content ...

  6. 哈希 poj 3274

    n个牛 二进制最多k位 给你n个数 求max(j-i)&&对应二进制位的和相同 7    1  1  1  倒的 6    0  1  1 7    1  1  1 2    0  1 ...

  7. shell实现两个数的相加

    刚开始的时候写,一直写不对:看似简单的功能,但是一定要小心:函数的定义: funciton functionName {.....}在functionName和{之间一定有空格啊! 我就是没加空格,就 ...

  8. NPOI导出模板样式

    /// <summary> /// 导出多种车辆统计表格 /// </summary> /// <returns></returns> [ActionN ...

  9. 「个人vim插件+配置」

    2016.10.4 filetype indent on syntax on set nu ai ci si set sw= ts= set autochdir set backspace= colo ...

  10. matplotlib 和 pandas 两个包的安装

    matplotlib是强大的python 绘图包.pandas 是强大的python分析工具包.numpy是强大的python统计包. 都超级好用,而且最近开始动手实践机器学习算法了.特此备注一下安装 ...