关于Razor中嵌套代码块声明的问题

时间:2011-05-23 15:40:17

标签: asp.net-mvc-3 razor

我最近将一个项目从MVC 1升级到了MVC 3,现在我正在尝试使用Razor。

在一个视图中,我有一个foreach代码块,但是嵌套的if语句似乎不希望它前面有@。

我原来的代码是:

@foreach(var r in Model.Results) 
{
    string css = r.Result.Count() > 0 ? "fail" : "pass";

    <p class="@css"><strong>@r.Description</strong></p>

    @if(r.Result.Count() > 0) 
    {
        <p>Count: @r.Result.Count()</p>
        <table>
            <thead>
                <tr>
                    <th>ID</th><th>Title</th><th>Description</th>
                </tr>
            </thead>
            <tbody>
            @foreach(var e in r.Result) {
                <tr><td>@e.Id</td><td>@e.Title</td><td>@e.Description</td></tr>
            }
            </tbody>
        </table>
    }
}

我会在@if中遇到运行时错误:在“@”字符后面出现意外的“if”关键字。进入代码后,您不需要使用“@”作为“if”的结构前缀。

如果我删除@代码运行正常。我期望需要@因为它前面的HTML。更令我困惑的是,在嵌套的foreach之前我确实需要@。这里有什么规则?

2 个答案:

答案 0 :(得分:39)

在剃刀的任何括号内,它都需要匹配的开始和结束标记。这就是解析器的工作原理。

到目前为止,以下示例有效:

@for (var i = 0; i < 10; i++) { 
<p>
    @i.ToString()
</p>
}

这不是:

@for (var i = 0; i < 10; i++) { 
<p>
   @i.ToString()
</p>
@if (i == 2) { 
 <p>2</p>
}
}

要解决此问题,您可以将其置于<text>块中,如:

@for (var i = 0; i < 10; i++) { 
<text>
<p>
   @i.ToString()
</p>
@if (i == 2) { 
 <p>2</p>
}
</text>
}

所以在你的情况下会变成:

@foreach(var r in Model.Results) 
{
  @string css = r.Result.Count() > 0 ? "fail" : "pass";
<text>

  <p class="@css"><strong>@r.Description</strong></p>

  @if(r.Result.Count() > 0) 
  {
    <p>Count: @r.Result.Count()</p>
    <table>
        <thead>
            <tr>
                <th>ID</th><th>Title</th><th>Description</th>
            </tr>
        </thead>
        <tbody>
        @foreach(var e in r.Result) {
            <tr><td>@e.Id</td><td>@e.Title</td><td>@e.Description</td></tr>
        }
        </tbody>
    </table>
  }
</text>
}

答案 1 :(得分:5)

嵌套的foreach位于HTML内部(恰好位于其他代码中)。

要从标记转到代码,您需要@ 只有在直接嵌套代码块时才有必要。