0

break <label>Javaでは、このように書かれています..私がこのコードを移植していたとき... andのようなものがないことに気づきました continue <label>

コマンドでgotoを使用する場合、これを行うためのよりクリーンな方法が必要なため、これらのコマンドが含まれていないことはわかっています..

しかし、私はそれをよりきれいに書き直す方法の下のC#コードを使用することになりましたか?

Java コード

for(JClass c : classes) {
    for(JMethod m : c.getMethods()) {
        JCode code = m.getCode();
        if(code == null)
            continue;
        label: for(int index = 0; index < code.getExceptionLookupTable().length; index++) {
            JException e = code.getExceptionTable().get(index);
            for(int index2 = e.getStartIndex(); index2 < e.getEndIndex(); index2++)
                if(code.getInstruction(index2).getOpcode() == NEW && ((NEW) code.getInstruction(index2)).getType().equals("java/lang/RuntimeException"))
                    continue label;
                if(e.getCatchTypeClassName().equals("java/lang/RuntimeException")) {
                    for(int index = e.getHandlerIndex(); index < code.getInstrLength(); index++) {
                        JInstruction instr = code.getInstruction(index);
                        if(instr.getOpcode() == ATHROW)
                            break;
                        else if(instr instanceof ReturnInstruction)
                            break label;
                    }
                    removeStuff(code, ei--);
                }
            }
    }
}

C# コード。

foreach(JClass c in classes) {
    foreach(JMethod m in c.getMethods()) {
        JCode code = m.getCode();
        if(code == null)
            continue;

        for(int index = 0; index < code.getExceptionTable().Length; index++) {
            bool continueELoop = false;
            bool breakELoop = false;
            JException e = code.getExceptionTable().get(index);
            for(int index2 = e.getStartIndex(); index2 < e.getEndIndex(); index2++) {
                if(code.getInstruction(index2).getOpcode() == JInstructions.NEW && ((NEW) code.getInstruction(index2)).getType().Equals("java/lang/RuntimeException")) {
                    continueELoop = true;
                    break;
                }
            }
            if(continueELoop) continue;

            if(e.getCatchTypeClassName().Equals("java/lang/RuntimeException")) {
                for(int index = e.getHandlerIndex(); index < code.getInstrLength(); index++) {
                    JInstruction instr = code.getInstruction(index);
                    if (instr.getOpcode() == JInstructions.ATHROW) {
                        break;
                    } else if (isReturnInstruction(instr)) {
                        breakELoop = true;
                        break;
                    }
                }
                removeStuff(code, ei--);
            }
            if (breakELoop) break;
        }
    }
}

Java版を見てから移植したC#版を見てみるとわかります..すっきり感がなくなります。コードを短くするような間違いをしましたか? または見栄えが良いですか?助けてくれてありがとう。

4

3 に答える 3

3

C# では、そもそもこのような醜いコードを書くことは決してないでしょう。

複数のメソッドにリファクタリングされ、架空のクラス階層で LINQ を使用するコードを次に示します。

IEnumerable<JCode> GetCodes(IEnumerable<JClass> classes)
{
    return from @class in classes
           from method in @class.Methods
           where method.Code != null
           select method.Code;
}

IEnumerable<Tuple<JCode, JException>> GetCandidates(IEnumerable<JCode> codes)
{
    return from code in codes
           from ex in code.ExceptionTable
           where !code.Instructions
                      .Skip(ex.Start)
                      .Take(ex.End - ex.Start + 1)
                      .Any(i => i.OpCode == New && ...)
           select Tuple.Create(code, ex);
}

その後

void RewriteMethods(IEnumerable<JClass> classes)
{
    var codes = GetCodes(classes);

    var candidates = GetCandidates(codes);

    foreach (var candidate in candidates)
    {
        var code = candidate.Item1;
        var ex = candidate.Item2;

        var instructionsToRemove = code.Instructions
                                       .Skip(ex.HandlerStart)
                                       .TakeWhile(i => i.OpCode != Return)
                                       .Where(i => i.OpCode == AThrow);

        code.RemoveAll(instructionsToRemove);
    }
}
于 2011-07-20T02:59:17.317 に答える
0

純粋に言語仕様について話しているのであれば、C# には続行と中断に相当するものはありません。そのように見えるのは goto だけで、実際には置き換えられません。Anders Hejlsberg はコードの一貫性を台無しにする可能性があるものを嫌っているように見えるので、そのようなものが将来の c# バージョンに含まれるとは思えません。

一部の機能が実装されていないことを示すドキュメントはありません。そのため、stackoverflow に関するそのような質問に対する別の回答を参照することしかできません :) Java の continue <label> と同等の C#?

于 2011-07-20T03:35:28.333 に答える
-1

ブレイク&コンティニューは一種のショートカット

If / Else コンストラクトを使用して、これをより適切にフォーマットし、よりクリーンにします。どちらの言語でも、continue ステートメントを取り除くことができます。for ループに条件を追加することで、同じ for break を行うことができます。繰り返しますが、これは個人的な好みです。

例えば

 if(code == null) continue;

if(code != null)
{

}
于 2011-07-20T02:54:09.997 に答える