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

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

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

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

Q&A

解決済

4回答

21609閲覧

C# クラス配列の宣言方法

rkw4a

総合スコア7

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

0グッド

0クリップ

投稿2017/08/22 01:05

###前提・実現したいこと
CからC#へ置き換えをおこなっています。
構造体をクラスに置き換えていますが
入れ子になっている場合、
下記宣言方法が問題ないのかアドバイスをいただけないでしょうか。

特にエラー等発生なく使用できましたが、間違った考え方だといけないので
質問させていただいています。

この構造体の用途は
グローバル変数として宣言し、ファイルから読込んだ値を流しこみ、
アプリ起動中常時参照します。

###該当のソースコード

C

1 2typedef struct{ 3 int A; 4 int B[10]; 5 int C[10]; 6}STR_A; 7 8typedef struct{ 9 char D[256]; 10 STR_A E[100]; 11}STR_B; 12 13typedef struct{ 14 STR_B F[200]; 15 STR_B G[200]; 16}STR_C; 17

C#

1 2public class CLS_A 3{ 4 public int A; 5 public int[] B = new int[10]; 6 public int[] C = new int[10]; 7} 8 9public class CLS_B 10{ 11 public string D; 12 public CLS_A[] E = new CLS_A[100]; 13 14 public CLS_B() 15 { 16 int i; 17 for (i = 0; i < 100; i++) 18 { 19 E[i] = new CLS_A(); 20 } 21 } 22} 23 24public class CLS_C 25{ 26 public CLS_B[] F = new CLS_B[200]; 27 public CLS_B[] G = new CLS_B[200]; 28 29 public CLS_C() 30 { 31 int i; 32 for (i = 0; i < 200; i++) 33 { 34 F[i] = new CLS_B(); 35 G[i] = new CLS_B(); 36 } 37 } 38} 39

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

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

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

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

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

guest

回答4

0

ベストアンサー

int の宣言は、forの中に入れられる。forの中だけでしか使わない変数なので、こっちのほうが絶対正しい。

for (int i = 0; i < 200; i++) { F[i] = new CLS_B(); G[i] = new CLS_B(); }

あと、初期化はこんな感じにかくのもありでは?
とか思いますけど。

F = Enumerable.Range(1,200).Select(_=>new CLS_B()).ToArray()

あと、パブリックのものがただのフィールドで、プロパティではないですね。
これもあまり良くありません。
クラスの外から見えるものは、プロパティを使う、というのが一応作法です。直接触れるようにはしない。

そのため、

public class CLS_C { public CLS_B[] F {get; private set;} = Enumerable.Range(1,200).Select(_=>new CLS_B()).ToArray(); public CLS_B[] G {get; private set;} = Enumerable.Range(1,200).Select(_=>new CLS_B()).ToArray(); }

こんな感じ書くのがいいのですかね。
外から、Setすることがないので、たぶん、privateをつけるべきかな。
元のコードだと、ノーガードで外から変更できるので、ちょいとやばいと思いますね。
予め防げるものは防ぐが原則なので、外からの変更は、メソッドを介すという原則がいいと思います。

追記
Enumerable.Repeatの仕様を勘違いしていたので修正

投稿2017/08/22 01:34

編集2017/08/23 02:19
kiichi54321

総合スコア1984

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

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

MMashiro

2017/08/23 01:10 編集

実際にあまり使ったことがないので間違えていたらすいません F = Enumerable.Repeat(new CLS_B(), 200).ToArray(); こちらの書き方だと実体は一つだけで配列の中には第1引数のインスタンスへの参照が200個生成されるとかだった気がします CLS_B型が参照型の為に発生する問題なのでCLS_B型をstruct(値型)に変更して問題ないのであればstructにすることで回避可能です
ozwk

2017/08/23 01:36

どうしてもLinqしたいなら Enumerable.Range(1,200).Select(_=>new CLS_B()).ToArray() とかどうでしょう
kiichi54321

2017/08/23 01:59 編集

お二人方、正しいですね。私は、Repeatの仕組みを勘違いしていました。
rkw4a

2017/08/23 23:37

回答ありがとうございました。
guest

0

自分的にですが(あくまで自分的にです)、.NET Framework の C# にパブリックフィールドや配列を使うのはピンときません。

可能であれば、代わりにパブリックプロパティ、List<T> 型を使うということも考えてはいかがでしょう?

具体的には以下の通りです。質問者さんの目的には合わないかもしれませんが、ご参考まで。

public class CLS_A { public int A { get; set; } public List<int> B { get; set; } public List<int> C { get; set; } public CLS_A() { this.B = new List<int>(); this.C = new List<int>(); } } public class CLS_B { public string D { get; set; } public List<CLS_A> E { get; set; } public CLS_B() { this.E = new List<CLS_A>(); } } public class CLS_C { public List<CLS_B> F { get; set; } public List<CLS_B> G { get; set; } public CLS_C() { this.F = new List<CLS_B>(); this.G = new List<CLS_B>(); } }

投稿2017/08/22 01:52

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

rkw4a

2017/08/23 01:01

配列よりlistのほうが利便性が良いので、通常(処理速度やメモリを気にしない)であればlistを優先して使用するという考えでよいのでしょうか?
退会済みユーザー

退会済みユーザー

2017/08/23 02:56

使いやすさ優先で考えれば良いと思います。実際、処理速度やメモリが問題になるということはまずなさそうですし。
rkw4a

2017/08/23 23:37

回答ありがとうございました。
guest

0

クラスの基本的な宣言としては問題ありませんが、アプリから常時参照する(唯一の値)のであればstaticクラスにすることをお勧めします。
C#でのポイントは、常時唯一の値を保持したい場合にはstaticを使用することまた名前空間も重要ですので一度調べては如何でしょうか?
<追記>
初めてなもので、ソースが下にまだ隠れていたのを見逃していました。
補足ポイント
①上のCでは、構造体の宣言のみですがC#では、「new CLS_A()」で値を放り込んでいますね。実際に使うときに値を入れるべきではないでしょうか?
②C#もCの構文と同じようにfor文の中で(int i = 0; i < 100; i++)とできますよ。

投稿2017/08/22 01:22

編集2017/08/22 01:38
magokoro

総合スコア25

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

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

rkw4a

2017/08/23 00:49

staticクラス参考になりました。
guest

0

まったく問題ありません。

投稿2017/08/22 01:24

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問