最近要写一个网络报表。
众所周知,网络报表是程序员的一大噩梦。
幸运的是,了解了下需求,然后去网络上大概看了下报表相关的信息。
感觉微软自带的RDLC报表能完全满足需求了。
不过,受伤的总是我。
老大一句:“要做成组件,下次调用直接拖拉就行,不需要手动操作!”
...
请问:天黑是神马感觉?!
这句话的意思就是要完全动态生成报表(不了解rdlc报表的童鞋们去操作下试试就知道了)。
里面要用到的技术,我大概想了一下。
主要就是XML的架构定义和序列号,反序列话。其他的没什么。不过,就这2个技术,让我感觉顿时落入了地狱(两个我都不熟悉。)。
没办法,只能硬着头皮搞了。
第一个遇到的问题就是c#特性机制了。
稍微看了下,懂了,就先记录下一些自认为经典的例子吧(2个小时前,我连这个[]叫特性都不知道,或者说都没怎么见过这个运算符这个用。2个小时看了几个代码,尤其是下面示例的代码,就懂了!)。
不多说,如下:
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace ConsoleApplication1 { class Program { static void Main(string[] args) { BoardingCheckAttribute boardingCheck = null; object[] customAttributes = typeof(HumanPropertyBase).GetCustomAttributes(true); Console.WriteLine(typeof(HumanPropertyBase).IsSerializable); foreach (System.Attribute attribute in customAttributes) { if (attribute is BoardingCheckAttribute) { boardingCheck = attribute as BoardingCheckAttribute; Console.WriteLine(boardingCheck.Name + "'s ID is " + boardingCheck.ID + ", he/she wants to " + boardingCheck.Destination + " from " + boardingCheck.Departure + " by the plane " + boardingCheck.FlightNumber + ", his/her position is " + boardingCheck.PositionNumber + "."); } } Console.ReadLine(); } } [Serializable] [BoardingCheckAttribute("123","louis",22,12,"aaaa","ccccc")] public class HumanPropertyBase { public string Name { get; set; } public int Age { get; set; } public int Gender { get; set; } [Obsolete("I'm so old, don't kill me!", true)] public void Run(int speed) { // Running is good for health. } } public class BoardingCheckAttribute : Attribute { public string ID { get; private set; } public string Name { get; private set; } public int FlightNumber { get; private set; } public int PositionNumber { get; private set; } public string Departure { get; private set; } public string Destination { get; private set; } public BoardingCheckAttribute(string id, string name, int flightNumber, int positionNumber, string departure, string destination) { this.ID = id; this.Name = name; this.FlightNumber = flightNumber; this.PositionNumber = positionNumber; this.Departure = departure; this.Destination = destination; } } }
[System.AttributeUsage(System.AttributeTargets.Class | System.AttributeTargets.Struct, AllowMultiple = true) // multiuse attribute ] public class Author : System.Attribute { string name; public double version; public Author(string name) { this.name = name; version = 1.0; // Default value } public string GetName() { return name; } } [Author("H. Ackerman")] public class FirstClass { // ... } // No Author attribute public class SecondClass { // ... } [Author("H. Ackerman"), Author("M. Knott", version = 2.0)] public class ThirdClass { // ... } class TestAuthorAttribute { static void Main() { PrintAuthorInfo(typeof(FirstClass)); PrintAuthorInfo(typeof(SecondClass)); PrintAuthorInfo(typeof(ThirdClass)); } private static void PrintAuthorInfo(System.Type t) { System.Console.WriteLine("Author information for {0}", t); System.Attribute[] attrs = System.Attribute.GetCustomAttributes(t); // reflection foreach (System.Attribute attr in attrs) { if (attr is Author) { Author a = (Author)attr; System.Console.WriteLine(" {0}, version {1:f}", a.GetName(), a.version); } } } }
网上搜到的解释:(对应第二段代码)
使用反射访问属性(C# 编程指南)
如果没有检索自定义属性的信息和对其进行操作的方法,则定义自定义属性并将其放置在源代码中就没有意义。C# 具有一个反射系统,可用来检索用自定义属性定义的信息。主要方法是 GetCustomAttributes,它返回对象数组,这些对象在运行时等效于源代码属性。此方法具有多个重载版本。有关更多信息,请参见 Attribute。
属性规范,如:
C#
[Author("H. Ackerman", version = 1.1)]
class SampleClass
在概念上等效于:
C#
Author anonymousAuthorObject = new Author("H. Ackerman");
anonymousAuthorObject.version = 1.1;
【备:我是第一次看到这个写法,然后兴匆匆的去试了下类的属性,发现这个写法不使用了。
疑惑:难道这个写法只适用于特性的实例化么?如果高手见到,还请指教。】
但是,直到查询 SampleClass 以获取属性时才会执行此代码。对 SampleClass 调用 GetCustomAttributes 会导致按上述方式构造并初始化一个 Author 对象。如果类还有其他属性,则其他属性对象的以类似方式构造。然后 GetCustomAttributes 返回 Author 对象和数组中的任何其他属性对象。之后就可以对此数组进行迭代,确定根据每个数组元素的类型所应用的属性,并从属性对象中提取信息。