【poj】1001
【题目】
Exponentiation
Time Limit: 500MS Memory Limit: 10000K
Total Submissions: 123707 Accepted: 30202
Description
Problems involving the computation of exact values of very large magnitude and precision are common. For example, the computation of the national debt is a taxing experience for many computer systems.
This problem requires that you write a program to compute the exact value of Rn where R is a real number ( 0.0 < R < 99.999 ) and n is an integer such that 0 < n <= 25.
Input
The input will consist of a set of pairs of values for R and n. The R value will occupy columns 1 through 6, and the n value will be in columns 8 and 9.
Output
The output will consist of one line for each line of input giving the exact value of R^n. Leading zeros should be suppressed in the output. Insignificant trailing zeros must not be printed. Don't print the decimal point if the result is an integer.
Sample Input
95.123 12
0.4321 20
5.1234 15
6.7592 9
98.999 10
1.0100 12
Sample Output
548815620517731830194541.899025343415715973535967221869852721
.00000005148554641076956121994511276767154838481760200726351203835429763013462401
43992025569.928573701266488041146654993318703707511666295476720493953024
29448126.764121021618164430206909037173276672
90429072743629540498.107596019456651774561044010001
1.126825030131969720661201
Hint
If you don't know how to determine wheather encounted the end of input:
s is a string and n is an integer
C++
while(cin>>s>>n)
{
...
}
c
while(scanf("%s%d",s,&n)==2) //to see if the scanf read in as many items as you want
/*while(scanf(%s%d",s,&n)!=EOF) //this also work */
{
...
}
----------------------------------------------------------------------------------------------------------------------------------------------------------
这道题是我真正意义上的第一道题。所以写的很乱。估计再过上一段时间后连自己都看不懂。
大概思路:因为没有任何一种数据类型能够容纳这种高精度运算的结果。所以用数组来储存数据。求幂的实质就是n-1次乘法运算。运算之前把对字符串进行处理(提取小数点,同时记录指数),转化为整形数组。值得注意的是,要在整形数组后面开始储存。比如储存1234,是a[MAX-1]=4;a[MAX-2]=3;..这样有助于乘法运算的时候向前进一位。运算过程结束后再转化为字符串。重新加入小数点。
这一题做了很久很久,走了不少弯路,也就收货了一些经验:
1.代码的书写很重要。最好能形成层层缩进的结构。这样美观大方修改起来十分方便。
2.在写代码之前最好能够先详细构思一下。也就是所谓的建模。不要一有idea就马上动笔写下来。这次我就是很快把核心部分写了下来,却因为没有一个完整的架构。修修补补花了大量的时间。
3.审题。一开始看错题目、、、、满满的都是泪啊。
4.测试的时候没有进行全面的考虑。后期引用了这位兄弟的测试数据才ac。http://www.cnblogs.com/lililqth/p/3187758.html
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#define MAX 500
int pows(char [],char [],int ,int []); //高精度计算pow
char *deal(char [],int ,int ); //处理计算好的字符串。
int find(char []); //寻找小数点所在位置,
int cf(int [],int ,int [],int ); //高精度乘法
int main(){
char ins[MAX],des[MAX],*res;
int n,index,len=;
int temp[MAX];
while(scanf("%s%d",ins,&n)!=EOF)
{ index=find(ins);
memset(temp,,*MAX);
len=pows(ins,des,n,temp);
if(len!=)
{
res=deal(des,len,temp[MAX-]); //传递指数
des[MAX-]='\0'; //给倒数第二位加上字符串结束符
printf("%s\n",res);
}
else return ; }
}
int find(char des[MAX])
{
int i=,index=-;
for(i=;i<=strlen(des);i++)
if(des[i]=='.'){
index=i;
return index;}
return -; //默认没有找到小数点
}
char *deal(char des[],int len,int en){
int i,ts=,zhi=,k=;
//重新加入小数点
zhi=MAX--(-en); //小数点位置
if((-en)==len) { //-,小数点在刚好在第一位
des[zhi]='.';
return &des[zhi]; }
else if((-en)>len) { //二,数值小于1的情况,0.0xx;
for(i=zhi;i<MAX--len;i++)
des[i]='';
des[zhi]='.';
return &des[zhi]; }
else if(en==) //三,若此为整数,则直接返回字符串值
return &des[MAX--len];
else
{ //四,数值大于1的情况
for(i=MAX--len;i<=zhi;i++)
des[i-]=des[i]; //将小数点所在位置的前方往左挪一位。
des[zhi]='.';
return &des[MAX--len];
}
} int pows(char ins[MAX],char des[MAX],int n,int temp[MAX])
{
int t2[MAX]={};
int i=,index=-,k=,m=,ji=,h=,kj=;
int len=strlen(ins);
index=find(ins);
m=len;
if(n==)
{ //n=0
for(i=;i<=len-;i++)
if(ins[i]!=''){ //这个数不等于0
temp[MAX-]=;
temp[MAX-]=;
break; }
else return ; //不可以计算0的0次幂
}
else if(n==)
{ //n=1
for(i=;i<=len-;i++) //从第一个位置开始遍历,检查前面是否有无意义的零
if(ins[i]!='')
break;
if(i==len)
{ //整个字符串全为0
m=;
temp[MAX-]=; //表明这是一个整数
}
else
{ //前面有无意义的0,此处包含无0的情况
m=len-i-;
//复制小数点之后的
for(kj=MAX-,h=len-;h>index;h--,kj--)
temp[kj]=ins[h]-;
temp[MAX-]=-(len-index-); //记录指数
//复制小数点之前的
for(k=h-;k>=i;kj--,k--)
temp[kj]=ins[k]-;
}
for(i=len-,k=;i>=;i--) { //从后面开始扫描有没有无意义的0
if(ins[i]!='')
break;
k++; } //计算有多少个0
if(i<len-)
{ //后面有无意义的0
if(ins[i]=='.')
{ //1.00样式
m -=k;
for(h=MAX-;h>=MAX--m;h--)
temp[h]=temp[h-k];
temp[MAX-]=; //标记为整数
}
else
{ //1.**00的样式
m -=k;
for(h=MAX-;h>=MAX--m;h--)
temp[h]=temp[h-k];
temp[MAX-] +=k; //修改指数
}
}
}
else
{ //n>=2
if(index==-)
{ //一。整数,没有找到小数点
for(i=;i<=len-;i++) {
temp[MAX--len+i]=ins[i]-;
t2[MAX--len+i]=ins[i]-;
}
}
else
{ //二找到小数点
if(index==)
{ //处理形式为 。**的形式的实数
for(i=len;i>=;i--)
ins[i]=ins[i-];
ins[]='';
len++;
}
i=len-;
while(ins[i]=='') i--; //从最后一个字符开始检查,检查最后是否带有无意义的0
if(i<len-)
{ //******后面带有无意义的0 if(ins[i]=='.')
{ //-----1.00的形式
len=i; //连同小数点一起抛弃
index=-; //将该数标记为整数
for(i=;i<=len-;i++)
{ //复制到整形数组中
temp[MAX--len+i]=ins[i]-;
t2[MAX--len+i]=ins[i]-;
}
}
else
{ //------1.**000的形式
len=i+;
for(i=;i<=index-;i++) { //用两个for循环将字符串转换为整形数组,并且要将从右边开始写入
temp[MAX--len+i]=ins[i]-; //从整形数组的倒数第三位开始储存
t2[MAX--len+i]=ins[i]-; }
//跳过小数点
for(k=index+;k<=len-;k++,i++){
temp[MAX--len+i]=ins[k]-;
t2[MAX--len+i]=ins[k]-; }
}
}
else
{ //******后面没有无意义的0 for(i=;i<=index-;i++) { //用两个for循环将字符串转换为整形数组,并且要将从右边开始写入
temp[MAX--len+i]=ins[i]-; //从整形数组的倒数第三位开始储存
t2[MAX--len+i]=ins[i]-;} for(k=index+;k<=len-;k++,i++) { //跳过小数点
temp[MAX--len+i]=ins[k]-;
t2[MAX--len+i]=ins[k]-; }
}
}
//在数组的最后一位记录这两个数的指数
if(index==-)
{ //如果为整数则将该指数部分置为0
temp[MAX-]=;
t2[MAX-]=;
k=len;
m=len;
}
else
{ //不为整数
temp[MAX-]=index-len+;
t2[MAX-]=index-len+;
k=len-;
m=len-;
}
for(i=;i<n-;i++) //pow的核心
{ //需要n-1次乘法
cf(temp,m,t2,k); //cf返回temp数组中所含数的长度
for(ji=;ji<=MAX-;ji++)
{ //计算temp中的数的长度
if(temp[ji]!=){
m=MAX-ji-;
break; }
}
}
}
//重新将字符转换为数值
for(i=MAX-;i>=MAX--m+;i--) //从整形数组倒数第三位开始复制到字符串的倒数第三位
des[i]=temp[i]+;
return m; //返回字符串的长度
}
int cf(int temp[MAX],int m,int src[MAX],int k){
int t[MAX]={};
int i=,num=;
int sr=,tmp=MAX-,j=sr;
temp[MAX-]=; //防止k在边界时溢出.在倒数第二位储存0
src[MAX-]=;
// 用两个for循环来完成两个数的相乘
for(sr=MAX-;sr>=MAX-m-;sr--) //数组的倒数第三位储存数的最低位
{
for(tmp=MAX-,j=sr;tmp>=MAX--m;tmp--,j--) //src从个位开始每位乘以temp中的每一位 //之所以使j=sr--我们在计算两数相乘时总会将下一行的数向左挪一位。
t[j] +=src[sr]*temp[tmp];
}
//将t复制到temp中
for(i=MAX-;i>=sr-m+;i--)
temp[i]=t[i];
for(i=MAX-;i>=sr-m;i--)
{ // 处理temp中的数,过十进一,每一位只保留个位,需要注意的是有可能是进百位。故要把i提前到temp左端的前两格
temp[i] +=num;
num= (temp[i]-temp[i]%)/; //保存当前一位的十位数和百位数.如128保留12;
temp[i]=temp[i]%; //进位后,只保存个位数
}
temp[MAX-] +=src[MAX-]; //处理指数部分的相乘
return ;
}
【poj】1001的更多相关文章
- 【POJ】1704 Georgia and Bob(Staircase Nim)
Description Georgia and Bob decide to play a self-invented game. They draw a row of grids on paper, ...
- 【POJ】1067 取石子游戏(博弈论)
Description 有两堆石子,数量任意,可以不同.游戏开始由两个人轮流取石子.游戏规定,每次有两种不同的取法,一是可以在任意的一堆中取走任意多的石子:二是可以在两堆中同时取走相同数量的石子.最后 ...
- 【BZOJ】【1986】【USACO 2004 Dec】/【POJ】【2373】划区灌溉
DP/单调队列优化 首先不考虑奶牛的喜欢区间,dp方程当然是比较显然的:$ f[i]=min(f[k])+1,i-2*b \leq k \leq i-2*a $ 当然这里的$i$和$k$都是偶数啦~ ...
- 【POJ】【2104】区间第K大
可持久化线段树 可持久化线段树是一种神奇的数据结构,它跟我们原来常用的线段树不同,它每次更新是不更改原来数据的,而是新开节点,维护它的历史版本,实现“可持久化”.(当然视情况也会有需要修改的时候) 可 ...
- 【POJ】1222 EXTENDED LIGHTS OUT
[算法]高斯消元 [题解] 高斯消元经典题型:异或方程组 poj 1222 高斯消元详解 异或相当于相加后mod2 异或方程组就是把加减消元全部改为异或. 异或性质:00 11为假,01 10为真.与 ...
- 【POJ】2892 Tunnel Warfare
[算法]平衡树(treap) [题解]treap知识见数据结构 在POJ把语言从G++换成C++就过了……??? #include<cstdio> #include<algorith ...
- 【POJ】【1637】Sightseeing tour
网络流/最大流 愚人节快乐XD 这题是给一个混合图(既有有向边又有无向边),让你判断是否有欧拉回路…… 我们知道如果一个[连通]图中每个节点都满足[入度=出度]那么就一定有欧拉回路…… 那么每条边都可 ...
- 【POJ】3070 Fibonacci
[算法]矩阵快速幂 [题解] 根据f[n]=f[n-1]+f[n-2],可以构造递推矩阵: $$\begin{vmatrix}1 & 1\\ 1 & 0\end{vmatrix} \t ...
- 【POJ】3233 Matrix Power Series
[算法]二分+矩阵快速幂 [题意]给定矩阵A和整数k,MOD,求A^0+A^1+A^2+...+A^k. [题解] 定义题目要求的答案为f(n),即: $$f_n=\sum_{i=0}^{n}A^i$ ...
随机推荐
- Webservice-Java-Xfire
最近公司最近需要将以前提供出去的接口统一用一个标准来实现,考虑到webservice这个是标 准,因此我花时间大概学习了一下webservice,也对JAVA的几个webservice框架进行了一些小 ...
- HTML5新增的一些属性和功能之六——拖拽事件
拖放事件的前提是分为源对象和目标对象,你鼠标拖着的是源对象,你要放置的位置是目标对象,区分这两个对象是因为HTML5的拖放事件对两者是不同的. 被拖动的源对象可以触发的事件: 1).ondragsta ...
- 数学之路(3)-机器学习(3)-机器学习算法-PCA
PCA 主成分分析(Principal components analysis,PCA),维基百科给出一个较容易理解的定义:“PCA是一个正交化线性变换,把数据变换到一个新的坐标系统中,使得这一数据的 ...
- sublime怎么实现函数之间的跳转
1.安装ctags应用程序. 到CTags的官方站点下载最新版本号,将解压后的ctags.exe放到系统环境变量的搜索路径中.通常是C:\windows\system32. 假设你想放到其它目录中,记 ...
- PHP5生成图形验证码(有汉字)
利用PHP5中GD库生成图形验证码 类似于下面这样 1.利用GD库函数生成图片,并在图片上写指定字符 imagecreatetruecolor 新建一个真彩色图像 imagecolora ...
- javascriptt切换组件MyTab.js封装
之前做的大多数是jquery的插件,就优雅性来说,我觉得还是原生的代码,写起来更舒服一点,虽然麻烦很多. 之前写了一个利用完美运动框架的轮播效果,因为使用的是原生的代码,因为不懂原生对象封装的原因一直 ...
- JS传递参数时对中文进行编码和解码
var b ="啊,我要过去"; var a = encodeURI(b);//对中文编码 ...
- IIS 7 支持10万并发请求
原文链接:http://www.cnblogs.com/dudu/archive/2009/11/10/1600062.html 今天下午17点左右,博客园博客站点出现这样的错误信息: Error S ...
- CC开发问题一
CC编译成功,启动失败,debug状态下报错如下,未能加载文件或程序集 这个问题查了一些资料,http://blog.csdn.net/shellching/article/details/82947 ...
- 轻量级的中文分词工具包 - IK Analyzer
IK Analyzer是一个开源的,基于java语言开发的轻量级的中文分词工具包.从2006年12月推出1.0版开始, IKAnalyzer已经推出了4个大版本.最初,它是以开源项目Luence为应用 ...