在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. HDU 1863 畅通工程 最小生成树

    思路: 比较典型的最小生成树的题目了..在这里用求最小生成树的经典算法K(Kruskal)算法和P(Prim)算法.我的 K 算法用的是结构体来存图,P 算法用的是邻接矩阵来存图,K算法的复杂度是O( ...

  2. Python与正则表达式[0] -> re 模块的正则表达式匹配

    正则表达式 / Regular Expression 目录 正则表达式模式 re 模块简介 使用正则表达式进行匹配 正则表达式RE(Regular Expression, Regexp, Regex) ...

  3. 路由器漏洞利用工具RouterSploit

     路由器漏洞利用工具RouterSploit 网络中存在大量的嵌入式设备,如路由器.智能摄像头.这类设备安全防护程度较低.由于这些设备更新不方便,一旦发现漏洞,往往不能及时修复.所以,在网络渗透测试中 ...

  4. UVA 11396 Claw Decomposition 染色

    原题链接:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem ...

  5. APIO2018练习赛伪题解

    传送门:https://pcms.university.innopolis.ru/statements/org/apio/2018/practice/statements.pdf 主要就在于后面三道构 ...

  6. VisualStudio Shell简介

    VisualStudio Shell是微软效仿Eclipse推出的一个免费的VisualStudio内核,开发者可以通过在其上挂载插件(和传统的VS插件一样),从而快速开发自己的程序.它是Visual ...

  7. MS SQL数据批量备份还原(适用于MS SQL 2005+)

    原文:MS SQL数据批量备份还原(适用于MS SQL 2005+) 我们知道通过Sql代理,可以实现数据库的定时备份功能:当数据库里的数据库很多时,备份一个数据库需要建立对应的定时作业,相对来说比较 ...

  8. Proxy server got bad address from remote server

    在ArcMap中,在GIS Servrvers中,打开已经设置好的服务器时,出现下面的弹窗问题. Proxy server got bad address from remote server(ver ...

  9. 脑科学对基金经理的八个启示 z

    脑科学对基金经理的八个启示 第一,总想要更多.人类大脑是在物资奇缺过程中进化的,所以获得任何“资源”,如食物.性.金钱等,都可以让人感觉良好,大脑也会鼓励我们继续下去. 事实上,可卡因等药物就是“绑架 ...

  10. JS 随机数字抽签

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <t ...