質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

Q&A

解決済

2回答

4702閲覧

C言語でガウス分布を実装したい

tam1006

総合スコア28

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

0グッド

0クリップ

投稿2020/07/08 11:03

編集2020/07/08 12:06

前提・実現したいこと

C言語でガウス分布(正規分布)を出すような関数を作っているのですが,上手く作れません。

https://omitakahiro.github.io/random/random_variables_generation.html#Prepare_MT
こちらのサイトを参考にして次のコードを書きました。しかし,標準正規分布の場合にして,例えばn=10000の場合に対してガウス分布を通したものに対して合計値を出力しても,その値が0になりません。
当然その値が0にはならないと思いますが,限りなく0に近い値になるはずです。少なくとも,後述するように-135程度の比較的大きい値になるとは考えられないです。
何がおかしいのか分かりません。型はfloatで,要素数nのベクトルをガウス分布で初期化する方法を教えてください。
よろしくお願いします。

該当のソースコード

C

1#include <stdio.h> 2#include <stdlib.h> 3#include <math.h> 4#include <time.h> 5 6//[0:1]の一様乱数を生成 7float uniform( void ){ 8 return rand()/(RAND_MAX+1.0); 9} 10 11//要素数nのoを正規分布で初期化 12void Gaussian_init(int n, float *o, float mu, float sigma){ 13 int i; 14 for(i=0;i<n;i++){ 15 float z =sqrt( -2.0*log(uniform()) ) * cos( 2.0*M_PI*uniform() ); 16 o[i] = sigma * z + mu; 17 } 18} 19 20int main() { 21 srand(time(NULL)); 22 int n = 10000; 23 float *x = malloc(sizeof(float) * n); 24 float sum = 0.0; 25 26 Gaussian_init(n, x, 0.0, 1.0); 27 for (int i = 0; i < n; i++) { 28 sum += x[i]; 29 } 30 printf("sum = %lf\n", sum); 31 32 return 0; 33} 34

出力結果

sum = -135.292908

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

和が0に近づくというのが間違いだと思います。

互いに独立なn個の確率変数X1, X2, ..., XnがN(μ, σ^2)に従うとすると,
正規分布の再生性から、確率変数の和X1+X2+...+XnはN(nμ, nσ^2)に従います。
今回μ=0, σ=1, n=10000であるため,
N(0, 10000)となります。
なのでsumが±100(1σ区間)に収まる確率が68 %, ±200(2σ区間)に収まる確率が95 %なので,
-135となるのはおおよそ当たり前だと言えます。
個数の1/2乗に比例して標準偏差(ばらつき)が大きくなっていくので、和が限りなく0に近づくというのは間違いです。

一方、sumをnで割った値は0に近づいていきます。
なぜなら、(X1+X2+...+Xn)/nはN(μ, σ^2/n)に従うことから、標準偏差が個数の1/2乗に反比例し
個数を増やせば増やすほどばらつきが小さくなるためです。

投稿2020/07/09 03:28

Penpen7

総合スコア698

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

float uniform()の返す値が,rand()/(RAND_MAX+1.0)であり,
分母の値が,rand()が取りえる最大値よりも1.0だけ多いことが,
結果として計算されるガウス乱数値を負の側にわずかに偏らせるのではないでしょうか.
( uniform()の返す値の範囲が [0:1] ではなく [0:1) になっている )

で,それが,ざっくりとn=10000倍したスケールで最終結果に出てきているのでは.

投稿2020/07/08 14:01

編集2020/07/08 14:05
fana

総合スコア11996

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問