日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

codesmith学习总结

發(fā)布時(shí)間:2025/4/14 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 codesmith学习总结 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

code smith 使用介紹


利用CodeSmith為SQL Server CE生成項(xiàng)目代碼
?
摘要:CodeSmith是很多.NET開發(fā)人員至愛的開發(fā)輔助工具,它能夠使開發(fā)人員從大量枯燥無味的重復(fù)編碼中解脫,集中精力解決實(shí)際業(yè)務(wù)問題和技術(shù)問題。本文將介紹如何將CodeSmith用于Windows Mobile項(xiàng)目,以SQL Server Compact Edition數(shù)據(jù)庫作為代碼生成的依據(jù)生成項(xiàng)目代碼。

什么是代碼生成器?
代碼產(chǎn)生器是產(chǎn)生式編程(Generative Programming)在實(shí)際應(yīng)用中的一種產(chǎn)物。它以系統(tǒng)建模為基礎(chǔ),通過抽象各種系統(tǒng)間的共性與特性,利用代碼模板和組件重用技術(shù),自動(dòng)產(chǎn)生滿足客戶需求的軟件產(chǎn)品。從而降低成本,減少軟件推向市場的時(shí)間,并且保證更好的產(chǎn)品質(zhì)量。最終取得規(guī)模經(jīng)濟(jì)和范圍經(jīng)濟(jì)的優(yōu)點(diǎn)。2004年我還在大學(xué)讀書的時(shí)候曾開發(fā)過一個(gè)基于模板引擎的代碼生成器——RapidTier,當(dāng)時(shí)這個(gè)作品還獲得了廣東省高校杯軟件設(shè)計(jì)大賽的二等獎(jiǎng)。在.NET的世界里,目前最有名的代碼生成器當(dāng)屬CodeSmith和MyGeneration。這兩款代碼生成器都是基于模板引擎的,使用起來方便靈活,而且模板資源非常豐富。

CodeSmith 簡介
CodeSmith 是一個(gè)基于模板的代碼生成器,你可以利用它來生成任何文本語言的代碼。CodeSmith 生成的代碼可以通過模板的屬性來定制。CodeSmith 模板的語法跟 ASP.NET 非常相似,你可以在模板中使用 C# 或 VB.NET 控制代碼的生成。通常,開發(fā)人員利用CodeSmith根據(jù)SQL Server或Oracle等大型數(shù)據(jù)庫生成各種項(xiàng)目代碼(如:存儲(chǔ)過程、數(shù)據(jù)訪問層、業(yè)務(wù)邏輯層、表現(xiàn)層等)。今天將向各位介紹如何利用 CodeSmith為移動(dòng)數(shù)據(jù)庫SQL Server Compact Edition生成Windows Mobile項(xiàng)目的代碼。

配置SchemaProvider
CodeSmith通過SchemaProvider來支持各種數(shù)據(jù)庫。在CodeSmith Professional 4.1.2安裝后,默認(rèn)只支持SQL Server、Oracle、Access、MySQL等數(shù)據(jù)庫類型。如果需要支持其他類型的數(shù)據(jù)庫,可以到網(wǎng)上搜索相關(guān)數(shù)據(jù)庫的 SchemaProvider實(shí)現(xiàn),或者自己手動(dòng)編寫代碼實(shí)現(xiàn)ISchemaProvider接口。
最近,我從網(wǎng)上找到了SQL Server Compact Edition(3.1)的SchemaProvider實(shí)現(xiàn),經(jīng)過代碼調(diào)整和調(diào)試,現(xiàn)在跟大家分享一下。可以從這里下載到這個(gè)SchemaProvider相關(guān)的DLL,下載后將其解壓,并將 SchemaExplorer.SqlCeSchemaProvider.dll和System.Data.SqlServerCe.dll文件復(fù)制到 CodeSmith的SchemaProvider目錄下(C:/Program Files/CodeSmith/v4.1/SchemaProviders)。

配置數(shù)據(jù)源
啟動(dòng)CodeSmith,點(diǎn)擊左邊停靠的Schema Explorer窗口上方點(diǎn)擊“Manage Data Sources”工具欄按鈕,打開Data Source Manager窗口。

Data Source Manager窗口打開后,點(diǎn)擊Add按鈕添加新的數(shù)據(jù)源。在Data Source窗口中,數(shù)據(jù)源名稱(Name)輸入“Northwind sqlce”,提供程序類型(Provider Type)選擇SqlCeSchemaProvider,連接語句輸入“data source=c:/Northwind.sdf”,假設(shè)你已經(jīng)將SQL Server Compact Edition自帶的Northwind.sdf數(shù)據(jù)庫復(fù)制到C盤根目錄了。

你可以通過點(diǎn)擊Test按鈕測試一下是否正常連接,然后點(diǎn)擊OK保存數(shù)據(jù)源,回到Data Source Manager窗口,現(xiàn)在看到多了一個(gè)叫“Northwind sqlce”的數(shù)據(jù)源了。

“Northwind sqlce”數(shù)據(jù)源已經(jīng)添加成功,點(diǎn)擊Close按鈕關(guān)閉Data Source Manager窗口。回到Schema Explorer窗口,展開“Northwind sqlce”數(shù)據(jù)源,瀏覽各個(gè)表的字段和屬性。

生成代碼
在CodeSmith右邊停靠的Template Explorer窗口打開自帶的模板“CodeSmith 4.1 Samples/ActiveSnippets/CSharp/TableProperties.cst”。這個(gè)模板可以從一個(gè)數(shù)據(jù)庫的表的所有字段生成對(duì)應(yīng)的實(shí)體類的所有屬性(Property)定義代碼。
模板被打開后,在CodeSmith右邊停靠的Properties窗口選擇SourceTable屬性,點(diǎn)擊旁邊的“...”按鈕瀏覽并選擇一個(gè)Northwind.sdf數(shù)據(jù)庫的表。這里我們選擇Categories表,并點(diǎn)擊Select按鈕確定。

現(xiàn)在可以按F5生成代碼了,生成的結(jié)果如下所示:

private int _categoryID;?

public int CategoryID
{
? ? get { return _categoryID; }
? ? set { _categoryID = value; }
}?

private string _categoryName;?

public string CategoryName
{
? ? get { return _categoryName; }
? ? set { _categoryName = value; }
}?

private string _description;?

public string Description
{
? ? get { return _description; }
? ? set { _description = value; }
}?

private System.Byte[] _picture;?

public System.Byte[] Picture
{
? ? get { return _picture; }
? ? set { _picture = value; }
}
你還可以選擇其他的表或其他模板生成代碼試一試,體驗(yàn)一下CodeSmith的強(qiáng)大之處。

總結(jié)
通過上面的介紹和示例演示,相信大家都認(rèn)同CodeSmith確實(shí)很好很強(qiáng)大。當(dāng)然,這里只是演示了一個(gè)很簡單的例子,你可以根據(jù)項(xiàng)目的實(shí)際需要自己編寫模板,按自己的方式去生成項(xiàng)目代碼。編寫模板最快捷的方法就是基于現(xiàn)有比較類似的模板進(jìn)行修改。CodeSmith之所以能夠用于SQL Server Compact Edition數(shù)據(jù)庫的代碼生成,除了前面提到的它通過SchemaProvider支持各種類型的數(shù)據(jù)庫,還有一點(diǎn)很重要的就是SQL Server Compact Edition支持桌面平臺(tái),如果是SQL Server Mobile就沒有辦法做到這一點(diǎn)了。
相關(guān)下載:SqlCeSchemaProvider.rar


http://upto.cnblogs.com/?

========

code Smith 使用學(xué)習(xí)

?CodeSmith在執(zhí)行模版時(shí)通過調(diào)用一些API來完成的,主要經(jīng)過了以下這幾步的操作:
? ? ? ? ?編譯一個(gè)模版
? ? ? ? ?顯示編譯錯(cuò)誤信息
? ? ? ? ?創(chuàng)建一個(gè)新的模版實(shí)例
? ? ? ? ?用元數(shù)據(jù)填充模版
? ? ? ? ?輸出結(jié)果
下面這段代碼顯示了這些操作:
?
CodeTemplateCompiler compiler = new CodeTemplateCompiler("..//..//StoredProcedures.cst");
compiler.Compile();
?
if (compiler.Errors.Count == 0)
{
? ? CodeTemplate template = compiler.CreateInstance();
?
? ? DatabaseSchema database = new DatabaseSchema(new SqlSchemaProvider(), @"Server=(local)/NetSDK;Database=Northwind;Integrated Security=true;");
? ? TableSchema table = database.Tables["Customers"];
?
? ? template.SetProperty("SourceTable", table);
? ? template.SetProperty("IncludeDrop", false);
? ? template.SetProperty("InsertPrefix", "Insert");
?
? ? template.Render(Console.Out);
}
else
{
? ? for (int i = 0; i < compiler.Errors.Count; i++)
? ? {
? ? ? ? Console.Error.WriteLine(compiler.Errors[i].ToString());
? ? }
}
?
如果你需要提供一個(gè)復(fù)雜的組合用戶界面來輸入元數(shù)據(jù),這時(shí)就要添加設(shè)計(jì)器的支持。換句話說,除此之外沒有別的辦法來輸入你自定義的元數(shù)據(jù)類型。添加設(shè)計(jì)器的支持,首先你要?jiǎng)?chuàng)建一個(gè)Editor作為自定義的類型,一個(gè)Editor其實(shí)就一個(gè)繼承于.NET 中的System.Drawing.Design.UITypeEditor類的子類。
安裝CodeSmith后在,在C:/Program File/CodeSmith/ SampleProjects 文件夾下有很多SampleCustomProperties的工程。例如:DropDownEditorProperty是一個(gè)把字符串和布爾類型的值結(jié)合在一起的元數(shù)據(jù),它提供了一個(gè)類DropDownEditorPropertyEditor繼承于System.Drawing.Design.UITypeEditor。
[Editor(typeof(CodeSmith.Samples.DropDownEditorPropertyEditor), typeof(System.Drawing.Design.UITypeEditor))]
public class DropDownEditorProperty
在使用的時(shí)候跟其它的元數(shù)據(jù)類型是一樣,不過別忘記添加對(duì)程序集的引用,引用CodeSmith默認(rèn)的是不認(rèn)識(shí)該類型的。
<%@ Property Name="DropDownEditorProperty" Type="CodeSmith.Samples.DropDownEditorProperty" Category="Options" Description="This property uses a custom dropdown editor." %>
<%@ Assembly Name="SampleCustomProperties" %>
當(dāng)用戶想要編輯DropDownEditProperty時(shí),單擊CodeSmith屬性面板將會(huì)顯示如下的自定義對(duì)話框:
?
在CodeSmith中,如果生成的代碼是SQL腳本,則可以在生成代碼完成時(shí)自動(dòng)執(zhí)行生成的代碼,也就是在生成的SQL腳本的同時(shí)在數(shù)據(jù)庫中創(chuàng)建新的對(duì)象。
用BaseTemplates.ScriptUtility對(duì)象提供ExecuteScript方法可以實(shí)現(xiàn),如果想在生成代碼完成后立即執(zhí)行生成的腳本,可以很方便的通過重載OnPostRender來實(shí)現(xiàn)。
在使用之前,先添加對(duì)下列程序集的引用:
<%@ Assembly Name="CodeSmith.BaseTemplates" %>
<%@ Import Namespace="CodeSmith.BaseTemplates" %>
看下面的這個(gè)例子:
?
protected override void OnPostRender(string result)?
{
? ? // execute the output on the same database as the source table.
? ? CodeSmith.BaseTemplates.ScriptResult scriptResult =?
? ? ?CodeSmith.BaseTemplates.ScriptUtility.ExecuteScript(this.SourceTable.Database.ConnectionString,?
? ? ?result, new System.Data.SqlClient.SqlInfoMessageEventHandler(cn_InfoMessage));?
? ? Trace.Write(scriptResult.ToString());
? ? base.OnPostRender(result);
}

在這個(gè)例子中SourceTable是一個(gè)類型為SchemaExplorer.TableSchema.的屬性,使用的時(shí)候需要調(diào)整部分代碼來獲取數(shù)據(jù)庫的連接以便在生成代碼完成后執(zhí)行腳本。
?
在CodeSmith中使用CodeTemplateInfo可以獲取當(dāng)前模版的一些信息:
屬性

返回值

CodeBehind ??

Gets the full path to the code-behind file for the template (or an empty string if there is no code-behind file).
ContentHashCode

Gets the hash code based on the template content and all template dependencies. ?

DateCreated

Gets the date the template was created.

DateModified

Gets the date the template was modified. ?

Description

Gets the description. ?

DirectoryName

Gets the name of the directory the template is located in. ?

FileName

Gets the name of the template file. ?

FullPath

Gets the full path to the template. ?

Language

Gets the template language. ?

TargetLanguage

Gets the target language. ?

看一下一個(gè)具體的使用例子:
?
<%@ CodeTemplate Language="VB" TargetLanguage="Text" Description="Demonstrates CodeTemplateInfo." %>
<% DumpInfo() %>
<script runat="template">
Public Sub DumpInfo()
? ? Response.WriteLine("Template: ? ? ? ?{0}", Me.CodeTemplateInfo.FileName)
? ? Response.WriteLine("Created: ? ? ? ? {0}", Me.CodeTemplateInfo.DateCreated)
? ? Response.WriteLine("Description: ? ? {0}", Me.CodeTemplateInfo.Description)
? ? Response.WriteLine("Location: ? ? ? ?{0}", Me.CodeTemplateInfo.FullPath)
? ? Response.WriteLine("Language: ? ? ? ?{0}", Me.CodeTemplateInfo.Language)
? ? Response.WriteLine("Target Language: {0}", Me.CodeTemplateInfo.TargetLanguage)
End Sub
</script>

執(zhí)行該模版輸出如下(環(huán)境不同,輸出也不同):
Template: ? ? ? ?CodeTemplateInfo.cst
Created: ? ? ? ? 6/29/2005 8:54:19 PM
Description: ? ? Demonstrates CodeTemplateInfo.
Location: ? ? ? ?C:/Program Files/CodeSmith/v3.0/SampleTemplates/Test/CodeTemplateInfo.cst
Language: ? ? ? ?VB
Target Language: Text
?
Progress對(duì)象可以在CodeSmith生成代碼時(shí)給用戶顯示一個(gè)進(jìn)度條,當(dāng)生成代碼的時(shí)間很長時(shí)非常有用。如果你使用的是CodeSmith Explorer,進(jìn)度條將顯示在Generate按鈕的左邊:

如果使用的是CodeSmith Studio,進(jìn)度條將顯示在狀態(tài)欄上:

使用Progress和在WinForm中使用進(jìn)度條差不多,需要設(shè)置它的最大值和步長:
this.Progress.MaximumValue = 25;
this.Progress.Step = 1;
如果想顯示出進(jìn)度,需要調(diào)用PerformStep方法:
this.Progress.PerformStep();
?
在CodeSmith中,以下幾個(gè)快捷鍵有助于我們快速輸入。
1.Ctrl + Shift + C
在空行上,按下Ctrl + Shift + C后將會(huì)錄入一個(gè)代碼塊。
<% ?%>
?2.Ctrl + Shift + Q
按下Ctrl + Shift + Q后錄入一個(gè)腳本塊。
?
<script runat="template">

</script>

3.Ctrl + Shift + V
對(duì)代碼塊反轉(zhuǎn),如有下面這樣一行代碼:
<% for(int i=0;i<10;i++){}%>
在兩個(gè)大括號(hào)之間按下Ctrl + Shift + V后,將變成如下代碼:
<% for(int i=0;i<10;i++){%> ?<%}%>
4.Ctrl + Shift + W
按下Ctrl + Shift + W后會(huì)錄入一個(gè)輸出的代碼塊:
<%= ?%>
注意:在使用快捷鍵的時(shí)候,如果想要把一段代碼之間放在錄入的標(biāo)記中間,首先選中這些代碼,再按下快捷鍵組合。比如我們有一段這樣的代碼,想把它放在<script>里面。
?
public enum CollectionTypeEnum
{
? Vector,
? HashTable,
? SortedList
}

public override void Render(TextWriter writer)
{
? ? StreamWriter fileWriter1 = new StreamWriter(@"C:/test1.cs", true);
? ? this.Response.AddTextWriter(fileWriter1);
? ? StreamWriter fileWriter2 = new StreamWriter(@"C:/test2.cs", true);
? ? ?this.Response.AddTextWriter(fileWriter2);
? ? base.Render(writer);
? ? fileWriter1.Close();
? ? fileWriter2.Close();
}

選中它,再按下Ctrl + Shift + Q后就會(huì)變成:
<script runat="template">
public enum CollectionTypeEnum
{
? Vector,
? HashTable,
? SortedList
}


public override void Render(TextWriter writer)

{
? ? StreamWriter fileWriter1 = new StreamWriter(@"C:/test1.cs", true);
? ? this.Response.AddTextWriter(fileWriter1);
? ? StreamWriter fileWriter2 = new StreamWriter(@"C:/test2.cs", true);
? ? ?this.Response.AddTextWriter(fileWriter2);
? ? base.Render(writer);
? ? fileWriter1.Close();
? ? fileWriter2.Close();
}

</script>
========

CodeSmith的基礎(chǔ)模版類


基礎(chǔ)模版類
類型描述:?
Batch ? ? ?
OutputFileCodeTemplate ?模版通過繼承此類能夠在生成過程中把他們的輸出保存到文件中?
ScriptError ? ?在腳本執(zhí)行中出現(xiàn)一個(gè)錯(cuò)誤
ScriptErrorCollection ??
ScriptResult ? ?一個(gè)腳本的運(yùn)行結(jié)果包含一些已經(jīng)發(fā)生的錯(cuò)誤
ScriptUtility ? ?這個(gè)類能用來在數(shù)據(jù)庫上執(zhí)行Sql腳本。
SqlCodeTemplate ? 繼承此類的模版當(dāng)從一個(gè)Sql數(shù)據(jù)源生成代碼時(shí)能夠獲得很多有用的幫助方法
StringUtility ? ?多種處理string型的方法
各類型下的成員屬性及方法
Batch Class
屬性
Content ??
LineCount ??
StartLineNumber?
方法
Finalize 在一個(gè)對(duì)象再次創(chuàng)建之前獲得空閑資源并且執(zhí)行其他的清空操作
MemberwiseClone 建立現(xiàn)有對(duì)象的副本
OutputFileCodeTemplate Class
屬性
CodeTemplateInfo ?得到當(dāng)前模版的信息
OutputFile ?此屬性用來指定一個(gè)保存模版輸出的輸出文件名
Progress ? 提供一種方式匯報(bào)模版的執(zhí)行進(jìn)程
Response ? 模版輸出返回流。此屬性可以在程序中寫出流
State ? 模版實(shí)例的狀態(tài)
ValidationErrors ?得到模版的錯(cuò)誤
方法
CopyPropertiesTo ?把匹配的屬性拷貝到另一個(gè)代碼模版實(shí)例中
GetCodeTemplateInstance 重載,得到指定模版的實(shí)例
GetFileName ?為模版的輸出得到一個(gè)默認(rèn)的名字
GetProperties ? 得到模版的所有屬性
GetProperty ? 得到模版的指定屬性
GetRequiredProperties ?得到模版上所有必要的屬性
GetType ? 得到當(dāng)前實(shí)例類型
ParseDefaultValue ?解析屬性的默認(rèn)值
SavePropertiesToXml ?以XML保存屬性
SavePropertiesToXmlFile 保存屬性到一個(gè)XML文檔
SetProperty ? 重載,保存指定的屬性值
ToString
ScriptError Class
屬性
方法
Finalize 在一個(gè)對(duì)象再次創(chuàng)建之前獲得空閑資源并且執(zhí)行其他的清空操作
MemberwiseClone 建立現(xiàn)有對(duì)象的副本
ScriptUtility Class
屬性
ConnectionString 執(zhí)行腳本時(shí)使用此連接字符串
Script ? 執(zhí)行的腳本
方法
ExecuteScript ?重載,執(zhí)行腳本
SqlCodeTemplate Class
屬性
CodeTemplateInfo ?得到當(dāng)前模版的信息
OutputFile ?此屬性用來指定一個(gè)保存模版輸出的輸出文件名
Progress ? 提供一種方式匯報(bào)模版的執(zhí)行進(jìn)程
Response ? 模版輸出返回流。此屬性可以在程序中寫出流
State ? 模版實(shí)例的狀態(tài)
ValidationErrors ?得到模版的錯(cuò)誤
方法
CopyPropertiesTo ?把匹配的屬性拷貝到另一個(gè)代碼模版實(shí)例中
GetCamelCaseName Returns a camel cased name from the given identifier.?
GetCodeTemplateInstance 重載,得到指定模版的實(shí)例
GetCSharpVariableType 基于給定列返回C#的變量類型
GetFileName ?為模版的輸出得到一個(gè)默認(rèn)的名字
GetMemberVariableDeclarationStatement
? ?重載,返回C#成員變量聲明語句
GetMemberVariableDefaultValue
? ?基于一個(gè)列的數(shù)據(jù)類型返回一個(gè)默認(rèn)值
GetMemberVariableName 為一個(gè)給定標(biāo)示返回一個(gè)C#成員變量名
GetProperties ? 得到模版的所有屬性
GetProperty ? 得到模版的指定屬性
GetPropertyName ?返回指定列的公有屬性的名字
GetReaderMethod ?Returns the name of the typed reader method for a given column.?
GetRequiredProperties ?得到模版上所有必要的屬性
GetSpacedName ?Returns a spaced out version of the identifier.
GetSqlDbType ?返回一個(gè)給定列的SqlDbType
GetSqlParameterExtraParams
? ?為ADO的參數(shù)聲明生成額外的參數(shù)
GetSqlParameterStatement
? ?重載,返回給定列的T-Sql的參數(shù)聲明
GetSqlParameterStatements
? ?重載,給指定列加一個(gè)參數(shù)到ADO對(duì)象生成一個(gè)指定聲明(Generates an assignment statement that adds a parameter to a ADO object for the given column. )
GetValidateStatements 基于某列生成一組確認(rèn)聲明
IncludeEmptyCheck 確定一個(gè)給定列是否可以為空
IncludeMaxLengthCheck 確定一個(gè)給定列的類型是否需要最大長度的定義
IsUserDefinedType 確定是否一個(gè)給定列用了一個(gè)UDT(用戶定義類型)
ParseDefaultValue ?解析屬性的默認(rèn)值
SavePropertiesToXml ?以XML保存屬性
SavePropertiesToXmlFile 保存屬性到一個(gè)XML文檔
SetProperty ? 重載,保存指定的屬性值
========

CodeSmith 使用實(shí)例



?StringCollection提供了一種集合的輸入方式,在代碼中,可以用Array的方式來引用。在使用這個(gè)類之前,在模版中我們必須添加對(duì)CodeSmith.CustomProperties程序集的引用:
<%@ Assembly Name="CodeSmith.CustomProperties" %>
添加完程序集之后,我們就可以使用StringCollection在腳本塊中定義一個(gè)屬性:
<%@ Property Name="List" Type="CodeSmith.CustomProperties.StringCollection" Category="Custom" Description="This is a sample StringCollection" %>
執(zhí)行該模版時(shí),這個(gè)屬性將在屬性窗體中顯示為一個(gè)按鈕:
單擊按鈕,將會(huì)彈出一個(gè)String Collection Editor對(duì)話框:
當(dāng)然也可以直接在屬性窗口中編輯StringCollection。
模版代碼如下:
<%@ CodeTemplate Language="C#" TargetLanguage="C#" %>
<%@ Assembly Name="CodeSmith.CustomProperties" %>
<%@ Property Name="List" Type="CodeSmith.CustomProperties.StringCollection" Category="Custom" Description="This is a sample StringCollection" %>
using System;
namespace Test
{ ? ? ??
? ? ? ? ?/** <summary>
? ? ? ? ?/// ? ? 測試StringCollection
? ? ? ? ?/// </summary>
? ? ? ? ?public class Test
? ? ? ? ?{
? ? ? ? ? ? ? ? ? ?public static void Main(string[] args)
? ? ? ? ? ? ? ? ? ?{
? ? ? ? ? ? ? ? ? ? ? ? ? ? <%for(int i = 0;i<List.Count;i++){%>
? ? ? ? ? ? ? ? ? ? ? ? ? ? Console.WriteLine(<%=List[i]%>);
? ? ? ? ? ? ? ? ? ? ? ? ? ? <%}%>
? ? ? ? ? ? ? ? ? ?}
? ? ? ? ?}
}
?
成后的代碼:
using System;
namespace Test
{ ? ? ??
? ? ? ? ?/** <summary>
? ? ? ? ?/// ? ? 測試StringCollection
? ? ? ? ?/// </summary>
? ? ? ? ?public class Test
? ? ? ? ?{
? ? ? ? ? ? ? ? ? ?public static void Main(string[] args)
? ? ? ? ? ? ? ? ? ?{
? ? ? ? ? ? ? ? ? ? ? ? ? ? Console.WriteLine(Apples);
? ? ? ? ? ? ? ? ? ? ? ? ? ? Console.WriteLine(Fish);
? ? ? ? ? ? ? ? ? ?}
? ? ? ? ?}
}
? StringCollection的重要屬性和方法:
公共屬性
名稱
?
描述
Count
?
獲取StringCollection中包含的字符串的數(shù)目
IsReadOnly
?
獲取用于指示StringCollection是否為只讀的值
IsSynchronized
?
獲取一個(gè)值,該值指示對(duì)StringCollection 的訪問是否為同步的(線程安全的)
Item
?
獲取或設(shè)置指定索引處的元素。在C# 中,該屬性為 StringCollection 類的索引器
SyncRoot
?
獲取可用于同步對(duì)StringCollection 的訪問的對(duì)象
公共方法
名稱
?
描述
Add
?
將字符串添加到 StringCollection 的末尾
AddRange
?
將字符串?dāng)?shù)組的元素復(fù)制到 StringCollection 的末尾
Clear
?
移除 StringCollection 中的所有字符串
Contains
?
確定指定的字符串是否在 StringCollection 中
CopyTo
?
從目標(biāo)數(shù)組的指定索引處開始,將全部 StringCollection 值復(fù)制到一維字符串?dāng)?shù)組中
IndexOf
?
搜索指定的字符串并返回 StringCollection 內(nèi)的第一個(gè)匹配項(xiàng)的從零開始的索引
Insert
?
將字符串插入 StringCollection 中的指定索引處
Remove
?
從 StringCollection 中移除特定字符串的第一個(gè)匹配項(xiàng)
RemoveAt
?
移除 StringCollection 的指定索引處的字符串
========

CodeSmith 實(shí)例2

?
FileNameEditor類給我們提供了在CodeSmith屬性面板中彈出打開或保存文件對(duì)話框的方式,在使用時(shí),首先在模版中得添加對(duì)程序集CodeSmith.CustomProperties的引用。然后就可以在模版中定義一個(gè)屬性來使用FileNameEditor:
?
?1<script runat="template">
?2
?3private string _userFileName = @"c:/temp/test.txt";
?4
?5?
?6
?7[Editor(typeof(FileNameEditor), typeof(System.Drawing.Design.UITypeEditor)),
?8
?9Category("Custom"), Description("User selected file.")]
10
11?
12
13public string UserFileName
14
15{
16
17 ? ? ?get {return _userFileName;}
18
19 ? ? ?set {_userFileName= value;}
20
21}
22
23</script>
24
25
當(dāng)我們執(zhí)行該模版時(shí),在屬性面板中同樣顯示為一個(gè)按鈕:


單擊該按鈕,彈出一個(gè)保存文件的對(duì)話框:


我們也可以通過FileDialogAttribute來自定義彈出的對(duì)話框,修改模版為:
?
?1private string _openFileName = @"c:/temp/test.txt";
?2
?3
?4[Editor(typeof(FileNameEditor), typeof(System.Drawing.Design.UITypeEditor)),
?5
?6FileDialogAttribute(FileDialogType.Open, Title="Select Input File"),
?7
?8Category("Custom"), Description("User selected file.")]
?9
10?
11
12public string OpenFileName
13
14{
15
16 ? ? ?get {return _openFileName;}
17
18 ? ? ?set {_openFileName= value;}
19
20}
21
22
彈出的對(duì)話框如下所示:


當(dāng)我們想用一個(gè)文件夾的名稱來代替文件時(shí),可以使用FolderNameEditor類。
?
?1<%@ Assembly Name="System.Design" %>
?2<script runat="template">
?3private string _outputDirectory = @"c:/temp";
?4[Editor(typeof(System.Windows.Forms.Design.FolderNameEditor), typeof(System.Drawing.Design.UITypeEditor)),
?5Category("Custom"), Description("Output directory.")]
?6public string OutputDirectory
?7{
?8 ? ? ? get {return _outputDirectory;}
?9 ? ? ? set {_outputDirectory= value;}
10}
11</script>
12
13
FileNameEditor重要方法和屬性介紹:
公共方法:
名稱
描述
EditValue
使用由 GetEditStyle 方法提供的編輯器樣式編輯指定的對(duì)象
GetEditStyle
獲取 EditValue 方法所使用的編輯樣式
========

CodeSmith應(yīng)用實(shí)例(一)



版權(quán)聲明:本文為博主原創(chuàng)文章,未經(jīng)博主允許不得轉(zhuǎn)載。


一、一個(gè)簡單的例子
?
? ? ? ?這個(gè)例子僅是一個(gè)簡單的應(yīng)用,在我翻譯并學(xué)習(xí)完CodeSmith的英文幫助文檔后,對(duì)CodeSmith有了一定的了解,開始著手編寫一些CodeSmith應(yīng)用模板,今天按照最早提到的例子自行編寫了一個(gè)基于表的添加存儲(chǔ)過程的生成模板。具體語法前面基礎(chǔ)中已做過詳細(xì)解釋這里僅是一個(gè)小綜合應(yīng)用的例子,望對(duì)大家學(xué)習(xí)CodeSmith有很好的幫助。我的同事也寫了幾個(gè)CodeSmith的技巧的文章http://terrylee.cnblogs.com/大家有空去看看,寫的很不錯(cuò)哦,都是針對(duì)于CodeSmith自定義屬性編寫的東東:)
?1<%@ CodeTemplate Language="C#" TargetLanguage="T-SQL" Description="Create a procedure which have insert function base on a table." %>
?2<%@ Assembly Name="SchemaExplorer" %>
?3<%@ Import Namespace="SchemaExplorer" %>
?4<%@ Property Name="SourceTable" Type="SchemaExplorer.TableSchema" Category="DataTable" Description="Table that the stored procedures should be based on." %>
?5<%@ Property Name="Author" Type="String" Category="Context" Description="The author for this procedure."%>
?6<%@ Property Name="Description" Type="String" Category="Context" Description="The description for this procedure."%>
?7<script runat="template">
?8public string GetSqlParameterStatement(ColumnSchema column)
?9{
10 ? ?string param = "@" + column.Name + " " + column.NativeType;
11 ? ?switch (column.DataType)
12 ? ?{
13 ? ? ? ?case DbType.Decimal:
14 ? ? ? ?{
15 ? ? ? ? ? ?param += "(" + column.Precision + ", " + column.Scale + ")";
16 ? ? ? ? ? ?break;
17 ? ? ? ?}
18 ? ? ? ?default:
19 ? ? ? ?{
20 ? ? ? ? ? ?if (column.Size > 0)
21 ? ? ? ? ? ?{
22 ? ? ? ? ? ? ? ?param += "(" + column.Size + ")";
23 ? ? ? ? ? ?}
24 ? ? ? ? ? ?break;
25 ? ? ? ?}
26 ? ?}
27 ? ?return param;
28}
29</script>
30CREATE PROCEDURE dbo.<%=SourceTable.Name %>Insert
31/*
32==================================================
33Author:<%= Author %>
34CreatedTime:<%= System.DateTime.Now.ToShortDateString() %>
35Description:<%= Description %>
36==================================================
37*/
38<% for (int i = 0; i < SourceTable.Columns.Count; i++) { %>
39<%= GetSqlParameterStatement(SourceTable.Columns[i]) %><% if (i < SourceTable.Columns.Count - 1) { %>,<% } %> ? ?<% if (SourceTable.Columns[i].Description != "") { %>--<%= SourceTable.Columns[i].Description %><% } %>
40<% } %>
41AS
42Insert Into [<%= SourceTable.Name %>]?
43(
44<% for (int i = 0; i < SourceTable.Columns.Count; i++) { %>
45[<%= SourceTable.Columns[i].Name %>]<% if (i < SourceTable.Columns.Count - 1) { %>,<% } %> ? ?<% if (SourceTable.Columns[i].Description != "") { %>--<%= SourceTable.Columns[i].Description %><% } %>
46<% } %>
47)
48Values
49(
50<% for (int i = 0; i < SourceTable.Columns.Count; i++) { %>
51@<%= SourceTable.Columns[i].Name %><% if (i < SourceTable.Columns.Count - 1) { %>,<% } %>
52<% } %>
53)
二、具有刪除功能的模板


? ? ? ? 今天又根據(jù)CodeSmith的幾個(gè)基本組件寫出了基于表生成刪除功能的存儲(chǔ)過程代碼生成模板。
? ? ? ? 昨天覺得添加的存儲(chǔ)過程模板寫的比較簡單,今天準(zhǔn)備詳細(xì)介紹一下這個(gè)刪除的模板。
? ? ? ? 首先介紹我們使用到的一個(gè)教本函數(shù)GetSqlParameterStatement(ColumnSchema column),其函數(shù)代碼如下:


?1public string GetSqlParameterStatement(ColumnSchema column)
?2{
?3 ? ?string param = "@" + column.Name + " " + column.NativeType;
?4 ? ?switch (column.DataType)
?5 ? ?{
?6 ? ? ? ?case DbType.Decimal:
?7 ? ? ? ?{
?8 ? ? ? ? ? ?param += "(" + column.Precision + ", " + column.Scale + ")";
?9 ? ? ? ? ? ?break;
10 ? ? ? ?}
11 ? ? ? ?default:
12 ? ? ? ?{
13 ? ? ? ? ? ?if (column.Size > 0)
14 ? ? ? ? ? ?{
15 ? ? ? ? ? ? ? ?param += "(" + column.Size + ")";
16 ? ? ? ? ? ?}
17 ? ? ? ? ? ?break;
18 ? ? ? ?}
19 ? ?}
20 ? ?return param;
21}


? ? ? ? 大家可以看到,這個(gè)函數(shù)需要傳入一個(gè)ColumnSchema類型的參數(shù),它代表一個(gè)數(shù)據(jù)表中的列,并且是一個(gè)列,然后根據(jù)ColumnSchema這個(gè)類具有的屬性,對(duì)傳入的列進(jìn)行一些操作然后返回我們生成存儲(chǔ)過程時(shí)需要的代碼。
? ? ? ? 首先介紹一下ColumnSchema的一些常用屬性,如下表:?
屬性Property


描述Description


AllowDBNull


是否允許空值NULL


Database


通過DatabaseSchema對(duì)象得到當(dāng)前列所屬的數(shù)據(jù)庫


DataType


此數(shù)據(jù)對(duì)象的數(shù)據(jù)類型


Description


當(dāng)前對(duì)象的描述


ExtendedProperties


用來存儲(chǔ)SchemaObject的其他附加信息


IsForeignKeyMember


當(dāng)前列是否為外鍵


IsPrimaryKeyMember


當(dāng)前列是否為主鍵


IsUnique


當(dāng)前列是否唯一


Name


列的名稱


NativeType


列定義的數(shù)據(jù)類型


Precision


數(shù)據(jù)對(duì)象的精度


Scale


數(shù)據(jù)對(duì)象的范圍(個(gè)人理解為需要保留小數(shù)的范圍)


Size


數(shù)據(jù)對(duì)象的大小(例如:字符串長度為10)


SystemType


數(shù)據(jù)對(duì)象的系統(tǒng)類型


Table


當(dāng)前列所屬的數(shù)據(jù)表


? ? ? ? 下面為我們首先要生成存儲(chǔ)過程,要自動(dòng)生成的代碼分成了紅、綠、藍(lán)三部分。
CREATE PROCEDURE dbo.CustomersDelete
/*
==================================================
Author:Bear-Study-Hard
CreatedTime:2005-12-28
Description:Delete a record from table Customers
==================================================
*/
@CustomerID nchar(5) --客戶ID
AS
Delete From [Customers]
Where
[CustomerID] = @CustomerID


? ? 我們的這個(gè)腳本函數(shù)就是要實(shí)現(xiàn)拼出紅色的部分,GetSqlParameterStatement函數(shù)接收到ColumnSchema類型的參數(shù)后,從其Name屬性和NativeType屬性拼出@CustomerID nchar部分,然后由于不同的數(shù)據(jù)類型尤其是數(shù)值類型和字符串類型的區(qū)別,會(huì)導(dǎo)致數(shù)據(jù)類型的大小會(huì)有所不同,這里僅對(duì)Decimal的數(shù)據(jù)類型進(jìn)行了判斷(Numeric和float等均需要這種處理),然后根據(jù)Precision屬性得到精度并通過Scale屬性得到了需要保留小數(shù)的范圍。如果傳出的為非Decimal類型字段則直接通過Size屬性取出其大小即可。得到了(5)部分。最后的注釋是為了生成的存儲(chǔ)過程解讀性好加上的,使用的是Description屬性。
? ? 剩下的綠色部分和藍(lán)色部分生成時(shí)比較簡單,請(qǐng)各位自行學(xué)習(xí)。模板代碼為:


?1<%@ CodeTemplate Language="C#" TargetLanguage="T-SQL" Description="Create a procedure which have delete function base on a table.Must use PrimaryKey to delete a record." %>
?2<%@ Assembly Name="SchemaExplorer" %>
?3<%@ Import Namespace="SchemaExplorer" %>
?4<%@ Property Name="SourceTable" Type="SchemaExplorer.TableSchema" Category="DataTable" Description="Table that the stored procedures should be based on." %>
?5<%@ Property Name="Author" Type="String" Category="Context" Description="The author for this procedure." Optional="true"%>
?6<%@ Property Name="Description" Type="String" Category="Context" Description="The description for this procedure." Optional="true"%>
?7<script runat="template">
?8public string GetSqlParameterStatement(ColumnSchema column)
?9{
10 ? ?string param = "@" + column.Name + " " + column.NativeType;
11 ? ?switch (column.DataType)
12 ? ?{
13 ? ? ? ?case DbType.Decimal:
14 ? ? ? ?{
15 ? ? ? ? ? ?param += "(" + column.Precision + ", " + column.Scale + ")";
16 ? ? ? ? ? ?break;
17 ? ? ? ?}
18 ? ? ? ?default:
19 ? ? ? ?{
20 ? ? ? ? ? ?if (column.Size > 0)
21 ? ? ? ? ? ?{
22 ? ? ? ? ? ? ? ?param += "(" + column.Size + ")";
23 ? ? ? ? ? ?}
24 ? ? ? ? ? ?break;
25 ? ? ? ?}
26 ? ?}
27 ? ?return param;
28}
29</script>
30CREATE PROCEDURE dbo.<%=SourceTable.Name %>Delete
31/*
32==================================================
33Author:<%= Author %>
34CreatedTime:<%= System.DateTime.Now.ToShortDateString() %>
35Description:<%= Description %>
36==================================================
37*/
38<% for (int i = 0; i < SourceTable.PrimaryKey.MemberColumns.Count; i++) { %>
39<%= GetSqlParameterStatement(SourceTable.PrimaryKey.MemberColumns[i]) %><% if (i < SourceTable.PrimaryKey.MemberColumns.Count - 1) { %>,<% } %> ? ?<% if (SourceTable.Columns[i].Description != "") { %>--<%= SourceTable.Columns[i].Description %><% } %>
40<% } %>
41AS
42Delete From [<%= SourceTable.Name %>]?
43Where
44<% for (int i = 0; i < SourceTable.PrimaryKey.MemberColumns.Count; i++) { %>
45<% if (i > 0) { %>AND <% } %>[<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>] = @<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>
46<% } %>


? ? 如果有問題我會(huì)盡力幫助大家解決的,共同提高^_^


本文永久地址: http://www.livebaby.cn/blog/u/meil/archives/2007/984.html
?
========

CodeSmith使用基礎(chǔ)教程 四 — 控制臺(tái)與屬性編輯器



七、CodeSmith控制臺(tái)指南。


很多人僅僅知道CodeSmith像一個(gè)圖形應(yīng)用程序,或者可能是一個(gè)Visual Studio的附件,但是通過CodeSmith的控制臺(tái)應(yīng)用程序還有好多其他的使用方法。控制臺(tái)應(yīng)用程序是很有價(jià)值的,因?yàn)榭梢酝ㄟ^它去生成腳本,或者其他一些自動(dòng)工具。這篇文檔的目的就是要告訴你怎樣使用它的控制臺(tái)應(yīng)用程序并且如何去定義變量和參數(shù)。


Basic Usage


大多數(shù)情況下是用控制臺(tái)應(yīng)用程序來創(chuàng)建一個(gè)模板,一個(gè)屬性文件,然后保存輸出的文件。這有一個(gè)很好的例子介紹將合并模版的處理過程放到一個(gè)過程中,就像使用NAnt工具。


首先我們要確定完成一個(gè)什么樣的模版,為這個(gè)模板創(chuàng)建一個(gè)什么樣的XML屬性文件。XML屬性文件提供在執(zhí)行模版是需要的各個(gè)屬性。生成一個(gè)屬性文件最簡單的方法是在CodeSmith Explorer中打開一個(gè)模版,填寫屬性,點(diǎn)擊生成按鈕generate,然后再點(diǎn)擊Save Property Set XML按鈕。這個(gè)按鈕會(huì)在點(diǎn)擊完生成按鈕后找到,在Save Output和Copy Output按鈕旁邊。然后系統(tǒng)提示輸入保存XML屬性文件的文件名,下面看一個(gè)ArrayList.cst模版創(chuàng)建的XML屬性文件。


?1<?xml version="1.0" encoding="us-ascii"?>
?2<codeSmith>
?3 ? ? ? ? ? ?<propertySet>
?4 ? ? ? ? ? ? ? ? ? ? ? ?<property name="Accessibility">Public</property>
?5 ? ? ? ? ? ? ? ? ? ? ? ?<property name="ClassName">PersonArray</property>
?6 ? ? ? ? ? ? ? ? ? ? ? ?<property name="ItemType">Person</property>
?7 ? ? ? ? ? ? ? ? ? ? ? ?<property name="ItemValueType">False</property>
?8 ? ? ? ? ? ? ? ? ? ? ? ?<property name="ItemCustomSearch">False</property>
?9 ? ? ? ? ? ? ? ? ? ? ? ?<property name="KeyName">PersonID</property>
10 ? ? ? ? ? ? ? ? ? ? ? ?<property name="KeyType">int</property>
11 ? ? ? ? ? ? ? ? ? ? ? ?<property name="IncludeInterfaces">True</property>
12 ? ? ? ? ? ? ? ? ? ? ? ?<property name="IncludeNamespaces">False</property>
13 ? ? ? ? ? ?</propertySet>
14</codeSmith>
就像看到的一樣,也可以手動(dòng)創(chuàng)建這個(gè)文件,但是使用CodeSmith Explorer會(huì)更簡便。


現(xiàn)在我們有了這個(gè)XML文件,我們繼續(xù)看一下如何去執(zhí)行這個(gè)模版并是用控制臺(tái)工具保存結(jié)果。首先我們需要是用/template參數(shù)去聲明我們要是用的模版,像這樣:


C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst


在這個(gè)例子中我們使用了ArrayList.cst模版,它存儲(chǔ)在本地的Samples/Collections文件夾下。下一步我們要去聲明我們?cè)谧詈笠徊叫枰獎(jiǎng)?chuàng)建的XML文件,我們是用/propertyset參數(shù)去實(shí)現(xiàn)。


C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst ?/propertyset:PersonArray.xml


這個(gè)/property參數(shù)用來指定我們的XML屬性文件。最后一個(gè)我們需要用的參數(shù)是/output參數(shù),用來指定輸出怎樣被保存。


C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst /propertyset:PersonArray.xml /out:test.cs


使用/out參數(shù)指定將結(jié)果輸出到一個(gè)叫test.cs文件中保存。執(zhí)行這個(gè)命令后,模板將開始運(yùn)行,使用屬性文件將結(jié)果輸出到test.cs文件保存。


這是大多數(shù)情況下有效使用控制臺(tái)。


Merging Output


在各種代碼生成中最大的挑戰(zhàn)就是將生成的代碼和開發(fā)人員編寫或修改的代碼區(qū)分開。控制臺(tái)對(duì)這個(gè)問題提供了一個(gè)有效的獨(dú)特的解決方案,使用一個(gè)指定的參數(shù)在當(dāng)前已存在的代碼文件中需要將模板生成的代碼添加的地方指定一塊區(qū)域。


下面是一個(gè)簡單的代碼文件,包含了我們要添加生成代碼的區(qū)域。


1using System;
2
3namespace Entities
4{
5 ? ? ?GeneratedOrderEntity ?
9}
我們的目標(biāo)是將DatabaseSchema/BusinessObject.cst模版生成的代碼添加到類文件的GeneratedOrderEntity區(qū)域中。和上一個(gè)例子一樣,使用CodeSmith console控制臺(tái)應(yīng)用程序執(zhí)行這個(gè)模版,但是這次要使用另一個(gè)參數(shù)merge。


C:/Program Files/CodeSmith/v3.0>cs /template:Samples/DatabaseSchema/BusinessObject.cst /propertyset:OrderEntity.xml /out:OrderEntity.cs /merge:InsertRegion= "RegionName=Sample Generated Region;Language=C#;"


使用merge參數(shù)我們可以指定區(qū)域的名稱,在這個(gè)例子中是GeneratedOrderEntity,然后控制臺(tái)應(yīng)用程序?qū)?zhí)行模版,并將結(jié)果添加到這個(gè)區(qū)域中。我們來看一下執(zhí)行完這個(gè)指令后生成的代碼。


?1using System;
?2
?3namespace Infozerk.AuthServices.UnitTestSuite
?4{
?5 ? ? ?GeneratedOrderEntity就像看到的一樣,Order類被添加到了我們指定的區(qū)域中。在代碼文件中使用merge參數(shù)生成的內(nèi)容在其他部分被修改或手寫后很容易重新再次生成而不會(huì)產(chǎn)生影響。
?
參數(shù)介紹Parameter Reference


Specifying Output


/out:<file>


指定從模版創(chuàng)建的輸出文件的名稱。


/out:default


指定這個(gè)文件被默認(rèn)保存成模版是用的名稱。


/merge:<mergetype>=<init>


指定模版輸出的區(qū)域。可以簡寫為/m


Specifying Input


/template:<file>


選擇要執(zhí)行的模版,簡寫為/t


/propertyset:<file>


生成代碼時(shí)需要使用的XML屬性文件。簡寫為/p


Compiler Options


/debug[+|-]


指定模版需要包含的調(diào)試信息。(允許在運(yùn)行模版時(shí)進(jìn)行調(diào)試)


/tempfiles[+|-]


指定保留臨時(shí)文件。(如果在臨時(shí)文件上調(diào)試也可以)


Miscellaneous


/help


顯示幫助信息。


/nologo


禁止生成器版權(quán)信息。


八、編寫CodeSmith自定義屬性的編輯器(Writing Custom Property Editors)


? ? ? ? 當(dāng)你開始編寫自定義的CodeSmith模板時(shí),很可能對(duì)于使用它的strings或integers屬性很滿意,但有時(shí)你會(huì)發(fā)現(xiàn)需要?jiǎng)?chuàng)建一個(gè)不同類型的屬性,可能是一個(gè)自定義的類型或者是.NET framework中但是在屬性面板中沒有提供的類型。在模板中去作這些很簡單,但是怎樣指定一個(gè)類型在運(yùn)行模板時(shí)顯示在屬性面板中呢?例如創(chuàng)建了一個(gè)Person類并且具有很多不同的屬性,但是卻沒有辦法讓用戶去組裝這個(gè)類……除非創(chuàng)建一個(gè)自定義屬性編輯器。


? ? ? ? 屬性面板提供了方法去編寫自定義的屬性編輯器,當(dāng)用戶在面板上選擇一個(gè)屬性時(shí)可以激發(fā)相應(yīng)的方法,同時(shí)也可以通過編寫代碼實(shí)現(xiàn)提示用戶輸入一個(gè)必要的值。下面我們舉個(gè)例子,創(chuàng)建一個(gè)接受組建的屬性并且是用影射循環(huán)貫串所有的類,是所有的類都可以使用它和它的方法,去創(chuàng)建一個(gè)NUnit測試基礎(chǔ)。(這句翻譯的不好,原文:As an example we are going to build a template which accepts an assembly as a property and then using reflection loops through all of the classes, and the methods of those classes, to build NUnit test stubs.)


? ? ? ? 首先,我們來關(guān)注一下模板的組件屬性,暫且不看自定義編寫的代碼。模板的第一部分是一些聲明定義和屬性。將屬性放在腳本標(biāo)簽中代替使用屬性聲明,在下一部分將看到這樣做的必要。


?1<%@ CodeTemplate Language="C#" TargetLanguage="C#" Description="Builds a class for each class in the assembly, and a test stub for every method." %>
?2
?3<%@ Import NameSpace="System.Reflection" %>
?4
?5<script runat="template">
?6
?7private Assembly assembly;
?8
?9public Assembly AssemblyToLoad
10{
11 ? ? ?get{return assembly;}
12 ? ? ?set{assembly = value;}
13}
14
15</script>
?


? ? ? ? 然后我們?yōu)榻M建assembly中的每一個(gè)類創(chuàng)建一個(gè)類,為每一個(gè)類創(chuàng)建他的方法。然后直接將模板的輸出內(nèi)容放入Visual Studio.NET,然后在編寫組件的單元測試時(shí)使用向?qū)А?


?1using System;
?2using NUnit.Framework;
?3
?4<%
?5 ? ? ?foreach(Type T in AssemblyToLoad.GetTypes())
?6 ? ? ?{
?7 ? ? ? ? ? ?if(T.IsClass)
?8 ? ? ? ? ? ?{
?9 ? ? ? ? ? ? ? ? ?%>
10
11 ? ? ? ? ? ? ? ? ?[TestFixture]
12 ? ? ? ? ? ? ? ? ?public class <%=T.Name%>Tests
13 ? ? ? ? ? ? ? ? ?{
14 ? ? ? ? ? ? ? ? ?<%
15 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?MethodInfo[] methods = T.GetMethods ( BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static ?);
16 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?foreach(MethodInfo M in methods)
17 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?{
18 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?%>
19
20 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?[Test]
21 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?public void <%=M.Name%>Test
22 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?{
23 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?//TODO Write this test
24 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?} ? ? ? ? ? ? ? ??
25 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?<%
26 ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?}
27
28 ? ? ? ? ? ? ? ? ?%>}<%
29 ? ? ? ? ? ?}
30 ? ? ?}
31%>


?/Files/Bear-Study-Hard/AssemblyHelper.zip


? ? ? ? 首先我們需要?jiǎng)?chuàng)建一個(gè)繼承UITypeEditor的類。


1public class AssemblyFilePicker : UITypeEditor
2{
3 ? ? ?public AssemblyFilePicker(): base()
4 ? ? ?{
5 ? ? ?}
6}
?
? ? ? ? 關(guān)于UITypeEditor的說明請(qǐng)大家參看MSDN或Visual Studio.NET自帶幫助中的說明,其中有詳細(xì)的例子。


? ? ? ? 然后我們需要重載UITypeEditor類的兩個(gè)不同的方法。第一個(gè)需要重載點(diǎn)的方法是GetEditStyle,這個(gè)方法是告訴屬性面板對(duì)于當(dāng)前類型是用什么類型的編輯器,在這個(gè)例子中我們?cè)O(shè)置編輯類型為Modal。這樣大家可以在該屬性格子的右邊看到一個(gè)小按鈕,它將引發(fā)一個(gè)對(duì)話框等模式的對(duì)話(trigger a modal dialog)。這是我們的GetEditStyle方法:


1public override UITypeEditorEditStyle GetEditStyle(ITypeDescriptorContext context)?
2{
3 ? ? ?return UITypeEditorEditStyle.Modal;
4}
?




? ? ? ? 其中的Modal為顯示一個(gè)省略號(hào)按鈕。


? ? ? ? 首先我們要從當(dāng)前的服務(wù)和控件中得到一個(gè)參考,有了控件的參考我們可以通過它轉(zhuǎn)到ShowDialog方法。(原文:First we need to get a reference to the current service and control, we need the reference to the control so we can pass it to the ShowDialog method.) ? ? ? ? 然后我們創(chuàng)建一個(gè)openFileDialog類并填入適合的屬性。 ? ? ? ?然后我們通過控件的參考(reference)將對(duì)話框顯示給用戶。 ? ? ? ?下一步我們檢查用戶是否點(diǎn)擊了OK按鈕,如果點(diǎn)擊了,通過文件選擇對(duì)話框選擇文件后使用LoadForm方法加載這個(gè)組件,最后返回這個(gè)值。 ? ? ? ?這個(gè)值將被放在屬性面板中并可以被模板讀取,但是需要注意,在我們作這個(gè)之前要將組件import引入到模板中,并在模板中用一對(duì)屬性聲明。


?


1IWindowsFormsEditorService editorService = (IWindowsFormsEditorService)provider.GetService(typeof(IWindowsFormsEditorService));
2Control editorControl = editorService as Control;
3
4if (editorControl != null)?
5{
?


1OpenFileDialog openFileDialog = new OpenFileDialog(); ? ? ? ? ? ? ? ? ? ? ? ??
2
3openFileDialog.CheckFileExists = true;
4openFileDialog.DefaultExt = ".dll";
5openFileDialog.Multiselect = false;
6openFileDialog.Title = "Select an Assembly:";
7openFileDialog.Filter = "Assembly Files | *.dll";
?


1DialogResult result = openFileDialog.ShowDialog(editorControl);
?


?1if (result == DialogResult.OK)
?2 ? ? ? ? ? ?{
?3Assembly assembly = Assembly.LoadFrom( openFileDialog.FileName ) ;
?4 ? ? ? ? ? ? ? ? ?value = assembly;?
?5 ? ? ? ? ? ?}
?6 ? ? ? ? ? ?else
?7 ? ? ? ? ? ?{
?8 ? ? ? ? ? ? ? ? ?value = null;
?9 ? ? ? ? ? ?}
10 ? ? ?}
11}
12
13return value;
14}
?


? ? ? ? 加載這個(gè)模板我們僅需將這個(gè)組件assembly與模板放在同一目錄下,然后再模板中加入下面兩行代碼。


1<%@ Assembly Name="AssemblyHelper" %>
2<%@ Import NameSpace="AssemblyHelper" %>
?


? ? ? ? 需要重載的另一個(gè)方法是EditValue方法,當(dāng)用戶電擊屬性時(shí)會(huì)調(diào)用這個(gè)方法。按照我們需要加載的組件類型需要?jiǎng)?chuàng)建一個(gè)打開文件對(duì)話框(open file dialog)然后捕獲這個(gè)對(duì)話框,在屬性格子中返回對(duì)話框的結(jié)果。


1public override object EditValue(ITypeDescriptorContext context, IServiceProvider provider, object value)
2{
3
4if (provider != null)?
5{
? ? ? ? 這個(gè)模板僅僅可以編譯通過,但是由于我們編寫顯示了一個(gè)類型屬性面板并不知道如何去操作它,所以我們沒有辦法自定義指定組件在加載時(shí)想要加載的組件。


? ? ? ? 我們需要?jiǎng)?chuàng)建一個(gè)UITypeEditor,這是一個(gè)建立屬性面板是用的特殊屬性的類。UITypeEditor需要?jiǎng)?chuàng)建在一個(gè)和模板分離的組件中,我們是用Visual Studio創(chuàng)建這個(gè)類。




本文永久地址: http://www.livebaby.cn/blog/u/meil/archives/2007/982.html
========

實(shí)例演示CodeSmith與ECC開發(fā)模式結(jié)合---快速建站(一)

http://blog.csdn.net/breakersam/article/details/1551935
?
前言
? ? ? ?其實(shí)本文所說的思路產(chǎn)生以很久以前,本人也曾將用于多個(gè)小型項(xiàng)目開發(fā)之中,用下來的整體說還是比較滿意。按理說,自己既然用了那么多次也應(yīng)寫些心得或是總結(jié)出來吧,無奈一是自己太懶二是文字功底有限,只好罷了!拖到了現(xiàn)在,方提指敲寫,也不知成文怎樣,由大家評(píng)說吧。
? ? ? 本文作為開發(fā)筆記以流水的方式敲寫,其中不討論開發(fā)模式之間的優(yōu)劣;如大家有好的建議也希望直面地提出來,我呢,以求進(jìn)步!
文中所舉的例子:ASP.NET (Net1.1)+ Access 基礎(chǔ)上所搭建。
?
CodeSmith概述
? ? ? ?CodeSmith中文我不知叫什么,不太好解釋,必竟這是“工業(yè)化”的東西,有點(diǎn)抽象,顧名思義(Code Smith 代碼工匠),就是生成代碼的東西;CodeSmith作用在于生成代碼模板,減少手工書寫代碼的工作,提高生產(chǎn)效率,增加剝奪收入。習(xí)慣上,我們項(xiàng)目組里在應(yīng)用CodeSmith有這著這樣的感受:C#的語法,ASP的寫法。不過話也得說回來,在ASP時(shí)代,我們沒有這樣的“艷遇”,那時(shí)代碼的生成,只能在自己編寫ASP Generator里生成,并且編寫過程中也沒有這么好的IDE支持。
? ? ? ? 或許有人會(huì)這樣認(rèn)為,既然有了CodeSmith,那么好多邏輯不是一下子就可以生成了嗎?干嘛要用什么ECC,煩!事實(shí)上,這話有部分多,也有大部分不對(duì)!是的,CodeSmith能快速地根據(jù)模板生成大量的代碼,但不有能生成所有的代碼,因?yàn)樵谝粋€(gè)邏輯必須根據(jù)具體的應(yīng)用環(huán)境情況下隨時(shí)變動(dòng)邏輯的代碼,其沒有任何的意義可言。平時(shí)在開發(fā)過程中,我們也對(duì)沒有接觸過的CodeSmith的伙伴這樣說:Code Smith 主要的是生成DAO(特別是與數(shù)據(jù)庫的表中粒度一一對(duì)應(yīng)的書寫中)與相對(duì)穩(wěn)定的應(yīng)用對(duì)象邏輯代碼,其不是萬能的!生成的代碼必須是軟件開發(fā)過程中所已經(jīng)協(xié)商好的開發(fā)模式的反映,而不是什么都要CodeSmith生成,否則生成的代碼,誰也看不懂,更不用說維護(hù)了或是項(xiàng)目組內(nèi)成員的學(xué)習(xí)了。
?
? ? ? ?下面我就簡單提一下CodeSmith初學(xué)者經(jīng)常提到了幾個(gè)問題,至于CodeSmith具體是怎樣應(yīng)用的,我們可以查看其自帶的學(xué)習(xí)例子或是在網(wǎng)上搜索,教程一大把。如下:
1.CodeSmith怎樣與數(shù)據(jù)庫結(jié)合
CodeTemplate里本身已經(jīng)提供了訪問了數(shù)據(jù)庫的屬性:
///
數(shù)據(jù)庫的連接
?<%@ Property Name="SourceTable" ? ? ? ? ? ? ? ? Type="SchemaExplorer.TableSchema" Category="1. Context"
? ? ? ?Description="Table that the stored procedures should be based on." %>
在DataSource里Provider type t選擇ADOXschemaProvider,輸入類似的ConnectionString:Provider=Microsoft.Jet.OLEDB.4.0;Data Source=C:/開發(fā)項(xiàng)目/基于ACCESS的小型系統(tǒng).NET/BreakerSimply/DataBase/BreakerSimply.mdb
//
表的訪問
//writeable columns Data
#region IsWritableColumns
public string IsWritableColumns()
{
//把ID 排除
? ? ? ?string sTemp ="";
? ? ? ?foreach(ColumnSchema column in SourceTable.Columns)
? ? ? ?{
? ? ? ? ? ? ? ? ? ? ?if (column.Name.ToLower()=="id")
? ? ? ? ? ? ? ? ? ? ?{
? ? ? ? ? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? ? ? ?else
? ? ? ? ? ? ? ? ? ? ?{
? ? ? ? ? ? ? ? ? ? ? ? ? ? sTemp +=this.GetPropertyName(column)+",";
? ? ? ? ? ? ? ? ? ? ?}
? ? ? ?}
? ? return sTemp;
}
#endregion
?
public string GetNativeTypeString(ColumnSchema column)
{
? ? ? ?string sColumnType = "";
? ? ? ?string sTemp = column.NativeType.ToString();
? ? ? ?//Debug.Print(sTemp);
? ? ? ?//Response.WriteLine("{0}={1}", column.Name, sTemp);
? ? ? ?//Debugger.Break();
? ? ? ?sTemp = sTemp.ToLower();
? ? ? ?switch(sTemp)
? ? ? ?{
? ? ? ? ? ? ? case "advarwchar":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.VarWChar";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "adinteger":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Integer";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "addate":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Date";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "adlongvarwchar":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.VarWChar";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "adboolean":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Boolean";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "adcurrency":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Currency";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "float":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Float";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "image":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Image";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "int":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Int";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "money":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Money";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "nchar":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.NChar";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "ntext":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.NText";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "nvarchar":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.NVarChar";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "real":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Real";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "smalldatetime":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.SmallDateTime";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "smallint":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.SmallInt";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "smallmoney":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.SmallMoney";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "text":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Text";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "timestamp":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Timestamp";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "tinyint":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.TinyInt";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "uniqueidentifier":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.UniqueIdentifier";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "varbinary":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.VarBinary";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "varchar":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.VarChar";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? case "variantariant":
? ? ? ? ? ? ? ? ? ? ?sColumnType = "OleDbType.Variantariant";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ? ? ? ? default:
? ? ? ? ? ? ? ? ? ? ?sColumnType = "";
? ? ? ? ? ? ? ? ? ? ?break;
? ? ? ?}
? ? ? ?return sColumnType;
}
?
2.CodeSmith生成后的代碼存放。
如下
<script runat="template">
#region Template Property Script
private string _outputDirectory = String.Empty;
?
[Editor(typeof(System.Windows.Forms.Design.FolderNameEditor), typeof(System.Drawing.Design.UITypeEditor))]
public string OutputFolder
{
?get
?{
? ? if (_outputDirectory.Length == 0)
? ? ? ?{
? ? ? ? ? ? ? string sOutputFolder = this.CodeTemplateInfo.DirectoryName;
? ? ? ? ? ? ? if(!sOutputFolder.EndsWith("//"))
? ? ? ? ? ? ? ? ? ? ?sOutputFolder += "//";
? ? ? ? ? ? ?
? ? ? ? ? ? ? sOutputFolder += "Tempout//Model";
? ? ? ? ? ? ? if(!System.IO.Directory.Exists(sOutputFolder))
? ? ? ? ? ? ? {
? ? ? ? ? ? ? ? ? ? ?System.IO.Directory.CreateDirectory(sOutputFolder);
? ? ? ? ? ? ? }
? ? ? ? ? ? ? return sOutputFolder;
? ? ? ?}
? ? return _outputDirectory;
?}
?set
?{
?if (value.EndsWith("//")) value = value.Substring(0, value.Length - 1);
?_outputDirectory = value;
?}
}
#endregion
</script>
CodeSmith軟件自帶了很多很好的例子,我也是從上面學(xué)的~~。
什么ECC
? ? ? ECC由Engine、Class、Collection三部分構(gòu)成,Engine實(shí)際上是個(gè)控制器,負(fù)責(zé)怎樣創(chuàng)建返回Class,Class表示現(xiàn)實(shí)世界的實(shí)體,具備詳細(xì)的屬性,Collection就是裝載Class的集合容器了; ECC 有人稱之為“與實(shí)現(xiàn)無關(guān)無關(guān)的應(yīng)用框架”,或許這有點(diǎn)夸大之嫌,什么叫做“與實(shí)現(xiàn)無關(guān)”,暈死。ECC確實(shí)提供了松散耦合的體系結(jié)構(gòu),它的出現(xiàn)使被操作的對(duì)像之間更顯得獨(dú)立,更為清晰,并且在很大程度上也降低了業(yè)務(wù)與具體邏輯的耦合
在ASP.NET中,我們還得考慮這么個(gè)問題,數(shù)據(jù)控件的數(shù)據(jù)綁定問題:控件支持CollectionBase的綁定,支持自定義行為(特別跨對(duì)像之間的數(shù)據(jù)操作);這些都能在ECC的模式里得到良好的解決。
實(shí)例中的ECC架構(gòu)
? ? ?本文將以一個(gè)具體的例子進(jìn)行說明:(本想貼些圖,但發(fā)現(xiàn)貼了好幾次都不成功,只好罷了)
? ? ?這個(gè)例子主要是實(shí)現(xiàn)文章的管理,如分類與相關(guān)CRUD,也就是平時(shí)所說的做了個(gè)小小的新聞發(fā)布系統(tǒng):欄目表(ArticleColumn) 與 Article表,兩者是一對(duì)多的關(guān)系
ArticleColumn表結(jié)構(gòu)
ID ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 自動(dòng)編號(hào)
UUID ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?UUID?
Name ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 欄目名 ? ?
Class ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 級(jí)數(shù)
Order ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?排序
CreatedTime ? ? ? ? ? ? ? ? ? ?創(chuàng)建時(shí)間
IsActive ? ? ? ? ? ? ? ? ? ? ? ? ? 是否啟用
Article表結(jié)構(gòu)
ID ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 自動(dòng)編號(hào)
UUID ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?UUID?
ColumnID ? ? ? ? ? ? ? ? ? ? ? ?欄目ID ? ? ? ? ? ? ??
Title ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 標(biāo)題 ??
Content ? ? ? ? ? ? ? ? ? ? ? ? ? 內(nèi)容
Author ? ? ? ? ? ? ? ? ? ? ? ? ? ? 作者
CreatedTime ? ? ? ? ? ? ? ? ? ?創(chuàng)建時(shí)間
IsActive ? ? ? ? ? ? ? ? ? ? ? ? ? 是否啟用
? ? ? 一般情況下,我是按這樣的工作流程:
? ? ?1. ? ? ? ? ? ? ?分析所要設(shè)計(jì)的系統(tǒng)有幾個(gè)可以稱為“對(duì)像”的元素;
? ? ?2. ? ? ? ? ? ? ?接著,用toghter畫一下設(shè)計(jì)下邏輯圖,生成初步的“原始代碼”;
? ? ?3. ? ? ? ? ? ? ?在“原始代碼”,添血加肉,完成初步的邏輯;
? ? ?4. ? ? ? ? ? ? ?以上面的代碼為骨架,在CodeSmith編寫相應(yīng)的代碼。
? ? ?以純粹文章功能模塊為例:
ArticleMD (與數(shù)據(jù)庫的ARTICLE表建立一一的映射,將reader讀出數(shù)據(jù)裝載入Article)
Article (繼承MD,處理關(guān)聯(lián)的對(duì)像數(shù)據(jù),如在這里根據(jù)文章的ID獲取該文章的欄目)
ArticleDAO (數(shù)據(jù)庫訪問邏輯,原始的CRUD操作邏輯)
ArticleBL (繼承DAO,根據(jù)業(yè)務(wù)需要,對(duì)CRUD的操作進(jìn)行進(jìn)一步的封裝)
ArticleCollection 基于CollectionBase,創(chuàng)建Aritlce集合;這個(gè)類可以從MSDN的例子獲得。 
具體的碼如下:
========

codesmith資料

http://www.cnblogs.com/Terrylee/archive/2005/12/28/306254.html
CodeSmith開發(fā)系列資料總結(jié)


http://www.tuicool.com/articles/2EbIra
CodeSmith自己動(dòng)手寫模板


http://www.cnblogs.com/whitewolf/archive/2010/09/27/1836729.html
CodeSmith模板引擎系列-目錄


http://www.cnblogs.com/wintersun/p/3971072.html
軟件代碼生成之Codesmith模板.netTiers


http://www.cnblogs.com/knowledgesea/p/5016077.html
CodeSmith模板代碼生成實(shí)戰(zhàn)詳解


http://blog.csdn.net/gxiangzi/article/details/6865619
CodeSmith .NET三層架構(gòu)模板

總結(jié)

以上是生活随笔為你收集整理的codesmith学习总结的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

国产精品区二区三区日本 | 亚洲干视频在线观看 | 波多野结衣视频一区二区 | 国产色黄网站 | 成人免费视频网址 | 精品在线观看视频 | 超碰人人91 | 久久国产高清视频 | 欧美一级电影在线观看 | 欧美日韩一区三区 | 久久99精品国产麻豆婷婷 | 日本中文在线观看 | 91大神电影 | 精品久久久久久亚洲综合网 | 97超碰超碰久久福利超碰 | 日本69hd | 激情偷乱人伦小说视频在线观看 | 日韩高清dvd | 伊人丁香| 欧美精品一区在线发布 | 亚洲一区二区三区在线看 | 最近日本韩国中文字幕 | 成人资源在线播放 | 黄色日批网站 | 久久精品网站免费观看 | 九热在线 | 日韩在线播放视频 | 日韩在线观看高清 | 在线色视频小说 | 九色精品免费永久在线 | 久热免费在线观看 | 亚州国产精品 | 91精品一区二区三区蜜臀 | 亚洲精品播放 | 特级西西人体444是什么意思 | 丝袜美腿在线播放 | 亚洲天天综合 | 国产精品一区二区无线 | 日韩理论 | www.神马久久| 久久99国产综合精品免费 | 一级性视频 | 99久久婷婷| 国产成人精品在线观看 | 日韩羞羞 | 国产1区2区3区在线 亚洲自拍偷拍色图 | 久久精品国产一区二区三区 | 蜜臀av夜夜澡人人爽人人 | 免费在线视频一区二区 | 欧美91在线 | 色综合久久久网 | 精品国产伦一区二区三区观看说明 | 日本中文字幕系列 | 欧美影院久久 | 日韩精品在线一区 | 久久久伦理 | 99这里都是精品 | 国产视频18 | 日日婷婷夜日日天干 | 色婷婷婷 | 最近日本韩国中文字幕 | 欧美激情视频久久 | 高清在线一区二区 | 日韩xxxxxxxxx | 久久综合久久综合久久 | 免费日韩一区 | 欧美日韩亚洲精品在线 | 午夜精品久久久久久中宇69 | 国产亚洲一区二区三区 | 免费男女网站 | 99久久久国产精品免费99 | 日本午夜免费福利视频 | 毛片.com| 亚洲免费不卡 | 国产麻豆视频在线观看 | 国产电影一区二区三区四区 | 亚洲一级电影在线观看 | 日韩大片免费在线观看 | 国产 欧美 日本 | 狠狠色狠狠色综合日日小说 | www.夜夜爽 | 精品国产99国产精品 | 亚洲精品字幕在线观看 | 91人人澡人人爽 | 日韩三级.com | 99精品欧美一区二区蜜桃免费 | 91在线九色 | www好男人| 国产欧美精品一区aⅴ影院 99视频国产精品免费观看 | 久久久一本精品99久久精品66 | 99精品国产成人一区二区 | 亚洲国产日本 | 正在播放国产一区二区 | 99精品热视频只有精品10 | 国产精品视频免费在线观看 | 日韩av偷拍 | 国产福利一区二区三区视频 | 懂色av一区二区在线播放 | 精品一区二区三区四区在线 | 国内精品福利视频 | 国产在线第三页 | 黄色a级片在线观看 | 一区二区三区精品在线视频 | 亚洲伦理一区 | 免费三级骚| 天天爱天天色 | 国产日韩欧美中文 | 日韩精品在线视频免费观看 | 久久一区二区免费视频 | 99视频精品免费视频 | 久草在线这里只有精品 | 久久婷婷五月综合色丁香 | 99日韩精品 | 色婷婷综合久色 | 日本中文一区二区 | 亚洲国产欧美在线看片xxoo | www亚洲精品 | 2019中文最近的2019中文在线 | 久久久久久久毛片 | 新版资源中文在线观看 | 天天色综合天天 | 久久综合给合久久狠狠色 | 久久久www | 美女露久久| 精品久久久久国产免费第一页 | 97色在线观看免费视频 | 日韩av不卡在线 | 日本一区二区不卡高清 | 天天操综| 国产精品v欧美精品 | 亚洲在线成人精品 | 婷婷在线免费 | 国产精品18久久久久久首页狼 | 久久色在线播放 | 免费av一级电影 | 久久1电影院 | 激情综合五月天 | 亚洲激情六月 | 一区二区三区www | 日色在线视频 | h动漫中文字幕 | 高清免费在线视频 | 99精品国产免费久久 | 99爱视频在线观看 | 日韩女同av | 中文字幕在线看视频国产中文版 | 激情五月播播久久久精品 | 国产色视频网站2 | 狠狠伊人 | 亚洲精品美女在线观看播放 | 成人小视频在线免费观看 | 日韩成人精品一区二区 | 亚洲精品麻豆视频 | 亚洲动漫在线观看 | 91黄色在线看 | 国产在线一线 | 久草视频在线资源 | 欧美午夜一区二区福利视频 | 在线看片91 | 欧美日韩精品在线视频 | 亚洲免费婷婷 | 毛片1000部免费看 | 亚洲人成在线观看 | 欧美 亚洲 另类 激情 另类 | 99视频精品全部免费 在线 | 成人97视频一区二区 | 91亚洲精品乱码久久久久久蜜桃 | 国产亚洲精品久久久久久 | 综合久久影院 | 人人射人人射 | .精品久久久麻豆国产精品 亚洲va欧美 | 久久超级碰视频 | 狠狠色狠狠色合久久伊人 | 日韩精品aaa | 精品视频999| 久久99久久久久久 | 日韩 在线a| 麻豆成人在线观看 | 九九久久国产 | www.69xx| 成人免费在线看片 | 99久久激情 | 午夜的福利 | 国产精品久久一 | 日韩av成人在线观看 | 欧美性免费| 国产成人精品一区二三区 | 国产成人精品一区二三区 | 国产不卡av在线 | 中文字幕永久在线 | 精品国产乱子伦一区二区 | 亚洲精品久久久久www | 色网免费观看 | 美女网站在线观看 | 成人在线免费视频 | 久久综合桃花 | 91系列在线观看 | 99精品观看 | 黄色网www | 五月激情六月丁香 | 日韩在线免费观看视频 | 天天视频亚洲 | 右手影院亚洲欧美 | 日韩三级av| 久草免费在线观看 | 麻豆视频免费观看 | 欧美性极品xxxx娇小 | 久久丁香网 | 一区二区三区四区精品视频 | 午夜影院一区 | 久久久久久久av麻豆果冻 | 91av99| 黄色日视频 | 国产精品视频免费看 | 国产一级做a| 亚洲免费在线看 | 国内久久| 免费观看av | 国产成人三级在线观看 | 六月色婷 | 免费在线黄色av | 久久色在线播放 | 在线国产能看的 | 国产成人一区二区三区在线观看 | 欧美性天天 | 精品久久一 | 欧美激情精品久久久久久免费印度 | 精品九九九| 波多野结衣一区二区三区中文字幕 | 久久精品第一页 | 少妇bbw搡bbbb搡bbbb| 亚洲天堂网在线视频观看 | 久久久久久美女 | 久久久久久久福利 | 久久av高清 | 午夜精品一二三区 | 正在播放 国产精品 | 人人爽人人香蕉 | 97超碰人人澡人人爱学生 | 亚洲成人资源在线 | 精品视频免费 | 精品美女在线观看 | 最新国产在线视频 | 欧美日韩不卡一区二区三区 | 伊人色综合网 | 久久久久久久久久久综合 | www.狠狠| 免费观看www视频 | 91传媒在线看 | 国产精品99久久免费黑人 | 97国产超碰| 久久精品视频中文字幕 | 婷婷色综 | 婷婷午夜 | 日韩高清不卡一区二区三区 | 成人资源在线播放 | 成人黄色在线视频 | 国产在线观看 | 亚洲专区中文字幕 | 色婷婷综合久久久中文字幕 | 免费视频资源 | 国产精品网红直播 | 亚洲aⅴ在线观看 | 日韩亚洲在线 | 一级黄色大片 | 亚洲天堂色婷婷 | 亚洲国产剧情 | 国产精品黑丝在线观看 | 久久久91精品国产一区二区精品 | 国产午夜不卡 | 91视频 - 114av| 欧美日韩视频在线一区 | 欧美日韩高清不卡 | 欧美日韩免费观看一区二区三区 | 国产福利a | 亚洲不卡在线 | 黄色av网站在线观看 | 国产午夜免费视频 | 国产资源免费在线观看 | 欧美成人播放 | 欧美一级片免费 | 欧美最猛性xxxxx(亚洲精品) | 一区二区三区在线免费 | 亚州精品成人 | 免费瑟瑟网站 | 在线看片日韩 | 99精品视频播放 | 国产不卡一区二区视频 | 色婷婷激情电影 | 天天要夜夜操 | 亚洲精品国偷拍自产在线观看蜜桃 | 日韩精品一二三 | 久久久在线 | 免费日韩 精品中文字幕视频在线 | 成人h电影| 久操视频在线 | 综合伊人av| 在线日韩中文 | 久碰视频在线观看 | 日日日网 | 欧美在一区 | 黄视频网站大全 | 四虎永久视频 | 国产成人精品久久亚洲高清不卡 | 国产不卡免费 | 中文字幕黄色网 | 国产成人精品福利 | 婷婷中文字幕综合 | 亚洲精品国产精品99久久 | 亚洲午夜精 | 2024av在线播放| 粉嫩一二三区 | 亚洲综合色激情五月 | 久久久www成人免费精品张筱雨 | 免费看片网址 | 中文一区在线观看 | 国产亚洲精品v | 成人91视频| 久久久黄色免费网站 | 天天操天天色天天 | 在线播放视频一区 | 久久成人久久 | 欧美黄色免费 | 久久人人艹 | 国产欧美精品在线观看 | 成人av久久 | 免费看国产黄色 | 在线国产激情视频 | 日韩伦理片一区二区三区 | 亚洲欧美日韩在线看 | 97视频在线免费观看 | 国产精品永久免费 | 欧美激情综合色综合啪啪五月 | 亚洲一级片在线看 | 狠狠色狠狠色合久久伊人 | 天天插日日插 | 国产免费激情久久 | 国产精品a成v人在线播放 | 人人干免费 | 亚洲在线视频免费观看 | 国外成人在线视频网站 | 99国产精品免费网站 | 亚洲精品裸体 | 久久久国产一区二区三区 | 亚洲精品美女久久久 | 亚洲成人xxx | 久精品一区 | 成人aaa毛片 | 中文欧美字幕免费 | 99久久精品免费一区 | 最新av免费在线观看 | 亚洲视频 中文字幕 | 国产高清日韩欧美 | 在线观看网站av | www.天天射.com| 久久一级片 | 夜夜夜夜夜夜操 | 国产精品入口a级 | 中午字幕在线 | 欧美日韩精品影院 | 国产精品久久久久久爽爽爽 | 色视频在线免费观看 | 欧美性久久久 | 亚洲色图27p | 欧美日韩国内在线 | 波多野结衣在线中文字幕 | 久久综合九色欧美综合狠狠 | 人人澡人人爽欧一区 | 国产aaa大片 | 亚洲欧美乱综合图片区小说区 | 开心综合网 | 五月婷婷丁香激情 | 二区三区毛片 | 久久精品国产v日韩v亚洲 | 97精品国产手机 | 国产丝袜一区二区三区 | 亚洲视频综合 | 国产精选在线观看 | 在线视频一二三 | 在线观看国产日韩 | 国产精品综合在线观看 | 天天做天天爱夜夜爽 | 欧美日韩久久不卡 | 国产九九九视频 | 国产探花视频在线播放 | www黄在线 | 超碰人人在| 麻豆国产网站入口 | 亚洲狠狠操| 99热最新| 国产一级免费在线观看 | 国产精品麻豆三级一区视频 | 亚洲国产人午在线一二区 | a级黄色片视频 | 激情五月在线视频 | 在线国产福利 | 免费h精品视频在线播放 | 婷婷丁香花 | 亚洲一二区精品 | 麻豆传媒在线免费看 | 亚洲国产欧美在线人成大黄瓜 | 99精品国产亚洲 | 99色在线视频 | 草久在线播放 | 国产麻豆精品在线观看 | 日韩系列在线观看 | 99精品偷拍视频一区二区三区 | 色天天久久 | 日韩精品视频网站 | 色网av | 91免费观看网站 | 欧美日韩在线电影 | 欧美成人免费在线 | 激情大尺度视频 | 亚洲黄色一级电影 | 国语精品久久 | 亚洲狠狠操 | 人人看人人做人人澡 | 国产精品久久久久久久久久久久午 | 免费aa大片 | 国产高清不卡在线 | 国产一区播放 | 成人a大片 | 99人久久精品视频最新地址 | 欧美激情在线网站 | 99成人免费视频 | 日韩一区二区三区免费视频 | 中文字幕在线第一页 | av高清免费在线 | 在线中文字幕播放 | 日韩视频1 | av在线com| 亚洲精区二区三区四区麻豆 | 中文字幕 影院 | 97视频播放 | 欧美伊人网 | 女人高潮特级毛片 | 99视频免费 | 在线免费看黄色 | www天天干 | 国产一级精品在线观看 | 欧美一二三视频 | 97在线观 | 看污网站 | 国产97在线看 | 天天亚洲综合 | 免费国产在线观看 | zzijzzij亚洲成熟少妇 | 黄色aa久久| 热久久最新地址 | 美女视频免费精品 | 日韩在线观看a | 国产精品一区久久久久 | 91av成人 | 狠狠做深爱婷婷综合一区 | 超碰在线中文字幕 | 日韩com | 麻豆视频在线播放 | 久久免费黄色网址 | 免费看一级特黄a大片 | 成人动漫一区二区 | 在线免费av观看 | 全黄网站 | www色片 | 91av电影在线观看 | 一级黄色大片在线观看 | 国产亚洲aⅴaaaaaa毛片 | 天天综合网~永久入口 | 五月天天色 | www五月天| 99久久久久久久久久 | av在线观| 激情五月在线 | 精品国精品自拍自在线 | 中文字幕丰满人伦在线 | 国产毛片在线 | 日韩网站在线免费观看 | 不卡的av电影在线观看 | 成人福利在线观看 | 伊人婷婷色 | 在线一级片 | 亚洲 中文 在线 精品 | 国产丝袜一区二区三区 | 亚洲精品视频在线观看免费视频 | 国产成人黄色 | 国产精品婷婷 | 成人app在线免费观看 | 蜜臀久久99精品久久久无需会员 | 波多野结衣精品在线 | 精品国产电影一区 | 亚洲欧美日韩国产一区二区三区 | 国产一区二区不卡视频 | 亚洲国产精品久久久久 | 成人国产精品一区 | 日韩中文在线观看 | 国产精品久久久久久久久久免费 | 国产视频日本 | 色网站在线看 | 欧美成人播放 | 日韩免费电影一区二区三区 | 亚洲最大av在线播放 | 国产成人精品一区二区三区福利 | 久久精品国产美女 | 91九色在线观看视频 | 免费日韩电影 | 黄色录像av| 国产在线精品一区二区 | 日韩av免费大片 | 天堂av网站 | 久草久草在线观看 | 97视频亚洲 | 国产成人一区二区在线观看 | 999在线视频 | 国产成人a亚洲精品v | 欧美成人播放 | 一区二区视频在线看 | 久久精品国产一区二区 | 亚洲区视频在线 | 日韩av在线小说 | 欧美激情视频在线免费观看 | 国产高清免费在线观看 | 欧美美女一级片 | 国产精品爽爽久久久久久蜜臀 | av超碰在线观看 | 少妇做爰k8经典 | 国语精品久久 | 成人一级免费电影 | 人人干人人模 | 亚洲欧美va | 久久艹精品 | 久久精品久久久精品美女 | 成人黄色av网站 | 亚洲国产欧洲综合997久久, | 国产精品久久久久av | www黄色软件| 国产精品不卡av | 日韩深夜在线观看 | 最近中文字幕高清字幕免费mv | 综合色综合色 | 成年人免费看 | 日韩在线视频网址 | 精品久久久久久久久久 | 日日夜夜噜 | www.香蕉 | 久久99亚洲热视 | 91精品国产综合久久久久久久 | 国产丝袜美腿在线 | 午夜精品一区二区三区免费 | 久久综合九色99 | a√天堂资源 | 色综合久久久网 | 色婷婷精品 | 青青草国产免费 | 狠狠色狠狠色综合系列 | 高清一区二区三区 | 超碰av免费| 国产午夜精品av一区二区 | 久久久www成人免费毛片麻豆 | 黄色片网站av | 久久男人视频 | 乱子伦av| 欧美成年人在线视频 | 欧美精品二 | 久草在线视频资源 | 亚洲高清在线精品 | 四虎成人精品永久免费av | 精品一区二区6 | 综合国产视频 | 成人va天堂 | 黄色精品一区 | 九九九九精品九九九九 | 午夜精品一区二区三区四区 | 在线有码中文 | 开心色激情网 | 中文字幕国产精品一区二区 | 91亚洲激情| 在线激情av电影 | 一区二区三区av在线 | 中文字幕一区二区在线观看 | 亚洲精品资源在线 | 香蕉久草在线 | 国产日韩精品一区二区三区 | 欧美性大胆 | 三级毛片视频 | 综合激情网 | 午夜精品电影 | 国产精品免费成人 | 亚洲国产中文字幕在线观看 | 精品一区二区三区电影 | 免费国产在线观看 | 色婷婷免费 | 一区二区三区精品在线视频 | 丁香五婷| 国产一级视频在线 | 色五婷婷 | 国产精品中文字幕在线观看 | 不卡日韩av | 久久久久这里只有精品 | 97人人爽人人 | 一区二区成人国产精品 | 国产精品成人免费一区久久羞羞 | 激情五月婷婷 | 国产91精品欧美 | 一本一本久久a久久精品综合小说 | 正在播放一区二区 | 亚洲综合最新在线 | 日韩欧美一区二区三区视频 | 在线观看中文字幕亚洲 | 婷婷久久久 | 午夜.dj高清免费观看视频 | 成年人在线观看免费视频 | 国产一区免费在线 | 免费视频久久久 | 国产尤物在线 | 国产日韩av在线 | 黄色片网站av| 久久av免费 | 日韩欧美精品在线视频 | 日韩成人看片 | 日韩在线视频不卡 | 亚州成人av在线 | 精品国产精品一区二区夜夜嗨 | 日韩视频免费 | 成人在线免费观看网站 | 91最新在线 | 国产剧情在线一区 | 国产最新精品视频 | 久久久亚洲麻豆日韩精品一区三区 | 性色视频在线 | 亚洲片在线 | 日韩免费在线观看 | 国产a精品 | 2020天天干夜夜爽 | 狠狠狠干狠狠 | www.xxxx变态.com | 国产亚洲婷婷免费 | 国产高清黄色 | 色综合婷婷久久 | 91麻豆视频 | 国产一级电影网 | 四虎伊人 | 国产精品精品久久久 | 干狠狠| 大胆欧美gogo免费视频一二区 | 精品国产乱码久久久久久三级人 | 黄色大片中国 | 国产一级精品在线观看 | 国产69精品久久久久久 | 国产精品a成v人在线播放 | 亚洲成人二区 | 免费看黄在线网站 | 免费观看成人网 | 亚州成人av在线 | 韩国精品福利一区二区三区 | 97看片吧| 五月天六月丁香 | 国产这里只有精品 | 欧美日视频 | 日韩毛片精品 | 国产精品99久久久 | 日韩在线中文字幕 | 国产一级一级国产 | 久久另类小说 | av+在线播放在线播放 | 麻豆视频国产 | 一区二区三区四区五区在线视频 | 黄色网大全 | 亚洲国产午夜 | 九九热精品视频在线播放 | 伊人激情综合 | 免费黄色av. | 国内三级在线观看 | 国产日韩欧美在线观看 | 精品人人爽 | 午夜99| 日韩免费高清在线观看 | 91漂亮少妇露脸在线播放 | 激情中文在线 | 国产精品一区二区无线 | 日韩最新av | www.少妇| 国产永久免费高清在线观看视频 | 在线观看网站av | 国内精品久久久久久久影视麻豆 | 日韩另类在线 | 在线播放亚洲 | 黄网站免费大全入口 | 天堂av色婷婷一区二区三区 | 国内99视频 | 国产精品99久久久 | 天天曰视频| 超碰97在线看 | 色一色在线 | 天天干夜夜擦 | 国产欧美久久久精品影院 | 中文字幕亚洲综合久久五月天色无吗'' | av中文字幕日韩 | 激情五月婷婷综合网 | 国产一级一级国产 | 成人免费观看视频大全 | 日韩欧美一区二区三区视频 | 黄色小说18| 97电影在线看视频 | 日韩美精品视频 | 亚洲成aⅴ人在线观看 | 97超碰国产精品 | 免费看污黄网站 | 福利精品在线 | 在线中文字幕播放 | 国产丝袜网站 | 国产一区二区电影在线观看 | 国产精品视频全国免费观看 | 人人干免费 | 日韩美视频| a午夜在线 | 人人爽人人澡 | 久久不射电影院 | 欧美成人999| 日韩在线视频看看 | 中文字幕在线第一页 | 中文字幕精品一区二区精品 | 中文字幕在线免费看 | 亚洲人成精品久久久久 | 五月天久久久 | 日韩国产精品久久久久久亚洲 | 精品在线观看一区二区三区 | 在线中文字幕一区二区 | 国产在线97 | 欧美巨大 | 丁香婷婷社区 | 超碰在线天天 | www.91av在线| 免费av影视| 久久论理 | 国产一区二区精品 | 九七视频在线观看 | 麻豆视频在线 | 99久久精品无码一区二区毛片 | 中文国产在线观看 | 夜夜骑天天操 | 久精品视频在线观看 | 天天操天天干天天干 | 色久网| 99视频在线免费播放 | 免费影视大全推荐 | 日韩免费视频线观看 | 久久久久免费看 | 精品久久久久久久久久岛国gif | 99re视频在线观看 | 成人精品久久 | 中文字幕在线视频国产 | 91免费黄视频 | 国产在线免费观看 | 午夜三级在线 | 色综合久久久久久久 | 色网站国产精品 | 亚洲精品视频在线观看免费视频 | 成年人黄色免费网站 | 蜜桃视频日韩 | 四虎永久免费网站 | 69av免费视频 | 亚洲伦理电影在线 | 91综合视频在线观看 | 国产麻豆精品95视频 | 欧美日韩国产成人 | 日批视频在线播放 | 日韩理论片 | 欧美精品一区二区蜜臀亚洲 | 久久久精品综合 | 成人在线小视频 | 亚洲精品白浆高清久久久久久 | 欧美日韩国产综合网 | 精品999| 九九热久久免费视频 | 亚洲少妇久久 | 一级a毛片高清视频 | 91精品国产91久久久久 | 91免费观看国产 | 国产91影院| 亚洲一区尤物 | 日本女人的性生活视频 | 国产 视频 高清 免费 | 99久久精品久久久久久动态片 | 欧美日韩在线观看不卡 | 日韩超碰 | 精品国产电影一区二区 | 亚洲综合激情小说 | 亚洲 欧美变态 另类 综合 | 国产在线播放不卡 | 国产一区二区在线免费播放 | 中文在线免费看视频 | 国产精品色婷婷 | 天堂av在线中文在线 | 97干com| 国产自偷自拍 | 天天做综合网 | 久草新在线 | 91麻豆精品国产91久久久久 | 99久久99热这里只有精品 | 日韩1页| 国产精品成人在线 | 日韩精品一卡 | 成人网看片 | 欧美激情视频久久 | av电影av在线 | 人人澡人人草 | 99re在线视频观看 | 婷婷丁香七月 | 91成人精品国产刺激国语对白 | 精品视频久久久 | 中文字幕精| 麻豆视频成人 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 日一日干一干 | 激情www | 色综久久 | 天天操夜夜拍 | 久久免费视频精品 | 久久a国产 | 黄色免费网站 | 色全色在线资源网 | 国精产品永久999 | 欧美巨乳网 | 亚洲人视频在线 | 91网免费观看 | 国产精品高潮呻吟久久av无 | 天堂在线一区二区三区 | 国产区精品视频 | 欧美日韩在线电影 | 中文在线a在线 | 九九交易行官网 | 亚洲精品视频在线观看免费视频 | 久久人操 | 91成人破解版 | 亚洲激情网站免费观看 | 久草线 | 黄色大全免费网站 | 综合视频在线 | 97超碰影视| 久久er99热精品一区二区 | 午夜久久影视 | 中国一级特黄毛片大片久久 | 狠狠色噜噜狠狠 | 成人91在线 | 久久久久麻豆v国产 | 色网站免费在线看 | 亚洲成人精品av | 六月激情丁香 | 超碰官网| 日韩av图片 | 免费欧美精品 | 蜜桃麻豆www久久囤产精品 | 中文字幕 在线看 | 国产精品99蜜臀久久不卡二区 | 日本巨乳在线 | 精品久久久久久久久久久院品网 | 久久精品aaa| 超碰在线人人97 | 亚洲一区视频免费观看 | 国产美腿白丝袜足在线av | 成人av一区二区三区 | 国产日产精品一区二区三区四区的观看方式 | 青青看片 | 国产美女视频免费 | 免费人人干 | 亚洲一级在线观看 | 欧美另类z0zx | 日本成人黄色片 | 欧美日韩在线观看一区二区三区 | 久草在线资源免费 | a√资源在线| 久久久精品高清 | 欧美一级日韩三级 | 国产精品久久久久久久久久久久 | 一本一本久久a久久精品综合 | 成人av av在线| 99久久精品日本一区二区免费 | 99热这里只有精品8 久久综合毛片 | 五月婷婷深开心 | 日韩在线播放欧美字幕 | 免费成人黄色片 | 久久精品国产99 | 日韩av看片 | 久久亚洲精品国产亚洲老地址 | 久草视频在线播放 | 超碰在线最新 | 日韩av成人免费看 | 香蕉视频在线免费看 | 国产精品久免费的黄网站 | 日日草av| 狠狠色噜噜狠狠 | 日韩精品一区二区三区视频播放 | 亚洲日本中文字幕在线观看 | 色综合天天狠天天透天天伊人 | 国产精品第2页 | 日韩黄色网络 | 久久久久久久久久久久国产精品 | 国产99久久久国产精品成人免费 | 涩涩网站免费 | 国产精品6| 成人黄色片在线播放 | 日韩电影一区二区在线 | 国产 欧美 日产久久 | 国产精品网站一区二区三区 | 久久电影国产免费久久电影 | 国产网红在线观看 | 精品91视频 | 国产一区二区在线视频观看 | 色亚洲网 | 一区二区三区观看 | 99久久夜色精品国产亚洲96 | 免费国产在线精品 | 黄色免费看片网站 | 中文字幕高清有码 | 国产污视频在线观看 | 中文字幕在线观看资源 | 婷婷 中文字幕 | 综合久久久久 | 夜夜夜夜爽 | 国产高清在线不卡 | 久久99久久99精品免观看软件 | 国产精品国产三级国产不产一地 | 国产激情免费 | 久久久精品高清 | 91九色porn在线资源 | 免费看v片 | 最近日韩免费视频 | 99精品国产在热久久下载 | 国产精品一区免费在线观看 | 天天操人人干 | 欧美日韩高清在线 | 91丨九色丨高潮丰满 | 波多野结衣最新 | 五月婷婷黄色 | 在线视频久 | 久久综合久久综合久久综合 | 中文字幕在线久一本久 | 手机看片久久 | 亚洲精品在线视频网站 | 中文字幕免费久久 | 国产中文字幕一区 | 国产精品 视频 | 国产精品一区免费在线观看 | 97成人精品| 久久国语 | 超碰97在线人人 | 亚洲国产精品激情在线观看 | 日本婷婷色 | 91在线视频免费 | 免费在线一区二区 | 久久久精品午夜 | 精品久久久免费 | 我爱av激情网| 中文字幕资源网 | 国产精品涩涩屋www在线观看 | 日韩亚洲国产中文字幕 | 国产在线观看免费 | 欧美日韩在线播放 | 欧亚日韩精品一区二区在线 | 黄色午夜网站 | 91看片淫黄大片一级在线观看 | 国产精品专区在线 | 久久香蕉国产 | 激情在线免费视频 | 免费高清在线视频一区· | 69视频网站 | 操处女逼 | 99在线观看免费视频精品观看 | 午夜精品成人一区二区三区 | www黄色com | 国产精品亚洲人在线观看 | 国产精品黄网站在线观看 | 午夜精品久久久久久 | 国产视频在线一区二区 | 免费a视频在线观看 | 91精品国产91久久久久 | 久久资源总站 | 国产成人一区二区三区电影 | 在线免费观看一区二区三区 | 永久免费观看视频 | 日韩欧美精品在线视频 | 久久精品久久综合 | 日韩av成人在线 | 国产精品久久久久影院 | 五月天婷婷在线视频 | 亚洲伊人天堂 | 美女网站黄在线观看 | 99精品国产高清在线观看 | 毛片一二区 | 精品99在线| 国产午夜三级一区二区三桃花影视 | 国产视| 日韩中文字幕在线不卡 | 麻豆久久久久久久 | 日韩av二区| av夜夜操| 国产高清视频免费 | 青青草在久久免费久久免费 |