我有两个无符号长整数 X 和 Y,其中 X < Y,但两者都可能非常大。我想计算 X / Y 小数点后的第一个数字。例如,如果 X 是 11,Y 是 14,那么 11 / 14 是 .785……,所以结果应该是 7。
(X * 10) / Y 可以工作,除非 X * 10 溢出会产生错误的结果。如果我有理由相信它足够精确以计算正确的结果,则转换为 double 将起作用。
这是在 C 中。感谢您的帮助!
我有两个无符号长整数 X 和 Y,其中 X < Y,但两者都可能非常大。我想计算 X / Y 小数点后的第一个数字。例如,如果 X 是 11,Y 是 14,那么 11 / 14 是 .785……,所以结果应该是 7。
(X * 10) / Y 可以工作,除非 X * 10 溢出会产生错误的结果。如果我有理由相信它足够精确以计算正确的结果,则转换为 double 将起作用。
这是在 C 中。感谢您的帮助!
我不愿意通过浮点转换来保证准确的结果,除非尾数有足够的位来准确表示所有整数值。例如,tonsider y=9223372036854775807 和 x = (y div 10) - 1 = 922337203685477579。其中“div”是整数除法。x/y 是 0.09999999999999999981568563067746...,但使用双精度值会让你 >= 0.1。这是双精度数只有 52 位的有效数字的结果(而 y 需要 61 位,而 x 需要大约 58 位)
您可能能够使用 80 位或 128 位 FP 精度,在这种情况下,您会得到正确的答案,因为尾数将 >=64 位(ULL 是 64 位,对吗?),因此您可以无损地表示数字。
我将从一个近似值开始(使用整数或 FP 算术),然后尝试乘法以查看答案是否应该是 1 或更多。关键的见解是,只要您知道两个量之间的差异小于最大 unsigned int 的一半,您仍然可以比较两个可能溢出的整数。当 TCP 序列号溢出时,这种比较技术是必要的。
如果您只想使用整数算术,下面的函数“fdd(x,y)”可以工作。我已经包含了一个 main() 来显示一些结果:
#include <iostream>
using namespace std;
typedef unsigned char ull; // change char to any integral type e.g. long long
const ull maxull=(ull)-1;
const ull halfull = maxull/2;
typedef unsigned long long asint;
// x = X mod (maxull+1), y= Y mod (maxull+1). we only know x and y
// if we assume |X-Y|<halfull, then we return X<Y:
inline bool less_mod_near(ull x, ull y) {
return (x<=halfull == y<=halfull) ? x<y : y>x;
}
// assuming x<y, return first decimal digit of 10x/y (return is in [0..9])
inline int fdd(ull x, ull y) {
// assert(x<y);
if (x<=maxull/10) return (10*x)/y;
// for speed, and to ensure that y>10 to avoid division by 0 later
ull r=y/10;
if (r*10==y) return x/r;
ull ub=x/(r+1); // ub >= 10x div y (without overflow)
ull x10=x*10; // allow overflow
cout<<"ub="<<(asint)ub<<" x10="<<(asint)x10<<" r="<<(asint)r<<" ";
return less_mod_near(x10,ub) ? ub-1 : ub;
// we already handled the 10 evenly divides y case
}
int pdd(ull x, ull y,ull mustbe)
{
ull d=fdd(x,y);
cout << (asint)x << '/' << (asint)y << " = ." << (asint)d << "...";
if (d!=mustbe) cout << " (should be "<<(asint)mustbe<<")";
cout<<endl;
// assert(a==d);
}
int main() {
pdd(0,1,0);
pdd(1,2,5);
pdd(11,101,1);
pdd(10,101,0);
pdd(49,69,7);
pdd(50,69,7);
pdd(48,69,6);
pdd(160,200,8);
pdd(161,200,8);
pdd(159,200,7);
pdd(254,255,9);
}
输出:
0/1 = .0...
1/2 = .5...
11/101 = .1...
10/101 = .0...
ub=7 x10=234 r=6 49/69 = .7...
ub=7 x10=244 r=6 50/69 = .7...
ub=6 x10=224 r=6 48/69 = .6...
160/200 = .8...
161/200 = .8...
159/200 = .7...
ub=9 x10=236 r=25 254/255 = .9...
如果我有理由相信它足够精确以计算正确的结果,则转换为 double 将起作用。
如果您只需要第一个数字,那么 double 肯定足够精确。
编辑: wrang-wrang 在评论中的反例证明我错了。
转换为 Double 将放弃 64 位中除数和施主的 12 位精度。大多数情况下,它会给您正确的答案,但它偶尔会出现舍入错误,除非它使用 80 位浮点格式存储.
编辑:除非我太困而看不到错误,否则我认为 wrang-wrang 的答案会起作用。我早上上班,开车6小时到客户现场。乌格。夜晚。
编辑:最后一件事。x86 使用 80 位内部表示。如果你想扔几个 asm 指令,我认为有一些操作码可以将 int64 转换为 float80。这将是一个比纯 C 实现更优雅且肯定更快的解决方案,尽管它不是可移植的。
X % Y 给出余数 R
然后,您可以从 R 和 Y 计算答案的小数部分
R / Y
要获得第一个数字,请使用整数除法:
(long)((long)10*R/Y)
这应该将数字四舍五入并去掉任何额外的小数。
编辑:
为了匹配您的问题(对于那些想要挑剔的人)它
Y % X = R
和
R / X
怎么样
x / ((y + 9) / 10)
y + 9 用于将分母中的商 y/10 向上取整,因此整体结果向下取整。
对于大 x 和 y,它应该几乎总是正确的,但它并不完美,它为您的示例 11 / 14 产生 5。
问题是您由于分裂而丢失了信息。由于乘以十已经溢出,除非使用更大的数字数据类型,否则您无法解决这个问题。
如果您准备接受范围限制,您可以用整数算术完成所有操作:-
(X * 10) / Y
在您的示例中:
(11 * 10) / 14
=> 110 / 14
=> 7
限制是您将 X 的最大值减少了 10 倍。