第一章都是讲泛型的,距离上一篇Effective C#的随笔已经是很久以前的事情了。。。
 
    今天Item4,讲的是泛型的类型推断功能。东西好不好,都是比较出来了,当然也不是绝对的好或者绝对的不好。
 
    首先上一段不用泛型的代码。 

   1: public static class XmlPersistenceManager
   2: {
   3:     public static object LoadFromFile(Type typeToLoad, string filePath)
   4:     {
   5:         XmlSerializer factory = new XmlSerializer(typeToLoad);
   6:         if (File.Exists(filePath)) {
   7:             using (TextReader tr = new StreamReader(filePath)) {
   8:                 object rVal = factory.Deserialize(tr);
   9:                 return rVal;
  10:             }
  11:         }
  12:         return default(object);//null
  13:     }
  14:     public static void SaveToFile(string filePath, object obj)
  15:     {
  16:         Type theType = obj.GetType();
  17:         XmlSerializer factory = new XmlSerializer(theType);
  18:         using (var sw = new StreamWriter(filePath, false)) {
  19:             factory.Serialize(sw, obj);
  20:         }
  21:     }
  22: }

两个方法,一个读取Xml生成Object的实例,另一个把一个obj保存成一个Xml。这里有几个缺点

①每次调用LoadFromFile方法,必须有一个类型转换,从Object转成自己要的类型,写的时候肯定不会报错的,因为Object是所有类型的基类,但是运行的时候,就不一定了~~ 。

②这是一个性能问题。每次调用这两个方法的时候,都重新new 了一个XmlSerializer对象。Framework的设计者是会尽量降低new对象的代价,但毕竟是需要创建,然后销毁一些零时的变量。

看到第二点,大家都会想到把XmlSerializer类型对象factory作为一个XmlPersistenceManager的静态成员变量。

看到第二点可能会写出下面的代码。

   1: public static class XmlPersistenceManager2
   2: {
   3:     private static XmlSerializer factory;
   4:     public static object LoadFromFile(Type typeToLoad, string filePath)
   5:     {
   6:         if (factory == null)
   7:             factory = new XmlSerializer(typeToLoad);
   8:  
   9:         if (File.Exists(filePath)) {
  10:             using (TextReader tr = new StreamReader(filePath)) {
  11:                 object rVal = factory.Deserialize(tr);
  12:                 return rVal;
  13:             }
  14:         }
  15:         return default(object);//null
  16:     }
  17:     public static void SaveToFile(string filePath, object obj)
  18:     {
  19:         Type theType = obj.GetType();
  20:         if (factory == null) factory = new XmlSerializer(theType);
  21:         using (var sw = new StreamWriter(filePath, false)) {
  22:             factory.Serialize(sw, obj);
  23:         }
  24:     }
  25: }

性能问题是解决了,但是,明显,有个bug。20行,先ClassA类型的obj调用,factory生成一个实例,木有问题;然后来一个ClassB类型的obj调用,factory != null ;然后,22行,调用,异常就来了。 原先我以为是不会出错的,充其量应该只是生成一个空的xml文件,但原文用了Exception这个词,然后自己测试了一下。证明,我错了,确实是Exception,再看一下代码,factory实例化的时候传入了参数theType。为什么要传这个参数呢?我想应该还是性能问题吧。new 一个 XmlSerializer 之后肯定不会只(反)序列化同类型的对象一次。

要解决这个bug也很容易,用一个Dictionary来存XmlSerializer对象。。。但是这样意味着要写更多代码,写更多编译器和JIT引擎可以帮你实现的代码。

接下来泛型上场,原文叫“correct answer”。

   1: public static class GenericXmlPersistenceManager<T>
   2: {
   3:     private static XmlSerializer factory;
   4:     public static T LoadFromFile(string filePath)
   5:     {
   6:         if (File.Exists(filePath)) {
   7:             using (XmlReader inputStream = XmlReader.Create(filePath)) {
   8:                 if (factory == null) factory = new XmlSerializer(typeof(T));
   9:                 T rVal = (T)factory.Deserialize(inputStream);
  10:                 return rVal;
  11:             }
  12:         }
  13:         return default(T);
  14:     }
  15:     public static void SaveToFile(string filePath, T data)
  16:     {
  17:         using (XmlWriter writer = XmlWriter.Create(filePath)) {
  18:             if (factory == null) factory = new XmlSerializer(typeof(T));
  19:             factory.Serialize(writer, data);
  20:         }
  21:     }
  22: }

代码上和最初的版本没有什么太大差异。解决了原先的几个问题。

①类型转换。泛型类中的LoadFromFile方法,返回的类型其实已经被限定了,就是T类型,至于T具体是什么类型,就看自己在调用的时候尖括号之间写的具体的值了。

②性能问题和那个Exception bug。用了静态变量,缓存了XmlSerializer对象,当序列化同个类型的obj的时候,不需再去重新new一个XmlSerializer。并且,如果传入了不同类型的obj,也会重新new一个对应类型的XmlSerializer 类型的factory,这样就不会报错。(想到一个问题,写完之后查资料了解一下)。

最后一段:

很多时候如果用了Type类型的参数,通常都可以定义出一个泛型的版本。编译器就会 “Create the Specific version for you.”。

转载于:https://www.cnblogs.com/sheldon-lou/p/3443734.html

用泛型来实现编译时期的类型推断相关推荐

  1. Java 8新特性探究(6):泛型的目标类型推断

    简单理解泛型 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.通俗点将就是"类型的变量".这种类型变量可以用在类.接口和方法 ...

  2. Java 泛型是如何工作的?类型擦除是什么?

    目录 1. 问题 2. 解答 1. 问题 泛型是如何工作的?类型擦除是什么意思? 2. 解答 泛型通过类型擦除来将变量变为一个类型,编译器在编译时擦出了所有类型相关的信息,所以在运行时不存在任何类型相 ...

  3. Java SE7新特性之泛型实例创建时的类型推断

    Java SE7新特性之泛型实例创建时的类型推断 标签: JDK7泛型 2014-01-04 15:34 6015人阅读 评论(0) 收藏 举报  分类: Java基础(9)  只要编译器从上下文中能 ...

  4. 实用typescript_TypeScript 泛型使用2-常见的工具类型

    TypeScript 泛型使用2-常见的工具类型 hucheng91:TypeScript 泛型使用1​zhuanlan.zhihu.com 接上篇,这篇梳理下TypeScript 封装了很多常见的基 ...

  5. 设计一个Windows应用程序,要求如下: 构造一个产品基类。 分别定义家电、日用百货、衣服等派生类,具体要求有不同的特征和行为。 定义一个泛型货架类,约束参数类型为产品

    设计一个Windows应用程序,要求如下: 构造一个产品基类. 分别定义家电.日用百货.衣服等派生类,具体要求有不同的特征和行为. 定义一个泛型货架类,约束参数类型为产品类.该泛型的货架类包括一个泛型 ...

  6. C#设计一个Windows应用程序,要求如下。 ①构造一个产品基类。 ②分别定义家电、日用百货、衣服等派生类,要求具有不同的特征和行为。 ③定义一个泛型货架类,约束参数类型为产品类。该泛型的货架类包

    设计一个Windows应用程序,要求如下. 1.构造一个产品基类. 2.分别定义家电.日用百货.衣服等派生类,要求具有不同的特征和行为. 3.定义一个泛型货架类,约束参数类型为产品类.该泛型的货架类包 ...

  7. Java 8 - 04 类型检查、类型推断以及限制

    文章目录 Pre 类型检查 同样的 Lambda,不同的函数式接口 菱形运算符 特殊的void兼容规则 类型推断 使用局部变量 Pre 当我们第一次提到Lambda表达式时,说它可以为函数式接口生成一 ...

  8. Java 10 实战第 1 篇:局部变量类型推断

    现在 Java 9 被遗弃了直接升级到了 Java 10,之前也发过 Java 10 新特性的文章,现在是开始实战 Java 10 的时候了. 今天要实战的是 Java 10 中最重要的特性:局部变量 ...

  9. java 鲜为人知的知识点_鲜为人知的Java 8功能:广义目标类型推断

    java 鲜为人知的知识点 遍历Java 8的功能列表 , 广义目标类型推断使我震惊,因为它是一个特别有趣,鲜为人知的瑰宝. 看起来Java语言设计人员将减轻过去使用泛型(Java 5-7)时遇到的某 ...

最新文章

  1. 校园安全责任重大 安防守护迭代升级
  2. jq发送动态变量_山东体育学院康复生物力学团队发文,探索手机行为双任务对动态稳定性控制的影响...
  3. kafka修改分区数_大数据技术:解析SparkStreaming和Kafka集成的两种方式
  4. OAuth2 实现单点登录 SSO
  5. Ubuntu中安装FastDFS
  6. 优酷背后的大数据秘密:资源弹性,可支撑EB级存储
  7. 消息长度_【消息】听说咱安阳的第一条封闭外环即将全线通车了?是的,长度相当于北京五环...
  8. rvest | 网络爬虫初步——使用CSS选择器
  9. 在web项目启动时,使用监听器来执行某个方法
  10. Objective-c 中 nil, Nil, NULL和NSNull的区别
  11. MySQL出现:ERROR 3 (HY000): Error writing file '/tmp/MYbEd05t' (Errcode: 28)
  12. 华为的哪个字体像苹果的_华为默认字体是什么字体
  13. InstallShield2022程序构建可靠
  14. c语言上机ex11答案,全国计算机等级考试C语言南开100题(上机必考).doc
  15. Hark的数据结构与算法练习之堆排序
  16. 若说耳机世界里有一股清流,那这款QCY耳机肯定是其中之一
  17. 7-20 sdust-Java-字符串集合求并集
  18. android adb 命令汇总
  19. GD25Qxxx使用笔记
  20. Week of 2.7

热门文章

  1. 根据服务器ip地址查看虚拟目录,IIS虚拟目录实现与文件服务器网络驱动器映射共享...
  2. c语言保存图片image,iOS 保存图片到【自定义相册】
  3. iphone字体_iOS 13终于能换花式字体了?!发在朋友圈里真的超好看!
  4. c语言字符串数组的合并,C语言实现合并字符串
  5. 企业人员管理项目咨询_祝贺直线管理咨询与振野智能营销咨询项目启动
  6. 无缝融入 Kubernetes 生态 | 云原生网关支持 Ingress 资源
  7. Knative 实战:如何在 Knative 中配置自定义域名及路由规则
  8. Knative 实践:从源代码到服务的自动化部署
  9. linux运维技巧,Linux运维需要掌握的17个实用技巧
  10. Java 中pdf部分内容加边线_Java 在PDF中添加骑缝章示例解析