网格数据绑定问题

如何解决网格数据绑定问题

| 我们的应用程序会在特定结构中生成大量结果。问题是,如果要在DataGrid中显示它,则必须创建一个DataTable并将其设置为将使用内存的网格的dataSource。因此,我创建了一个我创建的类的BindingList(称为myRow),在myRow的结构中,我拥有了所有需要的字段,这些字段作为属性指向实际结果列表中的值。但是问题在于用户可以添加自定义列结果列表;我无法动态更改myRow的属性,并且我不想使用DataTable(因为它将与我的实际结果重复),并且如果我直接在dataGrid中创建自定义列并逐个设置其值-cell,内存中网格的大小真的很高(我认为这是因为逐个单元设置值会导致为每个单元保存单元的属性,而不是更大的标准)。那么,有谁知道如何使用不同于将属性用作列的策略来创建行类,以便可以在运行时动态设置列数?     

解决方法

我认为这可以通过使用ѭ0来完成。 坏消息是:我以前从未做过,也无法提供太多帮助 好消息是:我在这里找到一个示例:DataGridView没有显示实现ICustomTypeDescriptor的对象的属性 //编辑 我使用代码(请参见上面的链接)构建了一个示例,该示例如何避免每个对象使用字典...
public class myRow
{
    //your data storage class ... 
    public string txt { get; set; }
    public int id { get; set; }
}
public class MyView:ICustomTypeDescriptor
{//your extendable view class ...
    private static PropertyDescriptorCollection props = null;
    static MyView()
    {
        TypeDescriptionProvider defaultProvider = TypeDescriptor.GetProvider(typeof(MyView));
        props = new PropertyDescriptorCollection(defaultProvider.GetTypeDescriptor(typeof(MyView)).GetProperties().Cast<PropertyDescriptor>().ToArray(),true);
    }

    public static void addProperty(string name,DataTable dt,Func<DataRow,object> getter,Action<DataRow,object> setter,Func<DataTable,MyView,DataRow> rowSelector,Type PropType)
    {
        List<PropertyDescriptor> tmp;
        if (props != null) tmp = props.Cast<PropertyDescriptor>().ToList();
        else tmp = new List<PropertyDescriptor>();
        PropertyDescriptor pd = TypeDescriptor.CreateProperty(typeof(MyView),name,PropType,null);
        pd = new MyViewPropertyDescriptor(pd,dt,getter,setter,rowSelector,PropType);
        tmp.Add(pd);
        props = new PropertyDescriptorCollection(tmp.ToArray(),true);
    }

    //the data storage obj this view is referencing
    public myRow obj;

    public string TXT { // view-member known at compile time
        get { return obj.txt; }
        set { obj.txt = value; }
    }

    internal class MyViewPropertyDescriptor : PropertyDescriptor
    {   // an example property descriptor that can link to data in a DataTable ... 
        DataTable dt;
        Func<DataRow,object> getter;
        Action<DataRow,object> setter;
        Func<DataTable,DataRow> rowSelector;
        Type type;
        public MyViewPropertyDescriptor(PropertyDescriptor descr,Type PropType)
            : base(descr)
        {
            this.dt = dt; // storage for additional data referenced by this property
            this.getter = getter; //a getter that will take a DataRow,and extract the property value
            this.setter = setter; //a setter that will take a DataRow and a value
            this.rowSelector = rowSelector;//a row selector ... takes a dataset and the view object and has to return the assiciated datarow
            this.type = PropType; // the type of this property
        }

        public override object GetValue(object component)
        {
            // using row selector and getter to return the current value ... you should add errorhandling here
            return getter(rowSelector(dt,(MyView)component));
        }
        public override void SetValue(object component,object value)
        {   // the setter ... needs errorhandling too
            setter(rowSelector(dt,(MyView)component),value);
        }
        public override void ResetValue(object component)
        {

        }
        public override bool CanResetValue(object component)
        {
            return false;
        }
        public override bool ShouldSerializeValue(object component)
        {
            return false;
        }
        public override Type PropertyType
        {
            get { return type; }
        }
        public override bool IsReadOnly
        {
            get { return false; }
        }
        public override Type ComponentType
        {
            get { return typeof(MyView); }
        }

    }

    ICustomTypeDescriptor defaultDescriptor = TypeDescriptor.GetProvider(typeof(MyView)).GetTypeDescriptor(typeof(MyView));
    AttributeCollection ICustomTypeDescriptor.GetAttributes()
    {
        return defaultDescriptor.GetAttributes();
    }
    string ICustomTypeDescriptor.GetClassName()
    {
        return defaultDescriptor.GetClassName();
    }
    string ICustomTypeDescriptor.GetComponentName()
    {
        return defaultDescriptor.GetComponentName();
    }
    TypeConverter ICustomTypeDescriptor.GetConverter()
    {
        return defaultDescriptor.GetConverter();
    }
    EventDescriptor ICustomTypeDescriptor.GetDefaultEvent()
    {
        return defaultDescriptor.GetDefaultEvent();
    }
    PropertyDescriptor ICustomTypeDescriptor.GetDefaultProperty()
    {
        return defaultDescriptor.GetDefaultProperty();
    }
    object ICustomTypeDescriptor.GetEditor(Type editorBaseType)
    {
        return defaultDescriptor.GetEditor(editorBaseType);
    }
    EventDescriptorCollection ICustomTypeDescriptor.GetEvents(Attribute[] attributes)
    {
        return defaultDescriptor.GetEvents(attributes);
    }
    EventDescriptorCollection ICustomTypeDescriptor.GetEvents()
    {
        return defaultDescriptor.GetEvents();
    }
    PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties(Attribute[] attributes)
    {
        return props; // should really be filtered,but meh!
    }
    PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties()
    {
        return props;
    }
    object ICustomTypeDescriptor.GetPropertyOwner(PropertyDescriptor pd)
    {
        return this;
    }

}
还有一个利用这一点的小例子...
private void button1_Click(object sender,EventArgs e)
{
    if (dataGridView1.DataSource == null)
    {
        List<myRow> data = new List<myRow>();
        data.Add(new myRow { id = 1,txt = \"test 1\" });
        data.Add(new myRow { id = 2,txt = \"test 2\" });
        data.Add(new myRow { id = 3,txt = \"test 3\" });
        DataTable dt = new DataTable();
        dt.Columns.Add(\"id\",typeof(int));
        dt.Columns.Add(\"additionalData1\",typeof(int));
        dt.Columns.Add(\"additionalData2\",typeof(int));
        Random rnd = new Random();
        foreach (var item in data)
        {
            dt.Rows.Add(new object[] { item.id,rnd.Next(),rnd.Next() });
        }
        MyView.addProperty(\"additionalData1\",row => row[\"additionalData1\"],(row,val) => row[\"additionalData1\"] = val,(tab,v) => tab.Rows.OfType<DataRow>().First(x => x[\"id\"].Equals(v.obj.id)),typeof(int));
        MyView.addProperty(\"additionalData2\",row => row[\"additionalData2\"],val) => row[\"additionalData2\"] = val,typeof(int));

        dataGridView1.DataSource = new BindingList<MyView>(data.Select(x => new MyView { obj = x }).ToList());
    }
}
当然,您将需要提供更好的rowSelector或将数据表替换为哈希表或所需的任何数据结构...仅是示例     

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)&gt; insert overwrite table dwd_trade_cart_add_inc &gt; select data.id, &gt; data.user_id, &gt; data.course_id, &gt; date_format(
错误1 hive (edu)&gt; insert into huanhuan values(1,&#39;haoge&#39;); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive&gt; show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 &lt;configuration&gt; &lt;property&gt; &lt;name&gt;yarn.nodemanager.res