• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
编写人生
写写代码,写写人生
博客园    首页    新随笔    联系   管理    订阅  订阅
回复 寒枫天伤 - PSP 的问题
      今天收到“寒枫天伤 - PSP”的问题(老大,名字能不能简单点,好难打也),询问载体设计的问题,乱谈一下吧,不好你当胡扯蛋了。

载体的设计有俩个思路,一个是表格式的,一个是层次的。

      ms的ADO、DataSet就是表格式的,采用行、列组成的表,然后表集合之间建立联系。他更贴近关系型数据库的结构,因此使用简单、可以充分利用已经成熟的大量研究成果。缺点就是他是抽象的,不直观。

      通常的xml和O/R的设计就是层次的,局部来说也是行(集合)、列(属性)组成表(对象),区别是表(对象)之间不是平等的关系,而是建立了有点像树一样的结构。好处吗,编写代码的时候看着舒服些罗(不是我打击你),缺点吗,一沓子了,我最头大的是数据跟踪问题。

      我无法在一片文章中说明所有的事情,例如序列化、继承原则、CRUD、数据跟踪一大堆要处理的事情。

      先说说 IBindList和ICancelAddNew接口吧,IBindList是列表绑定的基础接口,他继承于IList接口,如果你想绑定到某个表格或者列表中,IList基本上够了(实际上数组和ICollection也可以),但IBindList提供是否能新增、编辑和删除的选项,还提供排序、查找等功能(我可没有实现这个复杂的功能,我使用表格本身的功能),最重要的是他提供了ListChanged事件,这个是你通知外界你的集合发生改变的最好途径,所以你的集合最好是实现IBindList,而不紧紧是IList。

      ICancelAddNew接口用在表格的编辑中,你使用表格的时候都知道你新建一行的时候可以按ESC键取消新建,实际内部的工作原理是:已经新建了行并添加到行集合,当你按ESC时,删除掉刚才的一行,所以你必须记住刚才新建的行是第多少行。 (如果没有记错的话,.net 1.1是没有这个接口的 ,.net 2.0才有)

      下面的代码是部分的集合代码(不能运行的),不要以为我能写多好的程序,其实我是抄System.ComponentModel.Collections.Generic.BindingList<T>的。


Using directives#region Using directives
using System;
using System.Collections;
using System.ComponentModel;
using Mango.Common.Data;
using Mango.Common.Properties;
using System.Reflection;
#endregion


namespace Mango.Common.Data
{
    
/**//// <summary> 行集合对象的基础类 </summary>
    public class DataRowCollectionBase : CollectionBase, IBindingList, IList, ICollection, IEnumerable, ICancelAddNew
    
{
        
// Fields
        private int addNewPos;
        
private bool hookingItems;
        
private PropertyChangedEventHandler onItemPropertyChanged;
        
private bool allowNew;
        
private bool allowEdit;
        
private bool allowRemove;

        
private Type _itemType;
        
private object _parent;

        
// Events
        public event AddingNewEventHandler AddingNew;
        
public event ListChangedEventHandler ListChanged;

        
类的初始化方法#region 类的初始化方法
        
/**//// <summary> 创建无父对象的集合 </summary>
        public DataRowCollectionBase()
        
{
            
this.addNewPos = -1;
            
this.allowNew = true;
            
this.allowEdit = true;
            
this.allowRemove = true;
        }


        
/**//// <summary> 创建集合,并为集合设置父对象 </summary>
        public DataRowCollectionBase(object parent) :this()
        
{
            _parent 
= parent;
        }

        
#endregion


        
AddNew相关方法#region AddNew相关方法
        
/**//// <summary>
        
/// 获取集合类的名细类型
        
/// </summary>

        protected virtual Type GetCollectionItemType()
        
{
            
if (_itemType == null)
            
{
                Type collType 
= this.GetType();
                
object[] ps = collType.GetCustomAttributes(typeof(DbCollectionAttribute), true);
                
if (ps == null || ps.Length == 0)
                    
throw new ApplicationException(string.Format(Resources.Error_NotSetDbCollAtt, collType.Name));
                _itemType 
= ((DbCollectionAttribute)ps[0]).TypeContainedInCollection;
            }


            
return _itemType;
        }


        
/**//// <summary> 引发 AddingNew 事件 </summary>
        protected virtual void OnAddingNew(AddingNewEventArgs e)
        
{
            
if (this.AddingNew != null)
            
{
                
this.AddingNew(this, e);
            }

        }


        
/**//// <summary> 返回新的对象 </summary>
        private object FireAddingNew()
        
{
            AddingNewEventArgs addNewArgs 
= new AddingNewEventArgs(null);
            
this.OnAddingNew(addNewArgs);
            
return addNewArgs.NewObject;
        }


        
/**//// <summary> 在向 DataRowCollectionBase 实例中插入新元素之前执行其他自定义进程。 </summary>
        protected override void OnInsert(int index, object value)
        
{
            
//检查新对象的父对象
            DataRowBase row = value as DataRowBase;
            
if (row != null)
            
{
                
if (row.Parent != null)
                    
throw new ArgumentException(Resources.Error_ColHaveParent);
            }


            
this.EndNew(this.addNewPos);
            
this.HookItem(value, true);

            
base.OnInsert(index, value);
        }


        
/**//// <summary> 在向 DataRowCollectionBase 实例中插入新元素之后执行其他自定义进程。 </summary>
        protected override void OnInsertComplete(int index, object value)
        
{
            
//设置新对象的父对象
            DataRowBase row = value as DataRowBase;
            
if (row != null)
                row.SetParent(_parent);

            
base.OnInsertComplete(index, value);
            
this.FireListChanged(ListChangedType.ItemAdded, index);
        }


        
/**//// <summary> 将新项添加到列表。</summary>
        object IBindingList.AddNew()
        
{
            
object newObject = this.AddNewCore();
            
this.addNewPos = (newObject != null) ? base.List.IndexOf(newObject) : -1;
            
return newObject;
        }


        
/**//// <summary> 将新项添加到列表。 </summary>
        protected virtual object AddNewCore()
        
{
            
object newObject = this.FireAddingNew();
            
if (newObject == null)
            
{
                newObject 
= Activator.CreateInstance(GetCollectionItemType());
                
//自动填充关键字
                IDataRow dataRow = newObject as IDataRow;
                
if (dataRow != null)
                
{
                    DataRowType t 
= dataRow.GetDataRowType();
                    PropertyInfo pi 
= t.GetPrimaryKeyProperty();
                    pi.SetValue(dataRow, Guid.NewGuid().ToString(
"N"), null);
                }

            }

            
base.List.Add(newObject);
            
return newObject;
        }

        
#endregion


        
ListChanged事件的支持#region ListChanged事件的支持
        
/**//// <summary> 引发 ListChanged 事件 </summary>
        protected virtual void OnListChanged(ListChangedEventArgs e)
        
{
            
if (this.ListChanged != null)
            
{
                
this.ListChanged(this, e);
            }

        }


        
//内部引发ListChanged事件
        private void FireListChanged(ListChangedType listChangedType, int index)
        
{
            
this.OnListChanged(new ListChangedEventArgs(listChangedType, index));
        }


        
//内部引发ListChanged事件
        private void FireListChanged(ListChangedType listChangedType, int index,string propertyName)
        
{
            PropertyDescriptor propDesc 
= TypeDescriptor.CreateProperty(GetCollectionItemType(), propertyName,typeof(string), null);
            
this.OnListChanged(new ListChangedEventArgs(listChangedType, index,propDesc));
        }


        
/**//// <summary> 返回/设置是否引发ListChanged中ItemChanged项目 </summary>
        public bool RaiseItemChangedEvents
        
{
            
get
            
{
                
return this.hookingItems;
            }

            
set
            
{
                
if (this.hookingItems != value)
                
{
                    
this.HookItems(false);
                    
this.hookingItems = value;
                    
this.HookItems(true);
                }

            }

        }


        
/**//// <summary> 引发重新更新事件 </summary>
        public void ResetBindings()
        
{
            
this.FireListChanged(ListChangedType.Reset, -1);
        }


        
/**//// <summary> 引发某个元素的改动事件 </summary>
        public void ResetItem(int position)
        
{
            
this.FireListChanged(ListChangedType.ItemChanged, position);
        }


        
//拦截/取消元素
        private void HookItem(object item, bool hook)
        
{
            
if (!this.hookingItems)
            
{
                
return;
            }

            
if (this.onItemPropertyChanged == null)
            
{
                
this.onItemPropertyChanged = new PropertyChangedEventHandler(this.OnItemPropertyChanged);
            }


            IPropertyChange tmp 
= item as IPropertyChange;
            
if (tmp != null)
            
{
                
if (hook)
                    tmp.PropertyChanged 
+= this.onItemPropertyChanged;
                
else
                    tmp.PropertyChanged 
-= this.onItemPropertyChanged;
            }

        }


        
//拦截/取消指定索引的元素
        private void HookItemAt(int index, bool hook)
        
{
            
if ((this.hookingItems && (index >= 0)) && (index < base.Count))
            
{
                
this.HookItem(base.List[index], hook);
            }

        }


        
//拦截/取消所有元素
        private void HookItems(bool hook)
        
{
            
if (!this.hookingItems)
                
return;

            IEnumerator e 
= base.GetEnumerator();
            
while (e.MoveNext())
            
{
                
object tmp = e.Current;
                
this.HookItem(tmp, hook);
            }

        }


        
//在元素发生改变时调用
        private void OnItemPropertyChanged(object sender, PropertyChangedEventArgs e)
        
{
            
this.FireListChanged(ListChangedType.ItemChanged, base.List.IndexOf(sender), e.PropertyName);
        }


        
#endregion


        
Clear相关方法#region Clear相关方法
        
/**//// <summary> 在清除 DataRowCollectionBase 中所有实例之前执行其他自定义进程。</summary>
        protected override void OnClear()
        
{
            
this.EndNew(this.addNewPos);
            
this.HookItems(false);

            
//删除父对象关联,注意:这里不能会滚操作
            DataRowBase row;
            
foreach (object item in InnerList)
            
{
                row 
= item as DataRowBase;
                
if (row != null)
                    row.SetParent(
null);
            }


            
base.OnClear();
        }


        
/**//// <summary> 在清除 DataRowCollectionBase 中所有实例之后执行其他自定义进程。 </summary>
        protected override void OnClearComplete()
        
{
            
base.OnClearComplete();
            
this.FireListChanged(ListChangedType.Reset, -1);
        }

        
#endregion


        
Remove相关方法#region Remove相关方法
        
/**//// <summary> 在向 DataRowCollectionBase 实例中移出元素之前执行其他自定义进程。 </summary>
        protected override void OnRemove(int index, object value)
        
{
            
this.EndNew(this.addNewPos);
            
this.HookItemAt(index, false);
            
base.OnRemove(index, value);
        }


        
/**//// <summary> 在向 DataRowCollectionBase 实例中移出元素之后执行其他自定义进程。 </summary>
        protected override void OnRemoveComplete(int index, object value)
        
{
            
//删除父对象关联
            DataRowBase row = value as DataRowBase;
            
if (row != null)
                row.SetParent(
null);

            
base.OnRemoveComplete(index, value);
            
this.FireListChanged(ListChangedType.ItemDeleted, index);
        }


        
#endregion


        
Set相关方法#region Set相关方法
        
/**//// <summary> 当在 DataRowCollectionBase 实例中设置值后执行其他自定义进程。</summary>
        protected override void OnSetComplete(int index, object oldValue, object newValue)
        
{
            
//删除旧对象的父对象
            DataRowBase oldRow = oldValue as DataRowBase;
            
if (oldRow != null)
                oldRow.SetParent(
null);

            
//设置新对象的父对象
            DataRowBase newRow = newValue as DataRowBase;
            
if (newRow != null)
                newRow.SetParent(_parent);

            
base.OnSetComplete(index, oldValue, newValue);
            
this.FireListChanged(ListChangedType.ItemChanged, index);
        }

        
#endregion


        
ICancelAddNew支持#region ICancelAddNew支持
        
/**//// <summary> 取消新建的行 </summary>
        public void CancelNew(int itemIndex)
        
{
            
if ((this.addNewPos >= 0) && (this.addNewPos == itemIndex))
            
{
                
this.RemoveAt(this.addNewPos);
                
this.addNewPos = -1;
            }

        }


        
/**//// <summary> 结束新建行的编辑 </summary>
        public void EndNew(int itemIndex)
        
{
            
if ((this.addNewPos >= 0) && (this.addNewPos == itemIndex))
            
{
                
this.addNewPos = -1;
            }

        }


        
#endregion


        
集合是否可以改动的支持#region 集合是否可以改动的支持
        
/**//// <summary> 获取/设置是否可以使用 AddNew 向列表中添加项。 </summary>
        public bool AllowNew
        
{
            
get
            
{
                
return ((IBindingList)this).AllowNew;
            }

            
set
            
{
                
this.allowNew = value;
            }

        }


        
/**//// <summary> 获取是否可以使用 AddNew 向列表中添加项。 </summary>
        bool IBindingList.AllowNew
        
{
            
get
            
{
                
return this.AllowNewCore;
            }

        }


        
/**//// <summary> 获取是否可以使用 AddNew 向列表中添加项。 </summary>
        protected virtual bool AllowNewCore
        
{
            
get
            
{
                
return this.allowNew;
            }

        }


        
/**//// <summary> 获取/设置是否可更新列表中的项。 </summary>
        public bool AllowEdit
        
{
            
get
            
{
                
return ((IBindingList)this).AllowEdit;
            }

            
set
            
{
                
this.allowEdit = value;
            }

        }


        
/**//// <summary> 获取是否可更新列表中的项。</summary>
        bool IBindingList.AllowEdit
        
{
            
get
            
{
                
return this.AllowEditCore;
            }

        }


        
/**//// <summary> 获取是否可更新列表中的项。 </summary>
        protected virtual bool AllowEditCore
        
{
            
get
            
{
                
return this.allowEdit;
            }

        }


        
/**//// <summary> 获取/设置是否可以使用 Remove 或 RemoveAt 从列表中移除项。 </summary>
        public bool AllowRemove
        
{
            
get
            
{
                
return ((IBindingList)this).AllowRemove;
            }

            
set
            
{
                
this.allowRemove = value;
            }

        }


        
/**//// <summary> 获取是否可以使用 Remove 或 RemoveAt 从列表中移除项。 </summary>
        bool IBindingList.AllowRemove
        
{
            
get
            
{
                
return this.AllowRemoveCore;
            }

        }


        
/**//// <summary> 获取是否可以使用 Remove 或 RemoveAt 从列表中移除项。 </summary>
        protected virtual bool AllowRemoveCore
        
{
            
get
            
{
                
return this.allowRemove;
            }

        }

        
#endregion


        
排序和查询功能的支持#region 排序和查询功能的支持
        
/**//// <summary> 获取当列表更改或列表中的项更改时是否引发 ListChanged 事件。 </summary>
        public bool SupportsChangeNotification
        
{
            
get
            
{
                
return this.SupportsChangeNotificationCore;
            }

        }


        
/**//// <summary> 获取当列表更改或列表中的项更改时是否引发 ListChanged 事件。 </summary>
        protected virtual bool SupportsChangeNotificationCore
        
{
            
get
            
{
                
return true;
            }

        }


        
/**//// <summary> 获取列表是否支持使用 Find 方法进行搜索。 </summary>
        public bool SupportsSearching
        
{
            
get
            
{
                
return this.SupportsSearchingCore;
            }

        }


        
/**//// <summary> 获取列表是否支持使用 Find 方法进行搜索。 </summary>
        protected virtual bool SupportsSearchingCore
        
{
            
get
            
{
                
return false;
            }

        }


        
/**//// <summary> 获取列表是否支持排序 </summary>
        public bool SupportsSorting
        
{
            
get
            
{
                
return this.SupportsSortingCore;
            }

        }


        
/**//// <summary> 获取列表是否支持排序 </summary>
        protected virtual bool SupportsSortingCore
        
{
            
get
            
{
                
return false;
            }

        }


        
/**//// <summary> 获取是否对列表中的项进行排序。</summary>
        public bool IsSorted
        
{
            
get
            
{
                
return this.IsSortedCore;
            }

        }


        
/**//// <summary> 获取是否对列表中的项进行排序。 </summary>
        protected virtual bool IsSortedCore
        
{
            
get
            
{
                
return false;
            }

        }


        
/**//// <summary> 获取正在用于排序的 PropertyDescriptor。 </summary>
        public PropertyDescriptor SortProperty
        
{
            
get
            
{
                
return this.SortPropertyCore;
            }

        }


        
/**//// <summary> 获取正在用于排序的 PropertyDescriptor。 </summary>
        protected virtual PropertyDescriptor SortPropertyCore
        
{
            
get
            
{
                
return null;
            }

        }


        
/**//// <summary> 获取排序的方向。 </summary>
        public ListSortDirection SortDirection
        
{
            
get
            
{
                
return this.SortDirectionCore;
            }

        }


        
/**//// <summary> 获取排序的方向。</summary>
        protected virtual ListSortDirection SortDirectionCore
        
{
            
get
            
{
                
return ListSortDirection.Ascending;
            }

        }


        
/**//// <summary> 根据 PropertyDescriptor 和 ListSortDirection 对列表进行排序。 </summary>
        public void ApplySort(PropertyDescriptor property, ListSortDirection direction)
        
{
            
this.ApplySortCore(property, direction);
        }


        
/**//// <summary> 根据 PropertyDescriptor 和 ListSortDirection 对列表进行排序。 </summary>
        protected virtual void ApplySortCore(PropertyDescriptor property, ListSortDirection direction)
        
{
            
throw new NotSupportedException();
        }


        
/**//// <summary> 使用 ApplySort 移除任何已应用的排序。 </summary>
        public void RemoveSort()
        
{
            
this.RemoveSortCore();
        }


        
/**//// <summary> 使用 ApplySort 移除任何已应用的排序。</summary>
        protected virtual void RemoveSortCore()
        
{
            
throw new NotSupportedException();
        }


        
/**//// <summary> 返回具有给定 PropertyDescriptor 的行的索引。 </summary>
        public int Find(PropertyDescriptor property, object key)
        
{
            
return this.FindCore(property, key);
        }


        
/**//// <summary> 返回具有给定 PropertyDescriptor 的行的索引。 </summary>
        protected virtual int FindCore(PropertyDescriptor property, object key)
        
{
            
throw new NotSupportedException();
        }


        
/**//// <summary> 将 PropertyDescriptor 添加到用于搜索的索引 </summary>
        void IBindingList.AddIndex(PropertyDescriptor property)
        
{
            
//
        }


        
/**//// <summary> 从用于搜索的索引中移除 PropertyDescriptor。 </summary>
        void IBindingList.RemoveIndex(PropertyDescriptor property)
        
{
            
//
        }

        
#endregion

    }

}

posted on 2005-01-09 10:32  编写人生  阅读(961)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3