這篇文章主要介紹了asp.net使用DataGridTree實(shí)現(xiàn)下拉樹的方法,詳細(xì)的講述了DataGridTree實(shí)現(xiàn)下拉樹的原理與具體實(shí)現(xiàn)方法,具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
			本文實(shí)例講述了asp.net使用DataGridTree實(shí)現(xiàn)下拉樹的方法。分享給大家供大家參考。具體實(shí)現(xiàn)方法如下:
			下拉樹實(shí)現(xiàn)原理:輸出json到客戶端,客戶端實(shí)現(xiàn)動(dòng)態(tài)加載,中間不會(huì)和服務(wù)端交互。數(shù)據(jù)量支持上經(jīng)測(cè)試幾千還是很快的。本下拉樹控件是用c#+js樹實(shí)現(xiàn)。
			2.c# 計(jì)算器 計(jì)算字符串?dāng)?shù)學(xué)表達(dá)式源碼
			計(jì)算數(shù)學(xué)表達(dá)式原理 采用c#實(shí)現(xiàn) 很實(shí)用
		//a.建立兩個(gè)棧:第一個(gè)位操作數(shù)棧,第二個(gè)操作符符棧!(將棧定義為string類型)
		//b.對(duì)數(shù)字來說是無條件壓入數(shù)字棧中.
		//c.而對(duì)符號(hào)來說,只有當(dāng)前棧頂元素的優(yōu)先值小于掃到的符號(hào)時(shí)(比如”+”小于”*”),此符號(hào)才壓入棧;否則大于等于的情況是將當(dāng)前棧頂元素彈出棧,與當(dāng)前數(shù)字棧的前兩個(gè)數(shù)字組成式子進(jìn)行計(jì)算.計(jì)算結(jié)果當(dāng)作數(shù)字壓入數(shù)字棧作為棧頂元素(要舍棄已經(jīng)彈出的兩個(gè)數(shù)字),而那個(gè)掃描到的符號(hào)則將代替那個(gè)彈出的符號(hào)作為棧頂元素)。
		//d.最后說一下括號(hào),原則是掃描到左括號(hào)時(shí)無條件壓入符號(hào)棧,而掃到右括號(hào)時(shí),則彈出離棧頂最近的一個(gè)左括號(hào)以上的全部符號(hào)與數(shù)字棧的數(shù)字做運(yùn)算
			3.asp.net教程 datagridtree表格樹控件 
			繼承asp.net的datagrid控件實(shí)現(xiàn)的表格樹控件
		/*表格樹控件說明 
		* 此控件繼承datagrid 新增屬性說明:
		* 1.treeparentcode:頂級(jí)根節(jié)點(diǎn)parentcode
		* 2.treedisplaydeep:展現(xiàn)表格樹深度默認(rèn)為1 
		* 3.sumcolumns:自動(dòng)匯總到根節(jié)點(diǎn)的字段集合 針對(duì) decimal類型
		* 4.新增樹狀列模板templatetreecolumn 此模板繼承了templatecolumn 重寫了方法initializecell 
		* 客戶端新增特性配置說明 
		* 1.固定列 配置 itemstyle-css教程class='tdlockedclass' 
		* 2.固定表頭 配置 headerstyle-cssclass='trlockedclass'
		* 3.文本框 input 或 <asp:textbox 配置事件onchange='sumparent(this);' 數(shù)字改變相應(yīng)所有父節(jié)點(diǎn)也隨著改變 針對(duì)數(shù)字型 其他不支持 
		* 不過可以自定義js
		* 報(bào)表說明:
		* 1.datagridtree.enableviewstate=false;提高加載速度
		* 2.動(dòng)態(tài)定義列 實(shí)現(xiàn) boundcolumn column = new boundcolumn();
		column.headertext = "動(dòng)態(tài)列";
		column.datafield = "unitname";
		datagridnew.columns.add(column);
		* 也可以自定義默認(rèn)模板 動(dòng)態(tài)加載模板 定義模板例子templatetreecolumn,不用繼承templatecolumn,實(shí)現(xiàn)接口 itemplate initializecell 方法就可以了
		* 不足之處:1.對(duì)于復(fù)雜多行表頭 不知 如何實(shí)現(xiàn)
		* 2.表頭和列固定 數(shù)據(jù)量大時(shí) 會(huì)影響反映速度 一千左右的數(shù)據(jù)量 還時(shí)沒問題的 數(shù)據(jù)量在大的話 課考慮采用ajax動(dòng)態(tài)加載 目前此功能還沒實(shí)現(xiàn)
		實(shí)例代碼
			復(fù)制代碼代碼如下:
			private void maketree(datatable dtnodesets, string strparentcolumn, string strrootvalue, string strindexcolumn, string strtextcolumn, dropdownlist drpbind, int i)
		{
		//每向下一層,多一個(gè)縮入單位   
		i++;
		dataview dvnodesets = new dataview(dtnodesets);
		dvnodesets.rowfilter = strparentcolumn + "=" + strrootvalue;
		string strpading = ""; //縮入字符  
		//通過i來控制縮入字符的長度,我這里設(shè)定的是一個(gè)全角的空格   
		for (int j = 0; j < i; j++)
		strpading += " ";//如果要增加縮入的長度,改成兩個(gè)全角的空格就可以了  
		foreach (datarowview drv in dvnodesets)
		{
		treenode tnnode = new treenode();
		listitem li = new listitem(strpading + "├" + drv[strtextcolumn].tostring(), drv[strindexcolumn].tostring());
		drpbind.items.add(li);
		maketree(dtnodesets, strparentcolumn, drv[strindexcolumn].tostring(), strindexcolumn, strtextcolumn, drpbind, i);
		}
		//遞歸結(jié)束,要回到上一層,所以縮入量減少一個(gè)單位   
		i--;
		}
		/// <summary>   
		/// sql語句查詢,再綁定到droplist里面   
		/// </summary>   
		private void createtree()
		{
		//查詢zonelist   
		string sql = "select * from master_department where parent_department='003'";
		dataset ds = db.getds();
		datatable dt = ds.tables[0];
		maketree(dt, "parent_department", "003", "department_code", "department_name", dropdownlist1, -1);
		}
		網(wǎng)上找的另一個(gè)比較好的實(shí)例	
		復(fù)制代碼代碼如下:
			using system;
		using system.collections.generic;
		using system.text;
		using system.web.ui.webcontrols;
		namespace interface.common
		{
		    public interface idropdowntree : idisposable
		    {
		        /**//// <summary>
		        /// 返回dictionary里分別對(duì)應(yīng)id,文本,如果沒有子節(jié)點(diǎn)返回null
		        /// </summary>
		        /// <param name="parentid">父節(jié)點(diǎn)id</param>
		        /// <returns></returns>
		        dictionary<string, string> getchildcategory(string parentid);
		        /**//// <summary>
		        /// 代碼里寫return new interface.common.dropdowntree(this);
		        /// </summary>
		        dropdowntree dropdowntree
		        {
		            get;
		        }
		    }
		    public sealed class dropdowntree
		    {
		        idropdowntree _dropdowntree;
		        public dropdowntree(idropdowntree dropdowntree)
		        {
		            _dropdowntree = dropdowntree;
		        }
		        /**//// <summary>
		        /// 用于樹的前綴
		        /// </summary>
		        /// <param name="islast">是否是同級(jí)節(jié)點(diǎn)中的最后一個(gè)</param>
		        /// <param name="haschild">本節(jié)點(diǎn)是否擁有子節(jié)點(diǎn)</param>
		        /// <param name="parentstring">父節(jié)點(diǎn)前綴符號(hào)</param>
		        /// <returns>本節(jié)點(diǎn)的前綴</returns>
		        private string getprefix(bool islast, bool haschild, string parentstring)
		        {
		            string result = string.empty;
		            if (!string.isnullorempty(parentstring))
		            {
		                parentstring = parentstring.remove(parentstring.length - 1).replace("├", "│").replace("└", " ");
		                result += parentstring;
		            }
		            if (islast)
		            {
		                result += "└";
		            }
		            else
		            {
		                result += "├";
		            }
		            if (haschild)
		            {
		                result += "┬";
		            }
		            else
		            {
		                result += "─";
		            }
		            return result;
		        }
		        綁定下拉菜單#region 綁定下拉菜單
		        /**//// <summary>
		        /// 綁定連動(dòng)級(jí)的下拉菜單
		        /// </summary>
		        /// <param name="ddlgoodstype">傳進(jìn)一個(gè)被綁定的dropdownlist</param>
		        /// <param name="removeid">被排除綁定的節(jié)點(diǎn)id</param>
		        /// <param name="autodispose">是否自動(dòng)釋放</param>
		        public void bindtodropdownlist(dropdownlist ddlgoodstype, string removeid,string parentid, bool autodispose)
		        {
		            if (ddlgoodstype != null)
		            {
		                listitem listitem = null;
		                string currentid = parentid;//根節(jié)點(diǎn)/父id
		                string currentsign = string.empty;//當(dāng)前節(jié)點(diǎn)符號(hào);
		                string parrentsign = string.empty; //父節(jié)點(diǎn)符號(hào);
		                bool haschild = true;//是否有子
		                queue<string> parentkeylist = new queue<string>();//存 有子節(jié)點(diǎn)的 節(jié)點(diǎn)id
		                queue<string> parentsignlist = new queue<string>();//對(duì)應(yīng)節(jié)點(diǎn)id的前綴符號(hào)
		                int itemindexof = 0;//父節(jié)點(diǎn)所在的位置 
		                while (haschild)
		                {
		                    int lastonecount = 1;//用于計(jì)算在同級(jí)別中是否最后一個(gè)
		                    dictionary<string, string> childlist = _dropdowntree.getchildcategory(currentid);// 得到子節(jié)點(diǎn)列表
		                    if (childlist != null && childlist.count > 0)
		                    {
		                        if (!string.isnullorempty(removeid) && childlist.containskey(removeid))
		                        {
		                            childlist.remove(removeid);
		                        }
		                        foreach (keyvaluepair<string, string> entry in childlist)
		                        {
		                            if (_dropdowntree.getchildcategory(entry.key) != null)//存在子
		                            {
		                                currentsign = getprefix(lastonecount == childlist.count, true, parrentsign);
		                                listitem = new listitem(currentsign + entry.value, entry.key);
		                                parentkeylist.enqueue(entry.key);//當(dāng)前的節(jié)點(diǎn)id
		                                parentsignlist.enqueue(currentsign);//當(dāng)前的節(jié)點(diǎn)符號(hào)
		                            }
		                            else//不存在子
		                            {
		                                currentsign = getprefix(lastonecount == childlist.count, false, parrentsign);
		                                listitem = new listitem(currentsign + entry.value, entry.key);
		                            }
		                            if (ddlgoodstype.items.count != 0)
		                            {
		                                itemindexof = string.isnullorempty(currentid) ? itemindexof + 1 : ddlgoodstype.items.indexof(ddlgoodstype.items.findbyvalue(currentid)) + lastonecount;
		                            }
		                            ddlgoodstype.items.insert(itemindexof, listitem);//添加子節(jié)點(diǎn)
		                            lastonecount++;
		                        }
		                        if (parentkeylist.count > 0)//存在子節(jié)點(diǎn)時(shí)
		                        {
		                            currentid = parentkeylist.dequeue();
		                            parrentsign = parentsignlist.dequeue();
		                        }
		                        else
		                        {
		                            haschild = false;
		                        }
		                    }
		                    else
		                    {
		                        break;
		                    }
		                }
		                if (autodispose)
		                {
		                    _dropdowntree.dispose();
		                }
		            }
		        }
		        /**//// <summary>
		        /// 綁定連動(dòng)級(jí)的下拉菜單
		        /// </summary>
		        /// <param name="ddlgoodstype">傳進(jìn)一個(gè)被綁定的dropdownlist</param>
		        public void bindtodropdownlist(dropdownlist ddlgoodstype)
		        {
		            bindtodropdownlist(ddlgoodstype, string.empty,null, true);
		        }
		        /**//// <summary>
		        /// 綁定連動(dòng)級(jí)的下拉菜單
		        /// </summary>
		        /// <param name="ddlgoodstype">傳進(jìn)一個(gè)被綁定的dropdownlist</param>
		        /// <param name="removeid">被排除的id</param>
		        public void bindtodropdownlist(dropdownlist ddlgoodstype, string removeid)
		        {
		            bindtodropdownlist(ddlgoodstype, removeid,null, true);
		        }
		        /**//// <summary>
		        /// 綁定連動(dòng)級(jí)的下拉菜單
		        /// </summary>
		        /// <param name="ddlgoodstype">傳進(jìn)一個(gè)被綁定的dropdownlist</param>
		        /// <param name="removeid">被排除的id,若沒有,傳null</param>
		        /// <param name="parentid">起始父id</param>
		        public void bindtodropdownlist(dropdownlist ddlgoodstype, string removeid,string parentid)
		        {
		            bindtodropdownlist(ddlgoodstype, removeid,parentid, true);
		        }
		        #endregion
		    }
		}
		調(diào)用方法很簡單: 
	1.繼承自idropdowntree接口
	2.實(shí)現(xiàn)3個(gè)接口方法實(shí)現(xiàn)接口代碼示例[dispose方法自己實(shí)現(xiàn)],最主要的是自己實(shí)現(xiàn)獲得子級(jí)的方法	
		復(fù)制代碼代碼如下:
			idropdowntree 成員
		#region idropdowntree 成員
		public dictionary<string, string> getchildcategory(string parentid)
		{
		    string where = "parentid='" + parentid + "'";
		    if (string.isnullorempty(parentid))
		    {
		 where = "parentid is null or parentid='" + guid.empty + "'";
		    }
		    list<goodscategorybean> _goodscategorylist = selectlist(0, where, string.empty, false);
		    if (_goodscategorylist != null && _goodscategorylist.count > 0)
		    {
		 dictionary<string, string> categorylist = new dictionary<string, string>();
		 for (int i = 0; i < _goodscategorylist.count; i++)
		 {
		     categorylist.add(_goodscategorylist[i].id.tostring(), _goodscategorylist[i].gategoryname);
		 }
		 return categorylist;
		    }//51aspx.com
		    return null;
		}
		public interface.common.dropdowntree dropdowntree
		{
		    get { return new interface.common.dropdowntree(this); }
		}
		#endregion
		頁面調(diào)用代碼: 類名.dropdowntree.bindtodropdownlist(下拉控件id);	
		 
			希望本文所述對(duì)大家的asp.net程序設(shè)計(jì)有所幫助。