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

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

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

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

Q&A

解決済

1回答

1397閲覧

再レンダリングしない方法

Taka009

総合スコア2

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

0グッド

0クリップ

投稿2021/01/29 13:34

・作りたいもの
ビンゴゲームを作っています。
現在の目標は、9マスのビンゴシートを作ることです。
シートの数字は、ゲームのたびに、配置がランダムに変化するようにしたいです。

STARTボタンを押すたびに、数字をランダムで1つ選択し、ここで選択した数字と同じナンバーをシート上で消していき、縦、横、斜め、いずれかで1列消えたらビンゴとしたいです。(現在はシート作りをしており、その他の機能はまだ作り込んでいません)

・今解決したいこと
当初、useEffect内に書いているwhile文は、useEffectを使わずに書いていましたが、STARTボタンを押した時に表示される数字について、useStateを使って更新を行っていることから、ボタンを押すたびに再レンダリングしたくないシート部分まで再レンダリングされ、シート上の数字の配置が変わってしまうという問題が発生しました。

これを解決するため、useEffect内にwhile文を書くという変更を加えました。
useStateの使用による再レンダリングの影響から逃れられると考えたためです。
しかし、この変更により、シートが画面上に表示されなくなってしまいました。

問題解決のため、アドバイスをいただけると幸いです。

javascrip

1import { useEffect, useState } from 'react' 2import styled from 'styled-components' 3 4function App() { 5 6 const [selectedNum, setNum] = useState('START') 7 8 let nums = [] 9 useEffect ( () => { 10 let i = 1 11 while (i <= 9) { 12 const num = Math.floor(Math.random() * (9 + 1 - 1)) + 1 13 if(!nums.includes(num)){ 14 nums.push(num) 15 i++ 16 } 17 } 18 }, []) 19 20 function selectNum () { 21 const array = [0,1,2,3,4,5,6,7,8,9] 22 const selectedNum = array[Math.floor(Math.random() * 10)] 23 setNum(selectedNum) 24 } 25 26 return ( 27 <> 28 <Field> 29 { 30 nums.map((num, index) => { 31 return <div key={index}>{num}</div> 32 }) 33 } 34 </Field> 35 <SelectedNum>{selectedNum}</SelectedNum> 36 <button onClick={selectNum}>ボタン</button> 37 </> 38 ); 39} 40 41export default App; 42 43const Field = styled.div` 44 box-sizing: border-box; 45 height: 270px; 46 width: 270px; 47 display: flex; 48 flex-wrap: wrap; 49 margin-bottom: 20px; 50 div { 51 box-sizing: border-box; 52 border: 1px solid black; 53 height: 90px; 54 width: 90px; 55 line-height: 90px; 56 text-align: center; 57 font-size: 25px; 58 } 59` 60 61const SelectedNum = styled.div` 62 width: 100px; 63 height: 100px; 64 border: solid 1px black; 65 line-height: 100px; 66 text-align: center; 67`

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

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

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

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

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

hoshi-takanori

2021/01/29 21:24

まず、num も state にする必要があるでしょうね。 また、useEffect 内の while 文は何のためのループですか? React の再レンダリングは state が変わった部分にだけ影響するので、表示を変えたくない部分は、対応する state の値を変えなければいいだけですけど…。
Taka009

2021/01/30 08:36

返信が遅れ、大変申し訳ございません! そして、アドバイスありがとうございます。 基本的なことでお恥ずかしいのですが、stateが変わった部分だけ再レンダリングされるという点を認識しておりませんでした。おっしゃる通り、そうであれば、こちらもstateで状態を管理すればうまくいきそうです。 while文は9マスのシートを作るために、数字をランダムに並べた配列を作るために利用しましたが、ループ文を使わなくても、もっとシンプルにランダムな配列を作る方法はありそうですね。もう少し調べてみます。 大変助かりました!!
guest

回答1

0

ベストアンサー

numsがレンダリングを通じて変わらないようにしたいのであれば、useStateで管理するのが適切です。

javascript

1const [nums] = useState(() => { 2 const ret = []; 3 let i = 1 4 while (i <= 9) { 5 const num = Math.floor(Math.random() * (9 + 1 - 1)) + 1 6 if(!ret.includes(num)){ 7 ret.push(num) 8 i++ 9 } 10 } 11 return ret; 12});

依存配列へ[]を指定したuseEffectを使うと、最初にレンダリングされた時numsにだけ値を書き出す、ということになり、2回目以降のレンダリングではnumsは空のままとなります。

投稿2021/01/29 23:52

編集2021/01/29 23:52
maisumakun

総合スコア145184

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

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

Taka009

2021/01/30 08:43

返信が遅れ、大変申し訳ございませんでした! アドバイスありがとうございます。 useEffectのそのような挙動を理解しておりませんでした。 console上で配列が確認できていたのに、表示されない理由がこれでわかりました。 stateの初期値をこのように関数で設定することもできるんですね。大変勉強になりました! ご親切に、ありがとうございます????
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問