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

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

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

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

Q&A

解決済

1回答

2335閲覧

C言語 フィボナッチ応用 再帰なし

progokina8

総合スコア4

C

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

0グッド

0クリップ

投稿2020/11/25 10:00

前提・実現したいこと

c言語を始めたばかりの初心者です。
フィボナッチ数列の変形版についてのプログラムを書いています。
問題は以下の通りです。

数列f(n)を
f(0) = 3, f(1) = 0, f(2) = 2, f(n) = f(n-2) + f(n-3) (n>2)
と定義する。
n>1かつf(n)がnで割り切れるような整数nのうち、32番目に小さいnをkとする。
k, f(k)の値を出力せよ。

ex. 8番目に小さいnは19で, f(19) = 209.

この問題に対して再帰関数を用いようと思いましたが時間効率が悪いと知ったので、再帰を用いずに該当のソースコード(下に記す)を書きました。しかし、得られた出力(下に記す)が明らかに誤っています(nが負になっている)。
原因の分かる方がいたらご教授お願い致します。

該当のソースコード

C

1#include <stdio.h> 2#define TARGET 32 3 4int main(){ 5 long f0=3, f1=0, f2=2, fn; 6 int n, cnt=1; //n=2の時、f(2)=2で条件を満たすので予めcntを1にする 7 for(n=3; ; n++){ 8 fn = f1 + f0; 9 if(fn % n == 0) cnt++; 10 if(cnt >= TARGET){ 11 printf("%d %ld\n", n, fn); 12 break; 13 } 14 f0 = f1; 15 f1 = f2; 16 f2 = fn; 17 } 18 return 0; 19} 20

###得られた出力
-27355683 1231005735

試したこと

TARGETを8に変えて実行すると、exの通り 19 209 と出力される。

補足情報(FW/ツールのバージョンなど)

実行環境はWindows, VSCode
gccでコンパイル。

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

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

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

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

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

guest

回答1

0

ベストアンサー

原因

longの範囲では扱えなくなる(オーバーフロー).

投稿2020/11/25 10:15

fana

総合スコア11708

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

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

progokina8

2020/11/25 10:48

回答有り難うございます。 longでは足りないとの指摘を受け、unsigned longを用いたのですが、今度は間違いどころか出力すらされませんでした。(もちろん、%ldを%luに変更しましたが...。) どうすれば良いのでしょうか。 重ねての質問、お許しください。
tatsu99

2020/11/25 11:35

OSはlinuxですか? 32ビットのコンパイラですか? 64ビットならlongは8バイトになるのでOKですが、32ビットだとlongは4バイトなので桁数が足りません。 printf("%d\n",sizeof(f0)); の値はいくつですか。8ならOK、4ならNGです。
progokina8

2020/11/25 12:08

4でした...。NGですね。 この場合はなす術なしということでしょうか...
tatsu99

2020/11/25 12:18 編集

linux(gcc)ならlong long int型が使えます。 long long int型を出力時、printfを使用時は%lldで出力します。 long long int型は32ビットモードでも、8バイトの整数型であることが保障されています。 long long intが使えないなら OS,コンパイラを明記すれば、良い回答が得られるかも知れません。
progokina8

2020/11/25 12:33 編集

long long intにしたところ、131 9957916285211697という出力が得られました。割り切れるので恐らく合っていると思います。 アドバイスまで本当に有り難うございました。
progokina8

2020/11/25 12:35

ヒントを下さったfanaさんにも感謝を述べたいと思います。 有り難うございました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.46%

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

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

質問する

関連した質問