2011年2月25日 星期五

T-SQL Add/Drop Default Value

本篇文章主要是要透過撰寫T-SQL Script去異動一個欄位的預設值(Default Value)。


假設我們在Northwind這個資料庫中,有一個Table名為TestTable,其中有兩欄位MachineID與Name,如下圖:



如果你想要使用SQL 語法去異動欄位結構,則可使用下列語法:

1.將MachineID欄位設為允許Null值


   alter table TestTable alter column MachineID int null

2.為MachineID欄位新增一個default值=>1


   alter table TestTable ADD DEFAULT 1 FOR MachineID

現在要將上面兩項異動再還原回去,則可使用下列語法:

3.將MachineID欄位設不可為Null值


   alter table TestTable alter column MachineID int not null

4.移除MachineID欄位的default值須,先查詢該default的名稱,如下:



   use Northwind
   SELECT * FROM sys.objects


由上可知我們查出在sys.objects這個Table中,若欄位名稱為Type這個欄位的值為「D」時,表示該筆記錄為「Default Value」,該default的名稱為DF__TestTable__Machi__5812160E,接著使用下列語法來刪除MachineID欄位的default值。


   alter table TestTable DROP constraint DF__TestTable__Machi__5812160E

如果我們直接使用下列語法直接刪除Default Value,則會有下圖的錯誤產生。


   DROP DEFAULT DF__TestTable__Machi__5812160E


2011年1月28日 星期五

MVC3 Partial Output Cache - VaryByParam Demo

在本Blog前一篇文章中(MVC3 Partial Output Cache - Simple Demo),我們舉了一個簡單的例子,來說明MVC 3中的Partial Output Cache機制,那個例子在同一個網頁上顯示兩個時間字串,一個時間字串沒有cache,另一個時間字串有設定cache 10秒鐘,由那個例子可知道MVC 3 Partial Output Cache運作的基本方式與程式寫法。


但一般網站中都是佈滿各種資訊在一個web page中,有些需顯示即時資訊,有些靜態資訊則會設定cache,以加快網頁顯示速度,本篇文章中,我們以Northwind資料庫為例,針對查詢同一個產品類別(CategoryID)的結果,來進行Partial Output Cache:也就是透過設定VaryByParam的方式(本篇範例為VaryByParam=CategoryID),只要有網友查詢過A產品類別(CategoryID=A)之下所屬的所有產品,則查詢後的內容就會被cache起來,在設定的cache時間內(如Duration=100),若有網友又要查詢A這個產品類別(CategoryID=A),則伺服器不會再執行查詢程式,而是將cache中之前查過的相同內容直接輸出給第二次查詢同產品類別的網友,當然輸出部分,我們也是以之前文章所提到的Partial View()的方式來製作。我們以下先進行本範例展示,文章後半部再詳述各部份程式。


Demo


[02:17:11 PM] 點選/Product/QueryProduct這頁,可看到我們可下拉產品類別(CategoryID)來選擇我們要查詢的產品,如下圖:



[02:17:19 PM] 選擇CategoryID = Meat/Poultry,可看到目前該類別共有6個產品。如下圖:



[02:17:37 PM] 點選/Product/Create這頁,我們現在新增一個ProductName = KaKa,其CategoryID = Meat/Poultry,就是上個步驟所看到的那個產品類別。如下圖:



[02:17:49 PM] 再次回到/Product/QueryProduct這頁,如下圖選擇CategoryID = Meat/Poultry,可看到目前該類別仍維持有6個產品,伺服器並沒有再次執行查詢的動作,而是將Cache的內容直接回應(response)給查詢者,Cache機制產生效果了(OutputCache is working),而且又上方的時間仍然持續的被更新為最新時間(time is updated continually),所以Partial Output Cache 也產生效果了(Partial OutputCache is also working)。



[02:18:56 PM] 使用另外一個Browser,如下圖點選/Product/QueryProduct這頁,可看到目前該類別仍維持有6個產品,所以Cache機制持續生效中。



[02:19:10 PM] 時間過了我們設定的Cache時間(Duration=100秒),再次更新(refresh)一次/Product/QueryProduct這頁,可看到目前該類別已經變成7個產品,前面步驟新增的ProductName = KaKa這筆資料,現在已經被顯示出來,伺服器已經再次去執行查詢,並取回資料庫最新資料狀況,然後更新Cache內容。



下面我們詳述各部份程式如何撰寫。


Create a MVC3 Project


本範例中先建立一個MVC3的專案檔,並且配合Entity Framework Code First CTP5來連結與操作Northwind資料庫。建立好專案檔,加入EntityFramework.dll參考(Add reference),如下圖:



並加入資料庫連線到Web.config後(請參考之前文章這裡),請先建立Product這個Controller,並且ProductController之下,可去Create一個新Product的View與相關程式,這部份請參考一般MVC的作法,可參考Adding a Create Method and Create View


Model


這部分我們撰寫兩個cs檔案,用來處理與Category與Products相關的資料存取。
而QueryProductViewModel.cs則是上述Demo網頁(/Product/QueryProduct這頁)所用到的View Model描述。


1./Models/CategoryRepository.cs


   1:  public class CategoryRepository
   2:  {
   3:      Northwind northwind;
   4:      public CategoryRepository()
   5:      {
   6:          northwind = new Northwind();
   7:      }
   8:   
   9:      public IQueryable<Category> SelectAll()
  10:      {
  11:          return northwind.Categories;
  12:      }
  13:  }

2./Models/ProductRepository.cs


   1:  public class ProductRepository
   2:  {
   3:      Northwind northwind;
   4:      public ProductRepository()
   5:      {
   6:          northwind = new Northwind();
   7:      }
   8:   
   9:      public IEnumerable<Product> SelectAll()
  10:      {
  11:          return northwind.Products;
  12:      }
  13:   
  14:      public void Create(Product product)
  15:      {
  16:          northwind.Products.Add(product);
  17:          northwind.SaveChanges();
  18:      }
  19:   
  20:      public IQueryable<Product> Select(int id)
  21:      {
  22:          var list = from m in northwind.Products
  23:                     where m.CategoryID == id
  24:                     select m;
  25:          return list;
  26:      }
  27:  }

3./ViewModels/QueryProductViewModel.cs


   1:  public class QueryProductViewModel
   2:  {
   3:      public int CategoryID { get; set; }
   4:      public IQueryable<Product>  Products { get; set; }
   5:  }


Controller


這部分是本範例最重要的部份,我們在Product這個Controller中撰寫QueryProductPartialView這個Action,並且設定相關的OutputCache參數,這個Action,可以用來產生產生Partial Output Cache的Partial View。


1./Controllers/ProductController.cs


   1:  public class ProductController : Controller
   2:  {
   3:      ProductRepository productRep;
   4:      CategoryRepository categoryRep;
   5:      public ProductController()
   6:      {
   7:          productRep = new ProductRepository();
   8:          categoryRep = new CategoryRepository();
   9:      }
  10:   
  11:      public ActionResult QueryProduct()
  12:      {
  13:          ViewBag.Categories = categoryRep.SelectAll();
  14:          QueryProductViewModel qryPrd=new QueryProductViewModel();
  15:          return View(qryPrd);
  16:      }
  17:   
  18:      [HttpPost]
  19:      public ActionResult QueryProduct(QueryProductViewModel QryPrdView)
  20:      {
  21:          ViewBag.Categories = categoryRep.SelectAll();
  22:          QueryProductViewModel prds = new QueryProductViewModel()
  23:          {
  24:              CategoryID = QryPrdView.CategoryID
  25:          };
  26:          return View(prds);
  27:      }
  28:   
  29:      [OutputCache(Duration = 100, VaryByParam = "CategoryID")]
  30:      public ActionResult QueryProductPartialView(int CategoryID)
  31:      {
  32:          var result = productRep.Select(CategoryID);
  33:          QueryProductViewModel prds = new QueryProductViewModel()
  34:          {
  35:              CategoryID = CategoryID,
  36:              Products = result
  37:          };
  38:          return PartialView(prds);
  39:      }
  40:      
  41:      public ActionResult Create()
  42:      {
  43:      //請參考http://www.asp.net/mvc
  44:      } 
  45:   
  46:      [HttpPost]
  47:      public ActionResult Create(Product collection)
  48:      {
  49:      //請參考http://www.asp.net/mvc
  50:      }    
  51:  }


View


在查詢網頁上我們會產生一個可選擇不同產品類別(CategoryID)的DropDownList,然後下方會即時顯示查詢這屬於這類別的所有產品清單,顯示產品清單這部份,我們是用Partial View的方式來處理,所以我們也需撰寫QueryProductPartialView.cshtml,如下所示:


1./Views/Product/QueryProduct.cshtml


   1:  @model Mvc3Demo.ViewModels.QueryProductViewModel
   2:  <h2>QueryProduct</h2>
   3:   
   4:  @using (Html.BeginForm())
   5:  {
   6:      <div class="editor-field">Category Name:
   7:          @Html.DropDownListFor(model => model.CategoryID,
   8:              new SelectList(ViewBag.Categories,
   9:                  "CategoryID", "CategoryName"),
  10:              new { onchange = "this.form.submit();" })
  11:      </div>
  12:  }
  13:  @Html.Action("QueryProductPartialView", "Product",
  14:      new { CategoryID = Model.CategoryID })

2./Views/Product/QueryProductPartialView.cshtml


   1:  @model Mvc3Demo.ViewModels.QueryProductViewModel
   2:   
   3:  @if (Model.Products != null)
   4:  {
   5:  if (Model.Products.Count() > 0)
   6:  { 
   7:  <table>
   8:      <tr>
   9:          <th>Total:@Model.Products.Count()</th>
  10:          <th>
  11:              CategoryID
  12:          </th>
  13:          <th>
  14:              ProductName
  15:          </th>
  16:          <th>
  17:              UnitPrice
  18:          </th>
  19:      </tr>
  20:   
  21:  @foreach (var item in Model.Products)
  22:  {
  23:      <tr>
  24:          <td>
  25:              @Html.ActionLink("Edit", "Edit",
  26:                   new { id = item.ProductID })
  27:          </td>
  28:          <td>
  29:              @item.CategoryID
  30:          </td>
  31:          <td>
  32:              @item.ProductName
  33:          </td>
  34:          <td>
  35:              @String.Format("{0:F}", item.UnitPrice)
  36:          </td>
  37:      </tr>
  38:  }
  39:   
  40:  </table>
  41:  }
  42:  }

以上即是本範例的程式碼,Output Cache可幫助提供更快速的網頁瀏覽經驗,Partial Output Cache 更可彈性的提供部分內容的Cache機制。

2011年1月26日 星期三

MVC 3 Partial Output Cache - Simple Demo

在MVC 3中,我們可以使用PartialView()的方式來完成對網頁局部資料進行Output Cache,而不是全部網頁內容都Output Cache。

下面範例我們要建立兩個區域,兩個區域各顯示目前的時間字串,一個區域沒有進行Output Cache處理,所以當網頁refresh時,時間會一直被更新成最新時間;另外一個區域我們設定10秒鐘的Output Cache,所以雖然網頁一直被refresh,但時間字串經過10秒後,才被更新,故舊的時間字串被cache了10秒鐘,以下為實際範例程式碼描述。


建立Controller


我們建立一個MVC 3專案檔,並建立一個名為CacheDemoController的Controller,在其中我們建立一個可產生PartialView()的Action,就取名為PartialCache(),並且設定OutputCache的時間為10秒鐘。程式碼如下:



   1:  public class CacheDemoController : Controller
   2:  {
   3:      public ActionResult Index()
   4:      {
   5:          return View();
   6:      }
   7:   
   8:      [OutputCache(Duration = 10)]
   9:      public ActionResult PartialCache()
  10:      {
  11:          ViewBag.Time2 =
  12:              DateTime.Now.ToLongTimeString();
  13:          return PartialView();
  14:      } 
  15:  }

建立兩個View


這裡我們需要建立兩個View(),分別是Index.cshtml與PartialCache.cshtml。
Index.cshtml這個View用來顯示同一個網頁中,上半部時間字串內容沒有Cache;
下半時間字串內容被設定Cache.Index這個View內容如下:



   1:  <h2>OutputCache Demo</h2>
   2:  <p>--下行為沒有Cache區段(<b>No Cache</b>)</p>
   3:  <div class="bar1">@DateTime.Now.ToLongTimeString()
   4:  </div><br /><br />
   5:   
   6:  <p>--下行設定10秒鐘的Cache(<b>Partial Cache 10 mins</b>)
   7:  </p>
   8:  <div class="bar2">@Html.Action("PartialCache",
   9:      "CacheDemo",null)</div>

PartialCache.cshtml這個Partial View用來產生被設定Output Cache的時間字串內容,其內容就一行,如下:



   1:  <p>@ViewBag.Time2</p>

執行結果


我們執行上面建立好的MVC 3專案檔,可看到一開始時,上下兩個字串時間都一樣。如下圖:


隨後我們refresh這一頁,我們發現,上半部的時間字串已經被更新了,但下半部的時間字串仍然維持一開始的那個時間字串,由下半部的時間字串這可看到,這部份已經成功產生Output Cache部份網頁內容了,原來Partial View(由PartialCache.cshtml產生)這部份的內容並不會在Server端再被執行與產生。如下圖:


我們再持續refresh這一頁,上半部的時間字串也一直持續被更新了。過了10秒鐘,我們再refresh這一頁時,發現下半部時間字串現在被更新了,也就是上面的Partial View內容又自Server端執行一次(request),並將最新內容拋到(response)user端的Browser上了,如下圖:


有關MVC 3的Output Cache,基本上我們可參考Scott Guthrie的Announcing ASP.NET MVC 3這篇文章.

2011年1月22日 星期六

文章分類:MVC


2011-1-28 星期五

MVC3 Partial Output Cache - VaryByOaram Demo

在本Blog前一篇文章中(MVC3 Partial Output Cache - Simple Demo),我們舉了一個簡單的例子,來說明MVC 3中的Partial Output Cache機制,那個例子在同一個網頁上顯示兩個時間字串,一個時間字串沒有cache,另一個時間字串有設定cache 10秒鐘,由那個例子可知道MVC 3 Partial Output Cache運作的基本方式與程式寫法。


但一般網站中都是佈滿各種資訊在一個web page中,有些需顯示即時資訊,有些靜態資訊則會設定cache,以加快網頁顯示速度,本篇文章中,我們以Northwind資料庫為例,針對查詢同一個產品類別(CategoryID)的結果,來進行Partial Output Cache:也就是透過設定VaryByParam的方式(本篇範例為VaryByParam=CategoryID)...MORE



2011-1-26 星期三

MVC3 Partial Output Cache - Simple Demo

在MVC 3中,我們可以使用PartialView()的方式來完成對網頁局部資料進行Output Cache,而不是全部網頁內容都Output Cache。

下面範例我們要建立兩個區域,兩個區域各顯示目前的時間字串,一個區域沒有進行Output Cache處理,所以當網頁refresh時,時間會一直被更新成最新時間;另外一個區域我們設定10秒鐘的Output Cache,所以雖然網頁一直被refresh,但時間字串經過10秒後,才被更新,故舊的時間字串被cache了10秒鐘,以下為實際範例程式碼描述。...MORE



2011-1-22 星期日

MVC 專案範本可移除多餘web config描述

在減輕web server負荷以加快網站處理速度的議題中,我們通常可以透過移除沒有用到的元件(component)、模組(module)或是參考(reference),來減輕web server的負荷。這篇文章以MVC為範例,並說明基本上那些web config中沒有用到的模組是可以移除,讓web config看來更乾淨(Dry),進一步也減輕一些web server負荷。

我們在建立MVC專案時,若是以「Internet Application範本」來建立MVC專案,而非是以「空的專案範本」(Empty Template)來建立專案,
如下圖:...MORE



2010-12-21 星期二

MVC 與 HTML5

雖然MVC3 RC2已經發佈了(Announcing ASP.NET MVC 3 RC2),但是MVC本身對與HTML5的開發支援還是有待加強。好消息是在微軟的codeplex.com網站中,可下載在MVC中欲使用HTML5開發的輔助工具,名為MVC HTML 5 Toolkit。以下就實際舉一個MVC 2專案來說明如何使用這個HTML5的工具。...MORE



2010-10-10 星期日

.NET 4 + MVC2 設定真正 ValidateRequest = false 並回應更友善的錯誤頁面與錯誤訊息

因為.NET 4中,對惡意程式碼的欄截(AntiXSS),已經提升到BeginRquest的層面了,請看ASP.NET4 WhitePaper,若我們再搭配MVC2架構,想要「不啟用」ValidateRequest,則原先我們要在View頁面上自訂
...MORE



2010-10-03 星期日

Other Way For Setting Customer Error Page in MVC2

本文舉出3種方法去控制MVC2中,如何自訂Error Page,釐清在各個View中加入專屬於某個View的自訂Error page,或者是整個系統共用一個自訂Error Page的實際作法,一般微軟官方文件或其它技術Blog中並沒有針對這點加以整理與說明。
...MORE



2010-10-03 星期日

MVC2 Trim String

在MVC2中,加設我們有一個產品(Product)的資料需要維護,我們在Edit或者是Create的Form中,我們修改了一些欄位上的資料,我們希望可以將每個欄位多餘的空白字元去除(trim space),以免儲存回資料庫後,再次取出時,會產生錯誤,例如我們如果在Create的Form中,將產品的識別碼(PrdID)給輸入「A001  」,後面部小心多了一些空白字元,若是沒有在儲存回資料庫前,將這個欄位後面的空白去除掉,當這項產品儲存在資料庫後,我們再次輸入「A001」,要將這項產品編號的產品給查詢出來時,系統就會告訴我們找不到這項產品。因為資料庫存的是「A001  」,而非「A001」。本文基於MVC的架構,自View-Colltroller-Model這三個層次順序,由外圍到底層,說明如何簡單的使用既有的方法去達成去除字串後面多餘的空白字元。

...MORE


2010-08-03 星期二

如何在Html.TextBoxFor內設定Disable TextBox欄位

在一般Web Form應用中,我們可用TextBox控制項中的Enabled = "false" 來達成這個要求,但在MVC應用中,因為不採用控制項的方式,故我們需要在Html.TextBoxFor的屬性去標示。...MORE


2010-07-22 星期四

使用Entity Framework設定Complex Types-以MVC中使用stored procedure為範例

在MVC使用Entity Framework來存取資料庫中的Stored Procedure應用中,若是遇到Stored Procedure回應出來的是一組複雜資料欄位,而這組資料欄位是目前專案中沒有一個類別與之相對應,此時,我們可以另外建立一個新的類別,也可以使用Entity Framework中的Model Browser視窗中的Complex Types來建立這個新的資料欄位組合...MORE



2010-06-24 星期四

ASP.NET MVC2 如何使用多國語系資源檔

在ASP.NET MVC2中 , 我們想要在View或者是欄位驗證中 , 使用到 App_GlobalResources這個目錄下的多國語系資源檔 , 我們可以用下列方式來設定 .

1.假設目前有一個專案檔TOY , 且已經建立好App_GlobalResources這個目錄 , 我們點選該目錄後按右鍵選擇建立一個新的Resource檔案 :...MORE

MVC 專案範本可移除多餘web config描述

在減輕web server負荷以加快網站處理速度的議題中,我們通常可以透過移除沒有用到的元件(component)、模組(module)或是參考(reference),來減輕web server的負荷。這篇文章以MVC為範例,並說明基本上那些web config中沒有用到的模組是可以移除,讓web config看來更乾淨(Dry),進一步也減輕一些web server負荷。

我們在建立MVC專案時,若是以「Internet Application範本」來建立MVC專案,而非是以「空的專案範本」(Empty Template)來建立專案,
如下圖:







這樣建立起來的專案檔,我們可打開web.config檔案來看,我們可發現有很多「可能」不會用到的web config 描述,例如ASP.NET中內建的的Membership認證與授權機制。

因為一般網站專案,我們通常會自己建立授權機制,而不會去使用ASP.NET中內建的的Membership認證與授權機制,所以我們若真的沒用到時,可以大膽的將這些在web.config中不會用到的任何描述移除,以減輕Web Server的負荷,好比下列這些在web.config中的描述沒有使用時,都可以大膽將它們移除:










移除後,我們web.config看來「瘦」多了,網站執行起來,也不會有任何錯誤。當然,其他需用到的資源與元件,還是得乖乖的納入web.config中。

2011年1月20日 星期四

Entity Framework Code First 應用中如何撰寫Connection String

Entity Framework Code First 目前為CTP5版本這種方法,讓我們使用Entity Framework時,不用再產生一個EF檔案,可以藉由將EntityFramework.dll加入參考(Add reference),從此後,我們操作Entity Framework就好向操作一般元件一樣方便,當然,我們使用EF就是要存取資料庫中的資料,無論是使用SQL Server Compact Edition 4SQL Server Express或是正式的SQL Server版本,都需要加上一段ConnectionString。


但是之前使用Entity Framework,ConnectionString是VS 2010 幫我們加上去的,不需用手打到web.config上,現在使用Entity Framework Code First,我們只好乖乖的手動加到web.config的<connectionStrings>區段中,加的時候,千萬不要被之前自動產生的ConnectionString所影響,暫時不要使用「providerName="System.Data.EntityClient"」,而是使用「providerName="System.Data.SqlClient"」,才可以正常連到SQL Server。如果providerName使用System.Data.EntityClient.網站在動態執行時,則會出現下圖的「keyword not supported:'data source'」錯誤訊息。



所以web.config中,整段<connectionStrings>描述我們可如下撰寫:



   1:  <connectionStrings>
   2:    <add name="Northwind" 
   3:      connectionString="Data Source=192.168.100.101;
   4:      Initial Catalog=Northwind;
   5:      User ID=northwinduser;
   6:      Password=usernorthwind;" 
   7:      providerName="System.Data.SqlClient"/>
   8:  </connectionStrings>

經由上面的調整後,就可順利顯示出我們要的結果,如下圖,不會再有與ConnectionString相關的錯誤了。


我相信自動產生ConnectionString這一點小事,Entity Framework Code First未來版本一定會解決的,祝大家使用愉快!

利用SQL Server 工具自動產生提升SQL執行效能的的Index語法(Auto Create SQL Script of Index - Database Engine Tuning Advisor)

在提升網站速度這個議題上,資料庫能夠迅速回應前端網友的需求,也是一個很重要的部份,良好的索引(Index)設計,通常也可提升資料庫的處理速度,本篇文章即是使用MS SQL Server 2005中的「Database Engine Tuning Advisor」這個工具,進行提升SQL 語法執行效能。


先選定待提升SQL語法


先假設我們發現下面這段SQL 語法執行時,效能很差,我就希望能經由加上Index與其他語法補助,來提升這段sql的執行效能。


select * from systemeventlog where UserName like '%z000%'


使用工具


我們可在SQL Server管理介面上,將這段待執行的sql 語法給選取起來,並且點選「Database Engine Tuning Advisor」這個工具圖示,如下圖:



分析過程


接著,會出現待設定的視窗,如你可選擇想分析的資料庫,這裡我們使用預設,先不做任何修改。我們點選左側的綠色箭頭「開始分析」,開始分析這段語法如何增進執行效能。如下圖:


可以看到系統正在處理中如下圖:



產生可增進效能的SQL Script


分析結束後,我們點選結果頁中的「建議」這個頁籤,並且在索引建議的資料內,點開定義欄位下的那段建議的連結如下圖紅框處。如下圖:


系統接著出現一個小視窗「SQL指令碼預覽」如下圖紅框處,可將其中的語法copy下來,實際執行它,產生必要的Index,則可為我們增加最先前那段SQL語法的執行效能。在「建議」這個頁籤下方,我們可看到估計可改進約25%的效能,如下圖:



當然,增加Index也意味著資料儲存空間成本的增加,「網站效率提升」與「儲存空間成本的增加」兩者之間如何取捨,讀者可自行根據您系統上的要求的重點來加以考量。