ASP.Net MVC的ViewBag一個坑,不要跳進去

   如鵬的學習管理系統是使用ASP.net MVC 5開發的,今天一個新版本發佈後網站出現一個Bug,學生在下拉列表中選中的項再加載顯示的時候發現仍然沒被選中。詳細一點說吧:
假若有這樣一個Action:html

public ActionResult Index()
{
List<SelectListItem> persons = new List<SelectListItem>(); 
persons.Add(new SelectListItem { Text = "騰訊", Value = "qq" });
persons.Add(new SelectListItem { Text = "如鵬", Value = "rupeng", Selected = true });
ViewBag.persons = persons;
return View();
}

  Cshtml是這樣的:程序員

@Html.DropDownList("persons", (IEnumerable<SelectListItem>)ViewBag.persons)
生成的html是這樣的:數據庫

<select id="persons" name="persons">
<option value="qq">騰訊</option>
<option value="rupeng">如鵬</option>
</select>

   居然第二項沒有處於選中狀態,太詭異了吧!緩存

  只要把DropDownList第二個參數的"persons"改爲和」ViewBag.persons」的persons名字不同就能夠,好比:
@Html.DropDownList("persons1", (IEnumerable<SelectListItem>)ViewBag.persons)
   這樣就正確生成了:安全

<select id="persons1" name="persons1">
<option value="qq">騰訊</option>
<option selected="selected" value="rupeng">如鵬</option>
</select>

  

   好詭異!!!網絡


   咋辦?看源碼!
   DropDownList是定義在SelectExtensions擴展類中,DropDownList方法最終是調用SelectInternal方法,核心代碼是這一段:架構

if (!flag && obj == null && !string.IsNullOrEmpty(name))
{
obj = htmlHelper.ViewData.Eval(name);
}
if (obj != null)
{
selectList = SelectExtensions.GetSelectListWithDefaultValue(selectList, obj, allowMultiple);
}

  

這個name參數就是咱們傳遞給DropDownList的第一個參數"persons",上面代碼主要邏輯就是:首先到ViewData中查找名字爲"persons"的值,咱們知道ViewData和ViewBag是同樣的,因此htmlHelper.ViewData.Eval(name)取到的值就是咱們在Index中定義的List<SelectListItem>()集合。GetSelectListWithDefaultValue方法的代碼是:學習

private static IEnumerable<SelectListItem> GetSelectListWithDefaultValue(IEnumerable<SelectListItem> selectList, object defaultValue, bool allowMultiple)
{
IEnumerable enumerable= new object[]
{
defaultValue
};
IEnumerable<string> collection = 
from object value in enumerable
select Convert.ToString(value, CultureInfo.CurrentCulture);

HashSet<string> hashSet = new HashSet<string>(collection, StringComparer.OrdinalIgnoreCase);
List<SelectListItem> list = new List<SelectListItem>();
foreach (SelectListItem current in selectList)
{
current.Selected = ((current.Value != null) ? hashSet.Contains(current.Value) : hashSet.Contains(current.Text));
list.Add(current);
}
return list;
}

  

注意,咱們的List<SelectListItem>()集合被當成defaultValue參數傳遞給GetSelectListWithDefaultValue方法了(why?),在方法內部又把defaultValue給 Convert.ToString()一下,變成了」System.Collections.Generic.List`1[System.Web.Mvc.SelectListItem]」這麼一個玩意, GetSelectListWithDefaultValue的主要邏輯就是查找selectList中等於」System.Collections.Generic.List`1[System.Web.Mvc.SelectListItem]」的值,能找到纔算見了鬼呢!!!
通過上面的分析咱們還能夠知道,不能讓cshtml中DropDownList的第一個name參數和ViewBag中任何一個屬性重名,不然仍是會有問題,好比網站

public ActionResult Index()
{
List<SelectListItem> persons = new List<SelectListItem>(); 
persons.Add(new SelectListItem { Text = "騰訊", Value = "qq" });
persons.Add(new SelectListItem { Text = "如鵬", Value = "rupeng", Selected = true });
ViewBag.persons = persons;
ViewBag.persons1 = new string[] { };
return View();
}

  

Cshtml以下:
@Html.DropDownList("persons1", (IEnumerable<SelectListItem>)ViewBag.persons)搜索引擎

生成的html中第二條數據照樣不會被selected

不知道微軟爲何把DropDownList這麼簡單的一個東西搞的這麼複雜,正驗證了這句話「寫的越多,錯的越多」。固然也許微軟會給出理由說咱們用錯了,說「It’s not a bug,It’s a feature,by design」好吧!謝特!

 

如鵬網.Net培訓班正在報名,有網絡的地方就能夠參加如鵬網的學習,學完就能高薪就業,點擊此處瞭解

 

    三年前只要懂「三層架構」就能夠說「精通分層架構」;如今則須要懂IOC(AutoFac等)、CodeFirst、lambda、DTO等才值錢;

    三年前只要會SQLServer就能夠說本身「精通數據庫開發」;如今則需還須要掌握MySQL等開源數據庫才能說是「.Net開源」時代的程序員;

    三年前只要會進行用戶上傳內容的安全性處理便可;如今則須要熟悉雲存儲、CDN等才能在雲計算時代遊刃有餘;

    三年前只要掌握Lucene.Net就會說本身「熟悉站內搜索引擎開發」;如今你們都用ElasticSearch了,你還用Lucene.Net就太老土了;

    三年前發郵件仍是用SmtpClient;如今作大型網站發郵件必須用雲郵件引擎;

    三年前緩存就是Context.Cache;如今則是Redis、Memcached的天下;

    如鵬網再次引領.Net社區技術潮流!點擊此處瞭解如鵬網.Net最新課程

相關文章
相關標籤/搜索