程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> .NET網頁編程 >> C# >> C#入門知識 >> 使用NVelocity生成內容的幾種方式

使用NVelocity生成內容的幾種方式

編輯:C#入門知識

使用NVelocity也有幾個年頭了,主要是在我的代碼生成工具Database2Sharp上使用來生成相關代碼的,不過NVelocity是一個非常不錯的模板引擎,可以用來生成文件、頁面等相關處理,非常高效和方便。

它原先是在網站http://nvelocity.sourceforge.net/ 上維護,不過從0.41後,該網站就不再進行NVelocity更新了,現在可以在網站http://nvelocity.codeplex.com/上獲得最新版本的更新,接著版本的更新操作,我們把NVelocity的幾種生成文件的操作介紹一下,以便大家進行更深入的了解。

我在早期幾篇文章也介紹過NVelocity的使用,主要介紹了NVelocity的語法和邏輯的和使用,還有就是如何在實際項目中進行的內容動態生成的操作,有興趣可以翻下下面幾篇文章:

強大的模板引擎開源軟件NVelocity

Database2Sharp版本更新之自定義模板生成

使用NVelocity0.5實現服務器端頁面自動生成

1、基於NVelocity的幾種內容生成方式

從上面的圖示,我們可以看到,NVelocity的模板化生成包含了3種方式,一種是從文件到文件或者字符串,一種是從字符串到字符串,他們各自的處理方式有所不同,但是都能正確解析裡面的內容。

為了更好利用NVelocity的特性,我們對它進行一個初步的輔助類封裝。

    /// <summary>
    /// 基於NVelocity的模板文件生成輔助類
    /// </summary>
    public class NVelocityHelper
    {
        protected VelocityContext context;
        protected Template template;
        protected string templateFile;

        /// <summary>
        /// 存放鍵值的字典內容
        /// </summary>
        private Dictionary<string, object> KeyObjDict = new Dictionary<string, object>();


        /// <summary>
        /// 添加一個鍵值對象
        /// </summary>
        /// <param name="key">鍵,不可重復</param>
        /// <param name="value">值</param>
        /// <returns></returns>
        public NVelocityHelper AddKeyValue(string key, object value)
        {
            if (!KeyObjDict.ContainsKey(key))
            {
                KeyObjDict.Add(key, value);
            }
            return this;
        }
................

 上面的AddKeyValue方法,主要用來為模板引擎添加一些需要綁定在頁面上的變量對象,這樣頁面變量參數的內容就能正確解析出來了。

為了使用NVelocity的各種特性,我們需要在輔助類裡面構造模板的相關信息,設置相關參數。

        /// <summary>
        /// 初始化模板引擎
        /// </summary>
        protected virtual void InitTemplateEngine()
        {
            try
            {
                //Velocity.Init(NVELOCITY_PROPERTY);
                VelocityEngine templateEngine = new VelocityEngine();
                templateEngine.SetProperty(RuntimeConstants.RESOURCE_LOADER, "file");

                templateEngine.SetProperty(RuntimeConstants.INPUT_ENCODING, "utf-8");
                templateEngine.SetProperty(RuntimeConstants.OUTPUT_ENCODING, "utf-8");

                //如果設置了FILE_RESOURCE_LOADER_PATH屬性,那麼模板文件的基礎路徑就是基於這個設置的目錄,否則默認當前運行目錄
                templateEngine.SetProperty(RuntimeConstants.FILE_RESOURCE_LOADER_PATH, AppDomain.CurrentDomain.BaseDirectory);

                templateEngine.Init();

                template = templateEngine.GetTemplate(templateFile);
            }
            catch (ResourceNotFoundException re)
            {
                string error = string.Format("Cannot find template " + templateFile);

                LogTextHelper.Error(error);
                throw new Exception(error, re);
            }
            catch (ParseErrorException pee)
            {
                string error = string.Format("Syntax error in template " + templateFile + ":" + pee.StackTrace);
                LogTextHelper.Error(error);
                throw new Exception(error, pee);
            }
        }

在生成內容之前,需要把相關的對象屬性綁定到模板引擎的上下文對象裡面。

        /// <summary>
        /// 初始化上下文的內容
        /// </summary>
        private void InitContext()
        {
            context = new VelocityContext();
            foreach (string key in KeyObjDict.Keys)
            {
                context.Put(key, KeyObjDict[key]);
            }
        }

1)根據模板文件構造對應的文件內容

        /// <summary>
        ///根據模板創建輸出的文件,並返回生成的文件路徑
        /// </summary>
        public virtual string ExecuteFile()
        {
            string fileName = "";
            if (template != null)
            {
                string filePath = CheckEndBySlash(directoryOfOutput);
                fileName = filePath + fileNameOfOutput + fileExtension;

                if (!string.IsNullOrEmpty(filePath) && !Directory.Exists(filePath))
                {
                    Directory.CreateDirectory(filePath);
                }

                //LogTextHelper.Debug(string.Format("Class file output path:{0}", fileName));
                InitContext();
                using (StreamWriter writer = new StreamWriter(fileName, false))
                {
                    template.Merge(context, writer);
                }
            }
            return fileName;
        }

2)根據模板文件構造字符串內容

        /// <summary>
        /// 根據模板輸出字符串內容
        /// </summary>
        /// <param name="templateFile"></param>
        /// <returns></returns>
        public string ExecuteString()
        {
            InitContext();    
            System.IO.StringWriter writer = new System.IO.StringWriter();
            template.Merge(context, writer);
            return writer.GetStringBuilder().ToString();
        }

 

3)根據字符串內容構造字符串輸出

        /// <summary>
        /// 合並字符串的內容
        /// </summary>
        /// <returns></returns>
        public string ExecuteMergeString(string inputString)
        {
            VelocityEngine templateEngine = new VelocityEngine();
            templateEngine.Init();

            InitContext();

            System.IO.StringWriter writer = new System.IO.StringWriter();
            templateEngine.Evaluate(context, writer, "mystring", inputString);

            return writer.GetStringBuilder().ToString();
        }

上面幾種操作模板輸出的方式,其調用代碼如下所示。

        private void btnGenerateFile_Click(object sender, EventArgs e)
        {
            string tempalte = "Template/template.htm";//相對目錄

            TestInfo info = new TestInfo();
            info.Title = "測試標題";
            info.Content = "測試內容,這是測試內容";
            info.Datetime = DateTime.Now;

            NVelocityHelper adapter = new NVelocityHelper(tempalte);
            adapter.AddKeyValue("title", "This is a title")
                .AddKeyValue("content", "This is a Content")
                .AddKeyValue("datetime", System.DateTime.Now)
                .AddKeyValue("TestInfo", info);

            adapter.FileNameOfOutput = "testTemplate";
            string filePath = adapter.ExecuteFile();
            if (!string.IsNullOrEmpty(filePath))
            {
                Process.Start(filePath);
            }
        }
        private void btnGenerate_Click(object sender, EventArgs e)
        {
            string tempalte = "Template/template.htm";//相對目錄

            TestInfo info = new TestInfo();
            info.Title = "測試標題";
            info.Content = "測試內容,這是測試內容";
            info.Datetime = DateTime.Now;

            NVelocityHelper adapter = new NVelocityHelper(tempalte);
            adapter.AddKeyValue("title", "This is a title")
                .AddKeyValue("content", "This is a Content")
                .AddKeyValue("datetime", System.DateTime.Now)
                .AddKeyValue("TestInfo", info);

            this.txtCode.Text = adapter.ExecuteString();
        }
        private void btnMergeString_Click(object sender, EventArgs e)
        {
            System.Text.StringBuilder builder = new System.Text.StringBuilder();
            builder.Append(
                "${Title}\r\n" +
                "$Content\r\n" +
                "$Digest\r\n" +
                "$Author\r\n" +
                "$Keyword\r\n" +
                "$DateTime\r\n");
             NVelocityHelper adapter = new NVelocityHelper();
             adapter.AddKeyValue("Title", "標題").
                 AddKeyValue("Content", "內容").
                 AddKeyValue("Digest", "摘要").
                 AddKeyValue("Author", "作者").
                 AddKeyValue("Keyword", "關鍵詞").
                 AddKeyValue("DateTime", DateTime.Now);
             this.txtCode.Text = adapter.ExecuteMergeString(builder.ToString());
        }

2、模板引擎NVelocity的幾種應用場景

 上面的幾種操作模板內容的方式,能夠在絕大多數情況下滿足我們的應用要求,如可以在代碼生成工具裡面,定義一些自定義的內容模板,然後結合數據庫的元數據信息,實現豐富邏輯的代碼生成操作。

也可以在一些內容管理的應用上(如文章管理方面),根據輸入的內容,實現文章內容的文件生成操作,這個生成後,我們就直接使用文章的文件鏈接地址就可以了。

或者根據數據信息生成具體的頁面,用於套打操作,如下是Winform裡面的套打處理。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved