.NET ヘルプ

C# foreach with index (開発者のための仕組み)

ジョルディ・バルディア
ジョルディ・バルディア
2024年10月23日
共有:

C#では、foreach文は通常、配列、リスト、その他の列挙可能な型のようなコレクションを反復処理するために使用されます。しかし、foreachループには、現在の反復を追跡するための組み込みのインデックス変数がないという制約がある。 開発者は、現在の要素のインデックスにアクセスする必要があります。 以下では、この機能を実装するためのさまざまな方法と、その方法について説明します。IronPDFライブラリ.

foreachループの基本

foreachループは、IEnumerableを実装する配列、リスト、辞書、その他の型の反復処理を簡素化するように設計されています。 foreach文を使って整数データ型の配列をループ処理する基本的な例を示します:

int[] numbers = { 10, 20, 30, 40 };
foreach (int number in numbers)
{
    Console.WriteLine(number);
}
int[] numbers = { 10, 20, 30, 40 };
foreach (int number in numbers)
{
    Console.WriteLine(number);
}

この例では、数字は、各反復中のコレクションの要素を表しています。 ループは自動的に配列のすべての要素を繰り返し処理します。 しかし、現在の要素のインデックスにアクセスする組み込みの方法はありません。

foreachループでのインデックスの処理

C#はforeachループでインデックスを直接提供しませんが、いくつかのテクニックで解決できます。 これらの方法について詳しく説明しましょう。

方法1:別の変数を使用する

現在の要素のインデックスを取得する最も簡単な方法の1つは、外部インデックス変数を使用することです。 ループ内で手動でインクリメントする必要があります:

int[] numbers = { 10, 20, 30, 40 };
int numberIndex = 0;
foreach (int number in numbers)
{
    Console.WriteLine($"Index: {numberIndex}, Value: {number}");
    numberIndex++;
}
int[] numbers = { 10, 20, 30, 40 };
int numberIndex = 0;
foreach (int number in numbers)
{
    Console.WriteLine($"Index: {numberIndex}, Value: {number}");
    numberIndex++;
}

このコードでは、index変数はループ開始前に初期化され、各反復中にループ内でインクリメントされます。 このアプローチは有効ですが、手作業でインデックスを維持する必要があり、必ずしも理想的ではありません。

方法2:LINQのSelectメソッドを使う

LINQのSelectメソッドを使用すると、コレクションの各要素を、そのインデックスを含む新しいフォームに投影することができます。 Here's an example: 例があります。

int[] numbers = { 10, 20, 30, 40 };
foreach (var item in numbers.Select((value, index) => new { value, index }))
{
    Console.WriteLine($"Index: {item.index}, Value: {item.value}");
}
int[] numbers = { 10, 20, 30, 40 };
foreach (var item in numbers.Select((value, index) => new { value, index }))
{
    Console.WriteLine($"Index: {item.index}, Value: {item.value}");
}

この例では、Selectは、現在の要素の値とそのインデックスの両方を含む匿名オブジェクトを作成します。 foreachループは、これらのオブジェクトを繰り返し処理し、インデックスと値の両方に直接アクセスできます。

方法3:カスタムイテレータを使用する

yield return キーワードを使用すると、カスタム反復子拡張メソッドを実装して、現在の要素とそのインデックスの両方を返すメソッドを生成できます。 これは少し高度ですが、柔軟なソリューションを提供します。

public static IEnumerable<(int index, T value)> WithIndex<T>(this IEnumerable<T> source)
{
    int index = 0;
    foreach (T value in source)
    {
        yield return (index, value);
        index++;
    }
}
public static IEnumerable<(int index, T value)> WithIndex<T>(this IEnumerable<T> source)
{
    int index = 0;
    foreach (T value in source)
    {
        yield return (index, value);
        index++;
    }
}

これで、あなたのコレクションでこの拡張メソッドを使用できます:

int[] numbers = { 10, 20, 30, 40 };
foreach (var (index, value) in numbers.WithIndex())
{
    Console.WriteLine($"Index: {index}, Value: {value}");
}
int[] numbers = { 10, 20, 30, 40 };
foreach (var (index, value) in numbers.WithIndex())
{
    Console.WriteLine($"Index: {index}, Value: {value}");
}

このアプローチでは、手作業によるインデックス管理を再利用可能なメソッドに抽象化することで、foreach with index問題に対するよりエレガントなソリューションを作成します。

インデックスにアクセスするための while ループの使用

配列やリストのようなコレクションを扱う場合、インデックス変数と組み合わせてwhileループを使用すると、インデックスと現在の要素の両方にアクセスできます:

int[] numbers = { 10, 20, 30, 40 };
int index = 0;
while (index < numbers.Length)
{
    Console.WriteLine($"Index: {index}, Value: {numbers[index]}");
    index++;
}
int[] numbers = { 10, 20, 30, 40 };
int index = 0;
while (index < numbers.Length)
{
    Console.WriteLine($"Index: {index}, Value: {numbers[index]}");
    index++;
}

C# インデックス付き foreach(開発者向けの仕組み):図1 - インデックスの出力

この方法では、インデックス変数を配列またはリストの添え字として使用することで、インデックスと現在の要素の両方に直接アクセスできます。

.NET におけるカスタム・コレクションとイテレータ

カスタマイズされたコレクションを使用している場合は、インデックス付きアクセスをサポートするイテレータを実装することができます。 IEnumerable インターフェースを実装し、yield return ステートメントを使用することで、要素とそのインデックスの両方を返すイテレータを作成できます。

IEnumerableを実装したカスタムコレクションを作成する例を示します:

public class CustomCollection<T> : IEnumerable<T>
{
    private T[] _items;
    public CustomCollection(T[] items)
    {
        _items = items;
    }
    public IEnumerator<T> GetEnumerator()
    {
        for (int i = 0; i < _items.Length; i++)
        {
            yield return _items[i];
        }
    }
    IEnumerator IEnumerable.GetEnumerator()
    {
        return GetEnumerator();
    }
}
public class CustomCollection<T> : IEnumerable<T>
{
    private T[] _items;
    public CustomCollection(T[] items)
    {
        _items = items;
    }
    public IEnumerator<T> GetEnumerator()
    {
        for (int i = 0; i < _items.Length; i++)
        {
            yield return _items[i];
        }
    }
    IEnumerator IEnumerable.GetEnumerator()
    {
        return GetEnumerator();
    }
}

そして、foreachループでこのカスタムコレクションを使用することができます:

var customCollection = new CustomCollection<int>(new int[] { 10, 20, 30, 40 });
foreach (int number in customCollection)
{
    Console.WriteLine(number);
}
var customCollection = new CustomCollection<int>(new int[] { 10, 20, 30, 40 });
foreach (int number in customCollection)
{
    Console.WriteLine(number);
}

GetEnumeratorメソッドを実装し、yield returnを使用することで、.NETの他のコレクションと同様にforeachループでカスタムコレクションを使用できるイテレータを作成します。

辞書の使用とキーと値のペアの反復処理

辞書を扱う場合、foreachループを使えば、キーと値のペアを直接反復処理することができます。 これは、各反復中にキーと値の両方にアクセスする一般的なユースケースです:

Dictionary<int, string> dict = new Dictionary<int, string>
{
    { 1, "Apple" },
    { 2, "Banana" },
    { 3, "Cherry" }
};
foreach (var kvp in dict)
{
    Console.WriteLine($"Key: {kvp.Key}, Value: {kvp.Value}");
}
Dictionary<int, string> dict = new Dictionary<int, string>
{
    { 1, "Apple" },
    { 2, "Banana" },
    { 3, "Cherry" }
};
foreach (var kvp in dict)
{
    Console.WriteLine($"Key: {kvp.Key}, Value: {kvp.Value}");
}

この例では、kvp.Key が現在のキーを、kvp.Value が現在の値を示します。

C#とforeachループとインデックスでIronPdfを使う

C# インデックス付き foreach(開発者向けの仕組み):図2 - IronPDF

IronPDFはPDFを扱うためのライブラリです。HTMLからのPDF生成C# での PDF 関連の作業。 最新の.NET Frameworkにも対応しています。 IronPDFを使ってPDFを生成する場合、データのコレクションを繰り返し処理し、PDFファイルに動的にコンテンツを挿入する必要があるかもしれません。foreachループをインデックス処理と組み合わせることで、コレクション内の現在のアイテムのインデックスに基づいて、位置決め、番号付け、カスタムロジックを管理することができます。 IronPDFを使ってPDFを作成し、コレクションの各アイテムをインデックスとともにドキュメントに挿入する例を示します。

using IronPdf;
class Program
{
    static void Main(string[] args)
    {
        // Create a new PDF document
        var pdf = new ChromePdfRenderer();
        // Sample data array
        string[] items = { "First Item", "Second Item", "Third Item" };
        // Initialize the HTML content with foreach loop and index
        string htmlContent = "<html><body>";
        int index = 0;
        foreach (var item in items)
        {
            htmlContent += $"<h2>Item {index + 1}: {item}</h2>";
            index++;
        }
        htmlContent += "</body></html>";
        // Render the HTML to PDF
        var pdfDocument = pdf.RenderHtmlAsPdf(htmlContent);
        // Save the PDF document
        pdfDocument.SaveAs("output.pdf");
        // Notify completion
        Console.WriteLine("PDF created successfully with indexed items.");
    }
}
using IronPdf;
class Program
{
    static void Main(string[] args)
    {
        // Create a new PDF document
        var pdf = new ChromePdfRenderer();
        // Sample data array
        string[] items = { "First Item", "Second Item", "Third Item" };
        // Initialize the HTML content with foreach loop and index
        string htmlContent = "<html><body>";
        int index = 0;
        foreach (var item in items)
        {
            htmlContent += $"<h2>Item {index + 1}: {item}</h2>";
            index++;
        }
        htmlContent += "</body></html>";
        // Render the HTML to PDF
        var pdfDocument = pdf.RenderHtmlAsPdf(htmlContent);
        // Save the PDF document
        pdfDocument.SaveAs("output.pdf");
        // Notify completion
        Console.WriteLine("PDF created successfully with indexed items.");
    }
}

こちらが出力されたPDFファイルです。

C# インデックス付き foreach(開発者向けの仕組み):図3 - PDF出力

結論

C# インデックス付き foreach(開発者向けの仕組み):図4 - ライセンス

C#では、foreachループはコレクションを繰り返し処理する便利な方法ですが、インデックスのネイティブサポートがありません。 しかし、この制限を克服する方法はいくつかあります。 単純なインデックス変数でも、LINQのSelectメソッドでも、カスタムのイテレータでも、反復処理中に現在または次の要素のインデックスにアクセスすることができます。 これらのテクニックを理解することで、特に各要素のインデックスを知る必要がある場合に、foreachループをより効率的に使用することができます。

IronPDFを使えば、すぐにコミットする必要はありません。 当社では無料試用また、ソフトウェアの機能を深く掘り下げることができるような翻訳が求められます。 気に入った場合、ライセンスは$749から始まります。

ジョルディ・バルディア
ソフトウェアエンジニア
ジョルディは、Iron Softwareでのスキルを活かしていないときには、ゲームプログラミングをしており、Python、C#、C++に最も堪能です。彼は製品テスト、製品開発、研究の責任を共有しており、継続的な製品改善に大きな価値をもたらしています。この多様な経験は彼を常に挑戦的で魅力的に保ち、彼はIron Softwareで働く一番好きな側面の一つだと言っています。ジョルディはフロリダ州マイアミで育ち、フロリダ大学でコンピューターサイエンスと統計学を学びました。
< 以前
C# インデクサー(開発者のための仕組み)
次へ >
Socket io .NET(開発者向けの動作方法)