1. 程式人生 > >ASP.Net MVC的ViewBag一個坑,不要跳進去

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

如鵬的學習管理系統是使用ASP.net MVC 5開發的,今天一個新版本釋出後網站出現一個Bug,學生在下拉列表中選中的項再載入顯示的時候發現仍然沒被選中。詳細一點說吧:
假如有這樣一個Action:
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方法了,在方法內部又把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最新課程