专栏首页菩提树下的杨过结构struct(值类型)在实际应用要注意的二点:

结构struct(值类型)在实际应用要注意的二点:

.Net中的数据类型大致可以分为二类:一类是值类型,一类是引用类型;结构(struct)是值类型,从性能上考虑值类型更有优势(关于值类型与引用类型的详细讨论不在本文范围内,大家可以去查阅相关资料).对于一些特定场合:比如仅需要存储数据,不需要体现具体方法的时候,建议大家用struct来代替class,但在使用过程中,有几个容易被忽视的细节.

1.使用struct存储数据做为数据源,与数据显示控件绑定时:

代码如下:

Code

protected void Page_Load(object sender, EventArgs e)
        {
            List<MyStruct3> _list = new List<MyStruct3>();
            _list.Add(new MyStruct3() { Name = "abc", Value = "123" });
            _list.Add(new MyStruct3() { Name = "cde", Value = "345" });

 this.Repeater1.DataSource = _list;
 this.Repeater1.DataBind();
 
        }

 public struct MyStruct3
        {
 public string Name;
 public string Value;

        }

前端aspx关键代码:

Code

<asp:Repeater ID="Repeater1" runat="server">
 <ItemTemplate>
 <%# Eval("Name") %><br/>;
 </ItemTemplate>
 </asp:Repeater>

编译时一切正常,但是运行后,报类似如下错误:

DataBinding:“Test._Default+MyStruct3”不包含名为“Name”的属性

咋整?把MyStruct3的定义改成这样:

Code

public struct MyStruct3
        {
 public string Name{set;get;}
 public string Value { set; get; }

        }

即:我们把字段(Field)改成属性(property),再次运行,一切OK (应该是跟<%# Eval(...)%>采用反射机制有关)

2.结构的构造函数问题

看如下代码

Code

public struct MyStruct {
 private string _name;
 private string _value;

 public string Name {
 set { _name = value; }
 get { return _name; }
            }

 public string Value
            {
 set { _value = value; }
 get { return _value; }
            }

 public MyStruct(string pName, string pValue) 
            {
                _name = pName;
                _value = pValue;
            }
        }

一切跟在Class中写的一样,没什么特别的,但是我们改成下面的写法:

Code

public struct MyStruct
        {
 public string Name { set; get; }
 public string Value { set; get; }

 public MyStruct(string pName, string pValue) 
            {
                Name = pName;
                Value = pValue;               
            }
        }

即利用c#3.0的自动属性,简化了一下代码,这次编译时vs却提示有错: "错误 在控制返回到调用程序之前,自动实现的属性“Test._Default.MyStruct2.Value”的支持字段必须完全赋值。请考虑从构造函数初始值设定项中调用默认构造函数。 " "在给“this”对象的所有字段赋值之前,无法使用该对象"

究其原因,我们用Reflector看下编译器是如何处理"自动属性"的,先把结构改成普通的类(目的是让编译先通过,好观察最终生成的代码

Code

 public class MyClass
        {
 public string Name{set;get;}
 public string Value { set; get; }

 public MyClass(string pName, string pValue) 
            {
                Name = pName;
                Value = pValue;
            }

        }

用Reflector反编译成C# 1.0后,代码如下:

Code

public class MyClass
{
 // Fields
    [CompilerGenerated]
 private string <Name>k__BackingField;
    [CompilerGenerated]
 private string <Value>k__BackingField;

 // Methods
 public MyClass(string pName, string pValue)
    {
 base..ctor();
 this.Name = pName;
 this.Value = pValue;
 return;
    }

 // Properties
 public string Name
    {
        [CompilerGenerated]
 get
        {
 string str;
            str = this.<Name>k__BackingField;
        Label_0009:
 return str;
        }
        [CompilerGenerated]
 set
        {
 this.<Name>k__BackingField = value;
 return;
        }
    }

 public string Value
    {
        [CompilerGenerated]
 get
        {
 string str;
            str = this.<Value>k__BackingField;
        Label_0009:
 return str;
        }
        [CompilerGenerated]
 set
        {
 this.<Value>k__BackingField = value;
 return;
        }
    }
}

观察一下构造函数,变成了 ... base..ctor(); this.Name = pName; ...

关键就在这里:对于类来讲,并不要求在访问类的实例之前对所有成员赋值,所以这里引用this是合法的;而值类型要求在使用前必须对所有成员赋值,所以值类型如果在构造函数中直接给自动属性赋值,这里this代表的就是结构本身,而在此之前自动生成的二个私有成员private string <Name>k__BackingField和private string <Value>k__BackingField还没赋值,因此报错也就是合情合理了

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • .Net3.0中的自动属性(示例)

    using System; namespace LinqDemo {     class Program     {         stat...

    菩提树下的杨过
  • “default关键字”与“序列化传输”的注意事项

    注:此乃“流水帐”式的水文,营养成分较低,高手请自动无视以下内容,否则引起消化不良等症状等,一概不管 ^_^ c#自从3.0开始,提供了很多便捷的语法特性(俗称...

    菩提树下的杨过
  • AS3中的单件(Singleton)模式

    单件(singleton)模式在c#中是最容易实现的模式,其主要用意就在于限制使用者用new来创建多个实例。但在as3中,构造函数必须是public的(语法本身...

    菩提树下的杨过
  • junit测试和spring整合

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 ...

    逝兮诚
  • 基于JQuery EasyUI的WebMVC控件封装(含源码)

      JQuery EasyUI类库,大家不会陌生,出来已经有很多年了。个人感觉还是很好用的,作者更新频率也很快,bug也及时修复。   最近在整理以前的代码,找...

    用户1219352
  • 设计模式学习 - 适配器模式

    许杨淼淼
  • 通过扩展改善ASP.NET MVC的验证机制[实现篇]

    在《使用篇》中我们谈到扩展的验证编程方式,并且演示了本解决方案的三大特性:消息提供机制的分离、多语言的支持和多验证规则的支持,我们现在来看看这样的验证解决方案最...

    蒋金楠
  • 贴心 | Visual Studio 2019 空引用异常的改进

    空引用异常是最常见的程序错误之一。微软昨天发布的VS2019对此有了非常贴心的改进,我们来看看吧!

    Edi Wang
  • 【asp.net core 系列】13 Identity 身份验证入门

    通过前两篇我们实现了如何在Service层如何访问数据,以及如何运用简单的加密算法对数据加密。这一篇我们将探索如何实现asp.net core的身份验证。

    程序员小高
  • 看完这篇文章再也不怕面试被问@transactional不生效的原因了

    代理设计模式,用于无侵入性地增强方法功能。代理模式是指,目标类的方法执行,需要委托给代理类执行,代理类可以在执行目标方法前/后,处理一些其它事情,这样就可以起到...

    业余草

扫码关注云+社区

领取腾讯云代金券