在c#中要扩展一个现有类非常easy,比方这样:
?
1
2
3
4
5
6
7
public static class Utils
{
    public static void PrintToConsole(this string strSrc)
    {
        Console.WriteLine(strSrc);
    }  
}
这样就为String类加入了一个PrintToConsole的方法。用法例如以下:
?
1
2
3
4
5
6
7
class MainClass
{
    public static void Main
(
string[]
args)
    {
        "Hello
World!"
.PrintToConsole();           
    }
}
在objective-C中,也有类似的处理办法:
StringUtils.h 定义部分
  1. #import <Foundation/Foundation.h>
  2. @interface NSString(ExtNSString)
  3. -(void) PrintToConSole;
  4. @end
解释:@interface NSString(ExtNSString) 表示ExtNSString这个类将会扩展NSString,会为其添加一些通用的额外方法。
 
StringUtils.m 实现部分
  1. #import "StringUtils.h"
  2. @implementation NSString(ExtNSString)
  3. -(void) PrintToConSole
  4. {
  5. NSLog(@"%@",self);
  6. }
  7. @end
用法例如以下:
  1. #import <Foundation/Foundation.h>
  2. #import "StringUtils.h"
  3. int main (int argc, const charchar * argv[]) {
  4. NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
  5. NSString* str = @"Hello World!";
  6. [str PrintToConSole];
  7. [pool drain];
  8. return 0;
  9. }
只是有一点要特别注意:c#中假设开发者添加的扩展方法跟.net框架自带的现有方法重名,实际执行时将以系统自带的现有方法为准。但在obj-C中,这样的情况下开发者新添加的重名方法会覆盖系统原有的方法,并且没有不论什么提示!

一个好的习惯是为全部扩展方法(包含类名),都加一个特殊的前缀或后缀。以避免重名。

 
下一个话题:partial class
做过asp.net开发的程序猿都知道,c#中的partial class能够方便的将同一个类的代码,分散在多个不同的物理文件里,编译器在编译时能自己主动将它们合并。这是一个非常棒的功能。在团队开发中我常常把一个类的不同业务模块,分散成几个不同的物理文件(比方class_jimmy.cs。class_mike.cs...),然后jimmy仅仅在class_jimmy.cs中写代码。mike仅仅在class_mike.cs中写代码,在非常大程度上这样能够降低(或避免)终于svn提交合并时的冲突。
表面上看,partial class与扩展方法是风马牛不相及的二个概念。可是在obj-C中,这二个事实上是一回事。

场景:比方一个商城系统,对产品的增、删、改定义。我想单独放到文件Product.h中,而对订单的处理,我想单独放到文件Order.h中。可是这些跟业务相关的处理,我想在逻辑上把它们都归到同一个类BLL.h中。
看看obj-C中的做法:(主要是看几个文件是怎样组织成一个类的。代码仅仅是演示样例而已)
1、先定义BLL.h (主要用于放一些成员变量。基本上仅仅是一个壳而已)
  1. #import <Foundation/Foundation.h>
  2. @interface BLL : NSObject {
  3. NSString* connStr;
  4. }
  5. -(void) setConnString:(NSString*) connString;
  6. -(NSString*) connString;
  7. @end
BLL.m实现
  1. #import "BLL.h"
  2. @implementation BLL
  3. -(void) setConnString:(NSString *)connString
  4. {
  5. connStr = connString;
  6. }
  7. -(NSString*) connString
  8. {
  9. return connStr;
  10. }
  11. -(void) dealloc
  12. {
  13. [connStr release];
  14. [super dealloc];
  15. }
  16. @end
2、再定义Product.h用来扩展BLL类
  1. #import <Foundation/Foundation.h>
  2. #import "BLL.h"
  3. @interface BLL(Product)
  4. -(void) addProduct: (NSString* )productName productNo:(NSString*)proNo;
  5. -(void) deleteProduct:(NSString*) productNo;
  6. @end
Product.m
  1. #import "Product.h"
  2. #import "BLL.h"
  3. @implementation BLL(Product)
  4. -(void) addProduct: (NSString* )productName productNo:(NSString*)proNo
  5. {
  6. NSLog(@"connString=%@",connStr);//输出Bll.h中定义的成员connStr
  7. NSLog(@"addProduct success! productName:%@,productNo:%@",productName,proNo);
  8. }
  9. -(void) deleteProduct:(NSString*) productNo
  10. {
  11. NSLog(@"connString=%@",[self connString]);//也能够用属性来訪问
  12. NSLog(@"deleteProduct success! productNo:%@",productNo);
  13. }
  14. @end

3、定义Order.h继续扩展BLL类

  1. #import <Foundation/Foundation.h>
  2. #import "BLL.h"
  3. @interface BLL(Order)
  4. -(void) createOrder:(NSString*) productNo quantity:(int) amount;
  5. @end

Order.m

  1. #import "Order.h"
  2. @implementation BLL(Order)
  3. -(void) createOrder:(NSString*) productNo quantity:(int) amount
  4. {
  5. NSLog(@"thank you for order our product. productNo:%@,quantity:%d",productNo,amount);
  6. }
  7. @end

因为Product类与Order类都是扩展自BLL类。所以这三个类在逻辑上都是同一个类BLL,最后来看看怎样使用:

  1. #import <Foundation/Foundation.h>
  2. #import "BLL.h"
  3. #import "Product.h"
  4. #import "Order.h"
  5. int main (int argc, const charchar * argv[]) {
  6. NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
  7. BLL *bll = [[BLL alloc] init];
  8. bll.connString = @"I am connection string.";
  9. [bll addProduct:@"iphone4" productNo:@"0001"];//调用Product.h中定义的方法
  10. [bll createOrder:@"0001" quantity:5];  //调用Order.h中定义的方法
  11. [bll deleteProduct:@"0001"];
  12. [bll release];
  13. [pool drain];
  14. return 0;
  15. }

执行结果:

2011-02-26 22:29:30.369 Demo[1292:a0f] connString=I am connection string.

2011-02-26 22:29:30.376 Demo[1292:a0f] addProduct success! productName:iphone4,productNo:0001

2011-02-26 22:29:30.378 Demo[1292:a0f] thank you for order our product. productNo:0001,quantity:5

2011-02-26 22:29:30.379 Demo[1292:a0f] connString=I am connection string.

2011-02-26 22:29:30.380 Demo[1292:a0f] deleteProduct success! productNo:0001

皆大欢喜。非常多语言和技术真是“一门通。处处通”,或许:c#中的"扩展方法"与"部分类"的设计灵感正是来自objective-C。

objective-C中的扩展方法与partial class的更多相关文章

  1. C#3.0中的扩展方法

    在实际应用中,开发者完成代码的编译后,除非重新编译更改后的代码,否则开发者很难在原有代码中添加新的功能. 在C#3.0中,提供了一个扩展方法的新特性,可以使得开发者在编译后的程序集里边添加相关的方法, ...

  2. 记录C#中的扩展方法

    C#中的扩展方法. 系统自带的类型,我们无法去修改: 修改源代码需要较大的精力,而且可能会带来错误: 我们只是需要一个或者较少的几个方法,修改源代码费时费力: 被扩展的类是sealed的,不能被继承: ...

  3. C#编程(六十一)------------LINQ中的扩展方法

    原文链接: http://blog.csdn.net/shanyongxu/article/details/47208401 LINQ中的扩展方法 LINQ中where扩展方法,要想使用,必须导入us ...

  4. C#中的扩展方法(向已有类添加方法,但无需创建新的派生类型)

    C#中的扩展方法 扩展方法使你能够向现有类型"添加"方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型. 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样 ...

  5. C#中的扩展方法详解

    “扩展方法使您能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型.”这是msdn上说的,也就是你可以对String,Int,DataRow,DataTable等这些类 ...

  6. C#中的扩展方法

    扩展方法使你能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型. 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样进行调用. 以上是msdn官网对扩展方 ...

  7. Enum扩展及MVC中DropDownListFor扩展方法的使用

    public enum SearchState { /// <summary> /// 全部 /// </summary> [Description("全部" ...

  8. 19.C#逐一介绍IEnumerable和IEnumerable<T>中的扩展方法(10.3-10.5)

    今天没有太多的言语,只有代码,扩展方法多得太多,不能一一列完,书中一些,看多了也就会使用了. //Enumerable.Range 返回起始到结束范围,是一个Enumrable<int>类 ...

  9. MVC 中使用扩展方法

     扩展方法(Extension Method)是给那些不是你拥有.因而不能直接修改的类添加方法的一种方便的办法. 一.使用扩展方法 1.定义一个购物车的类-ShoppingCart using Sys ...

随机推荐

  1. js中立即执行函数写法理解

    在理解了一些函数基本概念后,回头看看( function(){…} )()和( function (){…} () )这两种立即执行函数的写法,最初我以为是一个括号包裹匿名函数, 并后面加个括号立即调 ...

  2. 我在16aspx网上下载了个C#源码,如何能在我自己的计算机上跑起来,很急!求详细操作过程!

    先搞清楚是WINDOWS程序还是WEB程序.

  3. hdu3001(状态压缩DP)

    hdu3001 题意 选择从任意一点出发,经过所有点的最小花费(经过每个点的次数不能多于 2 次). 分析 类似于 poj3311 经过每个点的次数有限制,考虑用三进制数存储每个点被访问过的次数,其它 ...

  4. 模拟【P1650】 田忌赛马

    顾z 你没有发现两个字里的blog都不一样嘛 qwq 题目描述--->p1650 田忌赛马 先%dalao sto GMPotlc orz 他教给的我,征求意见后终于来水题解. 分析 我们需要知 ...

  5. 十. 图形界面(GUI)设计1.图形界面设计基础

    早先程序使用最简单的输入输出方式,用户在键盘输入数据,程序将信息输出在屏幕上.现代程序要求使用图形用户界面(Graphical User Interface,GUI),界面中有菜单.按钮等,用户通过鼠 ...

  6. Visio中汇总两个箭头

    RT,相似度和信任度矩阵融合,两个箭头,选中其中一个,可以选格式--线条--终点选无,或者在快捷那选线条.

  7. ASIHTTPRequest学习(三)

    刚刚开始学习ASIHttpRequest,今天通过自己写的一个小demo分享一下学习心得. 首先,要想在ios项目中使用ASIHttpRequest,必须添加下列框架和类库: ASIHttpReque ...

  8. Android学Jni/Ndk 开发记录(一)

      治疗拖延症的唯一办法就是:一想起些什么 / 要做些什么就 TM 立马去做! 是的,突然想起我不会 JNI.NDK 开发.解决办法:立马去学! 一:配置 NDK 环境 下载 NDK 写入到配置文件 ...

  9. ios学习流水账2

    1.UISearchBar自定义背景.取消按钮中文设置 UISearchBar *seachBar=[[UISearchBar alloc] init]; //修改搜索框背景 seachBar.bac ...

  10. 跟着Sedgewick学算法(week 1 UnionFind)

    发现笔记转过来,没有图的~~~~~~~~~~~悲剧,给出共享笔记链接 https://www.evernote.com/pub/yanbinliu/algorithm 很久之前就在coursera看到 ...