- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
目前我计算日志如下:
#define MAXLOG 1001
double myloglut[MAXLOG];
void MyLogCreate()
{
int i;
double exp, expinc;
expinc = (2.0 - 0.1) / MAXLOG;
for (i = 0, exp = 0.1; i <= MAXLOG; ++i, exp += expinc)
myloglut[i] = log(exp);
myloglut[478] = 0; // this one need to be precise
}
double MyLog(double v)
{
int idx = (int)((MAXLOG*(v - 0.1)) / (2.0 - 0.1));
return myloglut[idx];
}
如您所见,我只对 0.1 - 2.0
范围感兴趣。但是,我需要 0
附近更精确。我怎样才能实现非线性计算?还有什么方法可以在此函数中使用一些插值以获得更好的精度?
最佳答案
#include <stdio.h> // for input/output.
#include <math.h> // for mathmatic functions (log, pow, etc.)
// Values
#define MAXELM 1000 // Array size
#define MINVAL 0.1 // Minimum x value
#define MAXVAL 1.9 // Maximum x value
#define EXPVAR 1.4 // Exponent which makes the variation non linear. If set to 1, the variation will be linear.
#define ACRTPT (MINVAL + MAXVAL)/2 // Accurate point. This value is used to know where to compute with maximum accuracy. Can be set to a fixed value.
// Behavior
#define STRICT 0 // if TRUE: Return -1 instead of the floored (or closest if out of range) offset when (x) hasn't been calculated for this value.
#define PNTALL 0 // if TRUE: Print all the calculated values.
#define ASKFOR 1 // if TRUE: Ask for a x value then print the calculated ln value for it.
// Global vars
double results[MAXELM]; // Array to store computed values.
// Func: offset to var conversion
double getvar(int offset)
{
double x = (double)MINVAL + ((double)MAXVAL - (double)MINVAL) * (double)offset / (double)MAXELM;
if(x >= (double)ACRTPT)
x = pow(x - (double)ACRTPT, (double)EXPVAR) + (double)ACRTPT;
else
x = -pow((double)ACRTPT - x, (double)EXPVAR) + (double)ACRTPT;
// This ^ is the equation used when NONLIN = 1; to have a non linear repartition. Feel free to change it. The inverse equation is in `int getoffset(double)`.
return x;
}
// Func: var to offset conversion
int getoffset(double var)
{
double x = var;
if(x >= (double)ACRTPT)
x = pow(x - (double)ACRTPT, 1.0/(double)EXPVAR) + (double)ACRTPT;
else
x = -pow((double)ACRTPT - x, 1.0/(double)EXPVAR) + (double)ACRTPT;
// This ^ is the equation used when NONLIN = 1; to calculate offset with a non linear repartition. Feel free to change it (but it must be the inverse of the one in
// `double getvar(int)` for this to work.). These equations are tied, so you cannot modify one without modifying the other. They are here because
// `pow(negative, non-integer)` always returns `-nan` instead of the correct value. This 'trick' uses the fact that (-x)^(1/3) == -(x^(1/3)) to cicumvent the
// limitation.
int offset = (x - (double)MINVAL) * (double)MAXELM / ((double)MAXVAL - (double)MINVAL);
#if STRICT
if(getvar(offset) != var)
return -1;
return (offset < 0)?-1:(offset > (MAXELM - 1))?-1:offset;
#else
return (offset < 0)?0:(offset > (MAXELM - 1))?MAXELM - 1:offset;
#endif
}
// Func: main.
int main(int argc, char* argv[])
{
int offset;
for(offset = 0; offset < MAXELM; offset++)
results[offset] = log(getvar(offset));
#if PNTALL
for(offset = 0; offset < MAXELM; offset++)
{
printf("[log(%lf) = %lf] ", getvar(offset), results[offset]);
if(!((offset + 1) % 6))
printf("\n");
}
printf("\n");
#endif
#if ASKFOR
double x;
printf("log(x) for x = ");
scanf("%lf", &x);
if((offset = getoffset(x)) < 0)
printf("ERROR: Value for x = %lf hasn't been calculated\n", x);
else
printf("results[%d]: log(%lf) = %lf\n", offset, getvar(offset), results[offset]);
#endif
return 0;
}
最新版本的特点:
log
的值。与上一个版本相比的优点:
cbrt
,而是使用pow
。ACRTPT
))#include <stdio.h> // for input/output.
#include <math.h> // for mathmatic functions (log, pow, etc.)
// Values
#define MAXELM 1000 // Array size
#define MINVAL 0.1 // Minimum x value
#define MAXVAL 1.9 // Maximum x value
#define ACRTPT (MINVAL + MAXVAL)/2 // Accurate point. This value is used to know where to compute with maximum accuracy. Can be set to a fixed value.
// Behavior
#define NONLIN 1 // if TRUE: Calculate log values with a quadratic distribution instead of linear distribution.
#define STRICT 1 // if TRUE: Return -1 instead of the floored (or closest if out of range) offset when (x) hasn't been calculated for this value.
#define PNTALL 0 // if TRUE: Print all the calculated values.
#define ASKFOR 1 // if TRUE: Ask for a x value then print the calculated ln value for it.
// Global vars
double results[MAXELM]; // Array to store computed values.
// Func: offset to var conversion
double getvar(int offset)
{
double x = (double)MINVAL + ((double)MAXVAL - (double)MINVAL) * (double)offset / (double)MAXELM;
#if NONLIN
x = pow((x - ACRTPT), 3) + ACRTPT;
// This ^ is the equation used when NONLIN = 1; to have a non linear repartition. Feel free to change it. The inverse equation is in `int getoffset(double)`.
#endif
return x;
}
// Func: var to offset conversion
int getoffset(double var)
{
#if NONLIN
int offset = ((
cbrt(var - ACRTPT) + ACRTPT
// This ^ is the equation used when NONLIN = 1; to calculate offset with a non linear repartition. Feel free to change it (but it must be the inverse of the one in
// `double getvar(int)` for this to work.)
) - (double)MINVAL) * (double)MAXELM / ((double)MAXVAL - (double)MINVAL);
#else
int offset = (var - (double)MINVAL) * (double)MAXELM / ((double)MAXVAL - (double)MINVAL);
#endif
#if STRICT
if(getvar(offset) != var)
return -1;
return (offset < 0)?-1:(offset > (MAXELM - 1))?-1:offset;
#else
return (offset < 0)?0:(offset > (MAXELM - 1))?MAXELM - 1:offset;
#endif
}
// Func: main.
int main(int argc, char* argv[])
{
int offset;
for(offset = 0; offset < MAXELM; offset++)
results[offset] = log(getvar(offset));
#if PNTALL
for(offset = 0; offset < MAXELM; offset++)
{
printf("[log(%lf) = %lf] ", getvar(offset), results[offset]);
if(!((offset + 1) % 6))
printf("\n");
}
printf("\n");
#endif
#if ASKFOR
double x;
printf("log(x) for x = ");
scanf("%lf", &x);
if((offset = getoffset(x)) < 0)
printf("ERROR: Value for x = %lf hasn't been calculated\n", x);
else
printf("results[%d]: log(%lf) = %lf\n", offset, getvar(offset), results[offset]);
#endif
return 0;
}
这个版本比以前的版本更干净、更容易维护。如果您还有什么需要,请留言。
您可以使用文件顶部的宏配置其行为。
特点:
log
的值。好吧,这是我的第二个解决方案。请参阅下面的原始评论。
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#define MIN_INC 0.001 // This is the minimum increment. If its set to 0, when tmp will be equal to avg, it will never leave this state, since INC_MUL * (tmp - avg)^2 will be 0.
#define INC_MUL 0.2 // This is a number which influences the precision you will get. The smaller it is, the more precise you will be, and the greater will be your result array cardinality.
typedef struct {
double offset;
double value; // value = log(offset). Since the results are not linarly widespread, this is pretty important.
} logCalc;
// Here, we need to use a pointer on a logCalc pointer, since we want to actually SET the address of the logCalc pointer, not the address of one of its copies.
int MyLogCreate(logCalc** arr, double min, double max)
{
if((*arr) != NULL)
return 0;
unsigned int i = 0;
double tmp, avg = (max + min) / 2.0;
for( ; min < avg; min += (INC_MUL * ((avg - min) * (avg - min)) + MIN_INC))
{
(*arr) = (logCalc*)realloc((*arr), sizeof(logCalc) * (i + 1));
(*arr)[i].offset = min;
(*arr)[i++].value = log(min);
}
for(tmp = avg ; tmp < max; tmp += (INC_MUL * ((tmp - avg) * (tmp - avg)) + MIN_INC))
{
(*arr) = (logCalc*)realloc((*arr), sizeof(logCalc) * (i + 1));
(*arr)[i].offset = tmp;
(*arr)[i++].value = log(tmp);
}
return i;
}
int main(int argc, char** argv)
{
logCalc *myloglut = NULL;
unsigned int i,
t = MyLogCreate(&myloglut, .1, 1.9);
for(i = 0; i < (t-1); i++)
{
printf("[log(%lf) = %lf], ", myloglut[i].offset, myloglut[i].value);
if(!((i+1)%6)) // Change 6 to what's best for your terminal $COLUMNS
printf("\n");
}
printf("\n");
free(myloglut);
return 0;
}
<小时/>
计算的线性度来自于您使用线性增量的事实。在 for 循环的每次迭代中,您将 exp
增加 (2.0 - 0.1)/MAXLOG
。
要获得 0 附近更精确的值,您需要:
i
(或取决于 exp
,具体取决于您的操作方式),因此您可以准确地知道您要尝试的数字的“偏移量”计算(以及您需要增加 exp
的量)。当然,你会在0左右计算出更多的结果。这是我当前的实现:
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#define CALCULATE_UNTIL 2.0
#define PRECISE_UNTIL 1.0
typedef struct {
double offset;
double value;
} logCalc;
logCalc *myloglut = NULL;
int MyLogCreate()
{
double exp = 0.1;
int i;
for (i = 0; exp <= CALCULATE_UNTIL; exp += (exp < PRECISE_UNTIL)?0.0001898:0.001898)
{
myloglut = realloc(myloglut, sizeof(logCalc) * (i + 1));
myloglut[i].offset = exp;
myloglut[i++].value = (i == 4780)?0:log(exp);
}
return i; // So you know how big the array is. Don't forget to free(myloglut); at the end of your code.
}
int main(int argc, char** argv)
{
int i,
t = MyLogCreate();
for(i = 0; i < t; i++)
{
printf("[log(%lf) = %lf], ", myloglut[i].offset, myloglut[i].value);
if(!(i%6)) // For formatting purposes.
printf("\n");
}
printf("\n");
free(myloglut);
return 0;
}
我创建了一个新类型来存储 exp 的值,这对于了解结果的对数可能很有用。
更新:我不确定您想做什么。您想要精确到 log(x) = 0 左右还是 x = 0 左右?对于第一种情况,我可能必须再次重写代码才能使其按您的要求工作。另外,您是否希望结果在接近 0 时更加精确,或者您希望结果在给定范围内(就像现在一样)更加精确?
关于c - 非线性 lut 对数精度,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13740334/
我想在 python 中找出一个整数的 log10,但我得到了一个错误,比如数学域错误 我的代码是这样的w=math.log10(q*q1)/math.log10(2) 其中 q1,q2 是整数 是的
舍入小数 在 NumPy 中,主要有五种方法来舍入小数: 截断 去除小数部分,并返回最接近零的浮点数。使用 trunc() 和 fix() 函数。 示例: import numpy as n
我有一个数值范围为 0 到 100 的 slider 。 我想将它们映射到 100 到 10,000,000 的范围内。 我在网上看到过一些函数,但它们都是用 C++ 编写的。我需要它在 Javasc
我想请用户输入一个整数(N),然后显示他/她输入的整数的 10 对数。我已经成功计算了 10 对数,但不知道如何像下面这样显示它: Write in an Integer: 455666 455666
我将 x 轴设置为对数刻度。最大值为10000,最小值为1。 GraphPane mypane = zedgraphcontrol.GraphPane; mypane.XAxis.Type = Axi
我正在尝试编写一个快速算法来计算 log gamma function 。目前我的实现看起来很幼稚,只是迭代了 1000 万次来计算 gamma 函数的对数(我还使用 numba 来优化代码)。 im
这个问题在这里已经有了答案: How to show minor tick labels on log-scale with Matplotlib (2 个答案) 关闭 7 年前。 将行 plt.y
抱歉标题不好 ;) 我正在尝试重新创建我在其他一些工作中遇到的 matlab 图,但我不太了解他们使用的比例。 y轴增量如下(从上往下[+ve y]): 0.9999,0.999,0.99,0.9,0
由于 1000 的以 10 为底的对数是 3,您可能期望 Math::log(1000, 10) 返回 3。相反,它返回 2.9999999999999996。 这是因为 Ruby 中的 float
我对对数 X 轴有疑问。阈值大于 0,x 的最小值为 1,并且所有 X 值都大于 0。并且仍然给我相同的错误 Can't plot zero or subzero values on a logari
我需要在我的应用程序中实现折线图,我想使用 MPAndroidChart。问题是 y 轴上的值将介于 1 和 1x10^-12 之间。这就是为什么我需要在该轴上的对数 View 。 有没有办法用那个库
我正在尝试按照 Logarithmic slider 中的示例进行操作. 这是我使用的代码: Timeline._MIN_PER_MINUTE = 1; Timeline._MIN_PER_HOUR
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
我尝试为对数 y 轴绘制条形图。不幸的是,如果我将 y 轴设置为对数,则不再有条形图。我该怎么做才能实现这一目标?是否可以在 bar-function 中设置引用点(默认似乎为零)? 我的代码是: i
所以我一直在努力掌握 Big Oh 的计算方法。我觉得我已经掌握了基础知识,但对看似非常简单的计算感到困惑。所以如果下面的计算有很大的 O(n log n)(我真的希望我至少做对了)改变循环的顺序对复
我知道二维绘图的 semilogx 和 semilogy。 SURF 和 MESH 有什么等价物吗? 最佳答案 如上述链接所述,要将所有三个轴设置为对数刻度,请使用 set(gca, 'XScale'
这看起来很简单,但我在用 Ruby 计算 log (Base 5) 时遇到了问题。 显然标准的 base-10 日志工作正常: >> value = Math::log(234504) => 12.3
这段代码是用 C 语言根据 pollard 的对数 rho 算法(来自 wiki)编写的。在此代码中,如果我输入 alpha=2、beta=5、N=1019,则必须返回 a=681、b=378、A=3
有了this question之后通过指向 an external site 的链接回答,我意识到我解决了一个问题,只是为了得到另一个问题:在对数刻度上,MESH 和 SURF 函数的 C=Z 参数不
我正在尝试解决 the SPOJ problem PGCD , 它询问最大公约数表中出现了多少个素数。 我想到的第一个想法是先通过筛分生成素数。 然后,对于每个素数 p,查看有多少对(a,b),其中
我是一名优秀的程序员,十分优秀!