Я попытался портировать код непосредственно из исходного кода java на pascal, однако он бросает ошибку времени выполнения.Gaussian in pascal
Как я могу получить правильную гауссову кривую? Как насчет паскалей, встроенных в функции?
Исходный код:
synchronized public double nextGaussian() {
// See Knuth, ACP, Section 3.4.1 Algorithm C.
if (haveNextNextGaussian) {
haveNextNextGaussian = false;
return nextNextGaussian;
} else {
double v1, v2, s;
do {
v1 = 2 * nextDouble() - 1; // between -1 and 1
v2 = 2 * nextDouble() - 1; // between -1 and 1
s = v1 * v1 + v2 * v2;
} while (s >= 1 || s == 0);
double multiplier = StrictMath.sqrt(-2 * StrictMath.log(s)/s);
nextNextGaussian = v2 * multiplier;
haveNextNextGaussian = true;
return v1 * multiplier;
}
}
Первая попытка паскаль порт (броски ошибки во время выполнения):
function log (n : double) : double;
begin
result := ln(n)/ln(10);
end;
var hgauss : boolean;
var ngauss : double;
function gauss() : double;
var x1, x2, w : double;
begin
if hgauss then
begin
result := ngauss;
hgauss := false;
end else
begin
repeat
x1 := 2.0 * rand() - 1.0;
x2 := 2.0 * rand() - 1.0;
w := x1 * x1 + x2 * x2;
until w >= 1.0;
w := sqrt((-2.0 * log(w))/w);
result := x1 * w;
ngauss := x2 * w;
hgauss := true;
end;
end;
Недопустимая операция с плавающей точкой здесь:
w := sqrt((-2.0 * log(w))/w);
Вторая попытка преобразования (работает, но я не уверен, что математика верна):
function log (n : double) : double;
begin
result := ln(n)/ln(10);
end;
var hgauss : boolean;
var ngauss : double;
function gauss() : double;
var x1, x2, w, num : double;
begin
if hgauss then
begin
result := ngauss;
hgauss := false;
end else
begin
repeat
x1 := 2.0 * rand() - 1.0;
x2 := 2.0 * rand() - 1.0;
w := x1 * x1 + x2 * x2;
until w >= 1.0;
num := -2.0 * log(w)/w;
w := sqrt(abs(num));
if num < 0 then w := -w;
result := x1 * w;
ngauss := x2 * w;
hgauss := true;
end;
end;
Какой компилятор Паскаля вы используете ? Как дескриптор java делится на 0? , Вы должны справиться с этим в своем паскале.Вы также пренебрегаете случаем, когда w = 0 в вашем цикле повтора, (s == 0 в вашей java) –
Я использую SCAR DIVI 3.41. Насколько я знаю, Java не позволяет делить на ноль, поэтому я думаю, что компиляторы должны обрабатывать его одинаково. Существует также функция RandG(), которая якобы возвращает «гауссовский», но, похоже, не находится в нормальном распределении. – Colby
Если вы используете код Free Pascal, вы также можете использовать функцию [randg] (http://lazarus-ccr.sourceforge.net/docs/rtl/math/randg.html) для получения случайных чисел Гаусса. См. Http://wiki.freepascal.org/Generating_Random_Numbers для получения дополнительной информации о том, как генерировать случайные числа из большого количества разных распределений. – jwdietrich