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 ...
随机推荐
- 怎样在Qt中建立使用动态链接库
参考网址: https://blog.csdn.net/q496713258/article/details/6990837 qt 的学习网址: http://c.biancheng.net/view ...
- 【springcloud】API Gateway 的路由和过滤(Zuul--1)
转自:https://blog.csdn.net/pengjunlee/article/details/87084646 Zuul是什么? API Gateway 是随着微服务(Microservic ...
- java输入字符
1.创建一个Scanner对象. 2.调用.next()返回一个String类型用一个变量接受. 3.调用该String变量的.charAt(0),获取第一个字符. Scanner scn=new S ...
- 基于mysql和Java Swing的简单课程设计
摘要 现代化的酒店组织庞大.服务项目多.信息量大.要想提高效率.降低成本.提高服务质量和管理水平,进而促进经济效益,必须利用电脑网络技术处理宾馆酒店经营数据,实现酒店现代化的信息管理.本次课程设计运用 ...
- 基于 Mysql 实现一个简易版搜索引擎
前言 前段时间,因为项目需求,需要根据关键词搜索聊天记录,这不就是一个搜索引擎的功能吗? 于是我第一时间想到的就是 ElasticSearch 分布式搜索引擎,但是由于一些原因,公司的服务器资源比较紧 ...
- Redis详解(二)——
https://www.cnblogs.com/yeya/p/14274948.html https://www.cnblogs.com/liang24/tag/redis/
- WEB漏洞——XSS
跨站脚本( Cross-site Scripting,简称为XSS或跨站脚本或跨站脚本攻击)是一种针对网站应用程序的安全漏洞攻击技术,是代码注入的一种. XSS攻击可以分为三种:反射型.存储型和DOM ...
- Python - 面向对象编程 - 实战(4)
需求:士兵突进 士兵许三多有一把 AK47 士兵可以开火 枪能够发射子弹 枪装填子弹,可以增加子弹数量 需求分析 很明显有两个类:士兵类,枪类 AK47 是枪名,是枪类的属性,每把枪都有子弹数,所以子 ...
- linux清空文件
https://www.cnblogs.com/mrwang1101/p/6166326.html
- 随机生成uuid序号
function guid() { function S4() { return (((1+Math.random())*0x10000)|0).toString(16).substring(1); ...