浮点数(UVa11809)题解

如题


计算机常用阶码-尾数的形式保存浮点数。如下所示,若阶码有6位,尾数有8位,可以表达的最大的浮点数为0.1111111112 * 2 ^ 1111112。注意小数点后一位必须有1,所以一共有9位小数。

0111111110111111

这个数换算成十进制之后就是0.998046875 * 2^63 = 9.205357638345294 * 10^18。你的任务是根据这个最大浮点数,求出阶码的位数E和尾数的位数M。输入格式AeB,表示最大浮点数A * 10^B。0 < A < 10,并且恰好包括15位有效数字。输入结束标志为0e0。对于每组数据,输出ME。输入保证有为一街,且0 <= M <= 9,1 <= E <= 30.


​ 该题的输入AeB十分奇怪,数与数之间插入了字母,那么我们选择先用字符数组读入,再进行处理:

char n[MAXN];
double a,b;
fgets(n,n + 60,stdin);//注意,fgets会读入最后的换行符,处理需谨慎
int p = 0;
while(n[p] != 'e' && n[p] != '\0')p++;
a[p] = ' ';
sscanf(n,"%lf%lf",a,b);//将a,b从n里面读入(e已被换成空格),同样,也有sprintf

​ 既然是关于二进制转换的,那么首先就写一个二进制转换代码。但是:对于每一个二进制小数,有一个唯一确定的十进制小数,但对于一个十进制小数,其算出的二进制小数一般都具有误差!!(具体参见十进制转二进制计算方法)

​ 因此,从十进制推二进制的路就走不通了。鉴于此题ME的取值范围不算太大,那么,尝试二进制推十进制?

基本思路:

​ 枚举ME的大小,即尾数的尾数和阶码的位数。通过这样,我们就可以算出对于每一对ME所唯一确定的十进制最大小数,进而与AeB进行比较,选择差距最小的进行输出。

ME的枚举代码如下:

double m[10] = {0.5},k = 0.5;//当m = 0的时候,其实仍有一位小数(参见题目),因此m[0] = 0.5
double e[40] =
for(int i = 1; i <= 9; i++){
m[i] = m[i - 1] + k / 2;//二进制小数化十进制
k /= 2;
}
int c = 2;
for(int i = 1; i <= 30; i++){
e[i] = c - 1;
c *= 2;
}

​ 乍看上去,似乎是一个不错的代码。但是忽视了一个问题,E最大为30,那么就会乘上2(230),二的三十次方约为10亿,该数的大小就不用赘述了。因此,这样的思路肯定行不通。那么稍作改进!!!

​ 由题意得:


A * 10B = M*2E


那么两边同时取以10为底的对数,仍相等,并可以显著减小数据量级,即:


log10A * B = log10M * E*log102


对于每个M,E进行如是处理:

double m[10] = {0.5},k = 0.5;
double e[40] =
for(int i = 1; i <= 9; i++){
m[i] = m[i - 1] + k / 2;
k /= 2;
}
for(int i = 0; i <= 9; i++){
m[i] = log10(m[i]);
}
int c = 2;
for(int i = 1; i <= 30; i++){
e[i] = (c - 1) * log10(2);//此步认真理解
c *= 2;
}

如此过后,再进行比对,记录答案

全代码:

#include<bits/stdc++.h>
using namespace std;
const int MAXN = 60;
char s[MAXN];
double ans[MAXN][MAXN];
double a;
double b;
void in(){
double m[15] = {0.5},e[30];
long long k = 2;
double t = 0.5;
for(int i = 1; i <= 9; i++){
m[i] = m[i - 1] + t / 2;
//cout << m[i] << " ";
t /= 2;
}
//cout << endl;
for(int i = 0; i <= 9; i++){
m[i] = log10(m[i]);
}
for(int i = 1; i <= 30; i++){
e[i] = (k - 1) * log10(2);
k <<= 1;
//printf("%lf\n",e[i]);
}
for(int i = 0;i <= 9; i++){
for(int j = 1; j <= 30; j++){
ans[i][j] = m[i] + e[j];
}
}
}
int main(){
in();
while(fgets(s,MAXN,stdin)){
if(strncmp(s,"0e0",3) == 0)break;//strncmp可以选择比对前多少位
int p = 0;
while(s[p] != 'e' && s[p] != '\0')p++;
s[p] = ' ';
sscanf(s,"%lf %lf",&a,&b);
b = a = log10(a) + b;//对原数取对数
int pi = 0,pj = 1;
for(int i = 0; i <= 9; i++){
for(int j = 1; j <= 30; j++){
if(fabs(a - ans[i][j]) < b){
b = fabs(a - ans[i][j]);
pi = i;
pj = j; }
}
}
cout << pi << " " << pj << endl;
}
}

(终)

浮点数(UVa11809)题解的更多相关文章

  1. FJUT寒假第一周作业浮点数查寻题解

    二分强化——浮点数序列查询 TimeLimit:4000MS  MemoryLimit:128MB 64-bit integer IO format:%I64d Problem Description ...

  2. 算法习题---3.12浮点数(UVa11809)

    一:题目 尴尬的非会员水印 二:题目摘要 1.int和float比较 int共32位,可以表示的最大的数为2^32次方 float虽然也是32位,但是是以指数形式保存,指数占8位(含符号),最大127 ...

  3. hdu2552 (浮点数复杂运算的四舍五入)题解

    三足鼎立 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

  4. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  5. bestcoder Round #7 前三题题解

    BestCoder Round #7 Start Time : 2014-08-31 19:00:00    End Time : 2014-08-31 21:00:00Contest Type : ...

  6. codevs 1203 判断浮点数是否相等

    http://codevs.cn/problem/1203/ 1203 判断浮点数是否相等  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 青铜 Bronze 题解  查看运行 ...

  7. [POJ] #1004# Financial Management : 浮点数运算

    一. 题目 Financial Management Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 173910   Acc ...

  8. [POJ] #1003# Hangover : 浮点数运算

    一. 题目 Hangover Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 116593   Accepted: 56886 ...

  9. [九度OJ]1137.浮点数加法

    原题链接:http://ac.jobdu.com/problem.php?pid=1137 题目描述: 求2个浮点数相加的和题目中输入输出中出现浮点数都有如下的形式:P1P2...Pi.Q1Q2... ...

随机推荐

  1. Response.Write中文乱码问题

    接手别人的一个ASP项目,功能是页面按钮下载Excel导出数据. 每次导出某一天的数据会出现excel中文乱码,其他天又没问题,因为数据量比较大,所以没有逐条去检查. 找了一些资料 https://w ...

  2. 【HarmonyOS学习笔记】Slider组件实现图形可调旋转

    哈喽大家好我是厚脸皮的小威 之前刚刚用华为的IDE跑通"HELLO,WORLD" 趁热又想去试试看跑一下基于TS拓展API接口的Slider组件,去实现图片的放大和缩小 凭借着大学 ...

  3. zabbix5.0报错PHP时区未设置(配置参数"date.timezone")

    解决办法 : #1.编辑文件/etc/opt/rh/rh-php72/php-fpm.d/zabbix.conf,取消注释并设置为所在地时区 vim /etc/opt/rh/rh-php72/php- ...

  4. 905. Sort Array By Parity - LeetCode

    Question 905. Sort Array By Parity Solution 题目大意:数组排序,偶数放前,奇数在后,偶数的数之间不用管顺序,奇数的数之间也不用管顺序 思路:建两个list, ...

  5. MongoDB启动报错:Unrecognized option: storage try 'mongod --help' for more information(已解决)

    问题说明: 今天在使用配置文件方式启动MongoDB时,一直启动失败,报错显示:Unrecognized option: storage try 'mongod --help' for more in ...

  6. TKE qGPU 通过 CRD 管理集群 GPU 卡资源

    作者 刘旭,腾讯云高级工程师,专注容器云原生领域,有多年大规模 Kubernetes 集群管理经验,现负责腾讯云 GPU 容器的研发工作. 背景 目前 TKE 已提供基于 qGPU 的算力/显存强隔离 ...

  7. DS18B20数字温度计 (三) 1-WIRE总线 ROM搜索算法和实际测试

    目录 DS18B20数字温度计 (一) 电气特性, 寄生供电模式和远距离接线 DS18B20数字温度计 (二) 测温, ROM和CRC算法 DS18B20数字温度计 (三) 1-WIRE总线 ROM搜 ...

  8. 『忘了再学』Shell基础 — 30、sed命令的使用

    目录 1.sed命令说明 2.行数据操作 (1)查看文件中的数据 (2)删除文件中的数据 (3)向文件中追加数据 (4)向文件中插入数据 (5)修改文件中的多行数据(删除,追加,插入) (6)替换文件 ...

  9. 爱快在PVE下不定时反复重启死机的解决方法

    太长不看版本: 爱快3.6.X在PVE乃至于ESXI下都存在一定的兼容问题! 官网下载3.6.X安装后进入系统设置-升级备份-版本升级,使用爱快3.4.9bin升降级包,下载其中的bin升降级包,将爱 ...

  10. 浅析DispatchProxy动态代理AOP

    浅析DispatchProxy动态代理AOP(代码源码) 最近学习了一段时间Java,了解到Java实现动态代理AOP主要分为两种方式JDK.CGLIB,我之前使用NET实现AOP切面编程,会用Fil ...