LuoguP1557 Kruscal的加法 题解
题目Link
就是这道题,做了我整整一天!
看到题目,首先想到的就是:就这?就这一道大水题也能是绿?然后十分钟写完代码,提交……
果不其然,绿题不是白绿的,看了一眼数据和讨论,又得写高精了……
先附上非高精代码:
#include<cstring>
#include<cstdio>
#include<iostream>
#include<cmath>
#define LL long long
using namespace std;
LL now;
LL tim;
char s[10000];
int k;
LL ans=0;
int main()
{
cin>>s;
for(int i=0;i<strlen(s);i)
{
k=1;
if(s[i]=='+')
{
k=1;
if(s[i+1]=='+')
{
tim=0;
while(s[i]=='+')
i++,tim++;i--;
}
i++;
}
if(s[i]=='-')
{
k=0;
if(s[i+1]=='-')
{
tim=0;
while(s[i]=='-')
i++,tim++;i--;
}
i++;
}
if(s[i]=='('){
i++;
tim=0;
while(s[i]>='0'&&s[i]<='9'){
tim=tim*10+s[i]-'0';
i++;
}
i++;
}
now=0;
while(s[i]>='0'&&s[i]<='9'){
now=now*10+s[i]-'0';
i++;
}
if(tim==0) tim=1;
if(k==1){
ans+=tim*now;
}
else{
ans-=tim*now;
}
tim=0;
}
printf("%lld\n",ans);
return 0;
}
从这份代码可以看出,对于输入字符串是以 ++a
或 +(n)a
的类型为一个板块输入并进行运算, tim
存入 +/-
的个数或者括号内的数字,代表乘数, now
代表 +/-
或括号后的数字,即代表被乘数,之后将 tim
与 now
相乘加入 ans
即可。
#include<cstring>
#include<cstdio>
#include<iostream>
#include<cmath>
using namespace std;
char s[20005];
int now[20005];
int t=0;
int tim;
int timm[20005];
int tt=0;
int k;
int ans[20005];
int c[20005];
int nb=0;
void w1(int m)// 加法运算
{
for(int i=1;i<=max(m,ans[0]);i++)
{
ans[i]=ans[i]+c[i];
ans[i+1]+=ans[i]/10;
ans[i]=ans[i]%10;
}ans[0]=max(m,ans[0]);
if(ans[max(m,ans[0])+1]) ans[0]++;
}
void w2(int m)// 减法运算
{
if(ans[0]<m||ans[0]==m&&ans[ans[0]]<c[m]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]<c[m-1]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]==c[m-1]&&ans[ans[0]-2]<c[m-2]) //暴力地判断减数和被减数的大小,下同;
{
ans[10000]=1;
for(int i=1;i<=max(m,ans[0]);i++)
{
c[i]-=ans[i];
if(c[i]<0){
c[i]+=10;
c[i+1]--;
}
}
ans[0]=max(m,ans[0]);
for(int i=1;i<=ans[0];i++){
ans[i]=c[i];
}
}
else
for(int i=1;i<=max(m,ans[0]);i++)
{
ans[i]-=c[i];
if(ans[i]<0){
ans[i]+=10;
ans[i+1]--;
}
}
}
void work1() // +++a的情况
{
memset(c,0,sizeof(c));
int m=t;
for(int i=1;i<=m;i++)
{
c[i]+=now[i]*tim;
c[i+1]+=c[i]/10;
c[i]=c[i]%10;
}
m++;
while(c[m]!=0)
{
c[m+1]=c[m]/10;
c[m]=c[m]%10;
m++;
}m--;
while(c[m]==0) m--;
if(k==1)
{
if(ans[10000]==1){
int flag=0;
if(ans[0]<m||ans[0]==m&&ans[ans[0]]<c[m]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]<c[m-1]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]==c[m-1]&&ans[ans[0]-2]<c[m-2])
flag=1;
w2(m);
if(flag==1) ans[10000]=0;
}
else
w1(m);
}
if(k==0)
{
if(ans[10000]==1){
w1(m);
}
else
w2(m);
}
while(ans[ans[0]]==0) ans[0]--;
}
void work2() // +(n)a 的情况
{
memset(c,0,sizeof(c));
for(int i=1;i<=t;i++)
for(int j=1;j<=tt;j++)
{
c[i+j-1]+=now[i]*timm[j];
}
for(int i=1;i<=t+tt;i++)
{
c[i+1]+=c[i]/10;
c[i]=c[i]%10;
}
int m=t+tt+1;
while(c[m]!=0)
{
c[m+1]=c[m]/10;
c[m]=c[m]%10;
m++;
}m--;
while(c[m]==0) m--;
if(k==1)
{
if(ans[10000]==1){
int flag=0;
if(ans[0]<m||ans[0]==m&&ans[ans[0]]<c[m]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]<c[m-1]||ans[0]==m&&ans[ans[0]]==c[m]&&ans[ans[0]-1]==c[m-1]&&ans[ans[0]-2]<c[m-2])
flag=1;
w2(m);
if(flag==1) ans[10000]=0;
}
else{
w1(m);
}
}
if(k==0)
{
if(ans[10000]==1){
w1(m);
}
else
w2(m);
}
while(ans[ans[0]]==0) ans[0]--;
}
int main()
{
cin>>s;
for(int i=0;i<strlen(s);i)
{
k=1;
tim=0;
if(s[i]=='+') //输入加号及个数
{
k=1;
if(s[i+1]=='+')
{
tim=0;
while(s[i]=='+')
i++,tim++;i--;
}
i++;
}
if(s[i]=='-') //输入减号及个数
{
k=0;
if(s[i+1]=='-')
{
tim=0;
while(s[i]=='-')
i++,tim++;i--; //由于n<=2000,故在这种情况时,tim<=2000,直接用int存即可;
}
i++;
}
tt=0;
if(s[i]=='('){ //输入乘数
i++;
while(s[i]>='0'&&s[i]<='9'){
i++;
}i--;
while(s[i]>='0'&&s[i]<='9'){
timm[++tt]=s[i]-'0';
i--;
}i++;
while(s[i]>='0'&&s[i]<='9'){
i++;
}
i++;
}
while(s[i]>='0'&&s[i]<='9'){ //输入被乘数
i++;
}i--;
t=0;
while(s[i]>='0'&&s[i]<='9'){
now[++t]=s[i]-'0';
i--;
}i++;
while(s[i]>='0'&&s[i]<='9'){
i++;
}
if(tim==0&&tt==0) tim++;
if(tim){ //根据不同输入类型,进行不同运算
work1();
}
else{
work2();
}
}
while(ans[ans[0]]==0) ans[0]--;//ans[0]代表答案长度
if(ans[0]<1) ans[0]=1; //答案等于0时,ans[0]会减为0,进行特判;
if(ans[0]==1&&ans[1]==0){ //答案等于0时,不能输出-0,直接进行特判;
printf("0\n");
return 0;
}
if(ans[10000]==1) printf("-");
for(int i=ans[0];i>=1;i--)
printf("%d",ans[i]);
printf("\n");
return 0;
}
再来看这份AC代码,输入与上面非高精代码相同,把运算换成高精,但多了亿点点细节。
- 首先,要把数字作为一个数组输入,为了保证数组顺序,先让
i
跑到下一个非数字字符,再从后往前跑,把数字存入,然后再让i
跑回去准备下一次的读取; - 其次,高精运算涉及到负数,用
ans[10000]
来存储当前得数的正负,0
代表正,1
代表负,在进行加减运算时,通过判断当前得数的正负,巧妙地把加减运算互相转化; - 最后,要注意输出时,要把得数前多余的
0
删掉,同时不存在-0
,要进行特判;
希望管理通过QwQ
LuoguP1557 Kruscal的加法 题解的更多相关文章
- LuoguB2104 矩阵加法 题解
Content 给定两个 \(n\times m\) 的矩阵 \(A,B\),求 \(C=A+B\). 数据范围:\(1\leqslant n,m\leqslant 100\). Solution 我 ...
- BZOJ5321 & 洛谷4064 & LOJ2274:[JXOI2017]加法——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=5321 https://www.luogu.org/problemnew/show/P4064 ht ...
- Challenge & Growth —— 从这里开始
做有挑战的事情,就从这里开始. 忘记这本书现在在哪儿了,以前还以为能祖祖辈辈留传,现在只能借助 Nowcoder 了.Now coder,Forever thinker. 想以自以为最优美的 code ...
- PAT(乙级)2020年春季考试
比赛链接:https://pintia.cn/market/item/1287964475579875328 7-1 对称日 题解 模拟,注意年月日不足位在前面补零. 代码 #include < ...
- 【题解】洛谷P1967 [NOIP2013TG] 货车运输(LCA+kruscal重构树)
洛谷P1967:https://www.luogu.org/problemnew/show/P1967 思路 感觉2013年D1T3并不是非常难 但是蒟蒻还是WA了一次 从题目描述中看出每个点之间有许 ...
- 力扣(LeetCode)整数形式的整数加法 个人题解
对于非负整数 X 而言,X 的数组形式是每位数字按从左到右的顺序形成的数组.例如,如果 X = 1231,那么其数组形式为 [1,2,3,1]. 给定非负整数 X 的数组形式 A,返回整数 X+K 的 ...
- PAT甲题题解-1023. Have Fun with Numbers (20)-大数加法
和1024一样都是大数据的题,因为位数最多要20位,long long最多19位给一个num,求sum=num+num问sum包含的数字,是否是num的一个排列,即数字都一样,只是顺序不同罢了. #i ...
- UNR #1 题解
A. 争夺圣杯 还是想说一下,这题是原题啊...想做的人可以戳codechef上的MTMXSUM(懒得贴链接了,套了个壳,不过正常人应该都能看得出来) 显然异或输出没什么奇怪的性质... 考虑一个元素 ...
- JSOI Round 2题解
强行一波题解骗一个访问量好了... http://blog.csdn.net/yanqval/article/details/51457302 http://absi2011.is-programme ...
随机推荐
- 【nodejs】request 和 response 对象
request 和 response 对象的具体介绍: Request 对象 - request 对象表示 HTTP 请求,包含了请求查询字符串,参数,内容,HTTP 头部等属性.常见属性有: req ...
- request库的简单使用
7种基本用法 首先安装requests库 如果你本地只有一个python环境直接运行pip install requests 像我本地装了3个不同的python怎么使你安装在你想要安装的python下 ...
- mzy git学习,初识git(一)
GIT学习 git工作区.暂存区.本地库.远程库 工作区:实际上我们工作的地方,进行写代码或者文件的地方. 暂存区:我们执行了git add 操作之后,就会被提交到暂存区. 本地库:其实最后我们需要执 ...
- 对于Oracle、mysql和sql server中的部分不同理解
1.在mysql中事务默认是自动提交的,只有设置autocommit为0的时候,才用自己commit:(提到commit不要忘了rollback哦,回滚)2.但是在oracle中必须自己commit: ...
- ES读写数据的工作原理
es写入数据的工作原理是什么啊?es查询数据的工作原理是什么?底层的lucence介绍一下呗?倒排索引了解吗? 一.es写数据过程 1.客户端选择一个node发送请求过去,这个node就是coordi ...
- 前端 | 页面触底自动加载 Vue 组件
不管是 web 端还是移动端,信息流都是现在很流行的信息展示方式.信息流经常搭配自动加载一起使用以获得更好的使用体验. 最近在使用 Vue 开发过程中也遇到了首页信息流自动加载的需求.大致了解了一下几 ...
- spring支持的Bean的作用域
Sigleton:单例模式,在整个Spring IoC容器中,使用Sigleton定义Bean将有一个实例 prototype:原型模式,每次通过容器的getBean方法获取propertype都将产 ...
- go语言内存对齐
内存对齐 为保证程序顺利高效的运行,编译器会把各种类型的数据安排到合适的地址并占用合适的长度,这就是内存对齐 每种类型的对齐值就是他的内存边界 64位 类型 对齐边界 (对齐值) int8 1byte ...
- Java的Class类及static块的执行时机
要理解RTTI在Java中的工作原理,首先必须知道类型信息在运行时是如何表示的,这项工程由Class对象完成,它包含了与类有关的信息.Java使用Class对象来执行其RTTI,即使你执行的是类似转型 ...
- 搭建私服仓库:(一)Windows安装Nuxus
Nexus下载 官网.官网下载.百度云盘 提取码:su33 将nexus下载下来,以2.14.5的windows版本为例子(3.x暂时下载不下来,迅雷会员都不行) 下载后进行解压,得到以下目录: 其中 ...