XAURLComponents-在低版本可以直接使用NSURLComponents的所有高版本属性和方法

深入理解Objective-C的Runtime机制

黑魔法——“低版本中使用高版本中出现的类”之技术实现原理详解

下载XAURLComponetns代码

NSURLComponents的一些属性在高版本才可以使用,例如:

@property (nullable, readonly, copy) NSString *string NS_AVAILABLE(10_10, 8_0);

@property (readonly) NSRange rangeOfScheme NS_AVAILABLE(10_11, 9_0);

@property (nullable, copy) NSArray<NSURLQueryItem *> *queryItems NS_AVAILABLE(10_10, 8_0);

等等。这些属性在较低版本中是无法使用的,此库可以在低版本中也能使用这些属性。

使用方法:直接拷贝导入

直接将XAURLComponents文件下的NSURLComponents+XAMatch.h/.m,XAURLQueryItem.h/.m拷贝到工程中即可,无需改动代码。

Method Resolution

首先Objective-C在运行时调用+ resolveInstanceMethod:或+ resolveClassMethod:方法,让你添加方法的实现。如果你添加方法并返回YES,那系统在运行时就会重新启动一次消息发送的过程。

当低版本调用高版本的方法时,因为找不到方法实现,会进入resolveInstanceMethod:方法,所以我们就可以在resolveInstanceMethod:方法中为低版本动态添加高版本的方法实现。

NSURLQueryItem是iOS8才有的类,如果想在iOS7中使用,我们可以用黑魔法——“低版本中使用高版本中出现的类”的技术来实现。

NSURLComponents+XAMatch.h

#import <Foundation/Foundation.h>

@interface NSURLComponents (XAMatch)

@end

NSURLComponents+XAMatch.m

#import "NSURLComponents+XAMatch.h"
#import <objc/runtime.h>

@implementation NSURLComponents (XAMatch)

#pragma mark - string
- (NSString *)xa_string {
    return self.URL.absoluteString;
}

#pragma mark - queryItems
- (NSArray *)xa_queryItems {
    NSMutableArray *queryItems = [NSMutableArray array];
    NSDictionary *params = [self paramsWithUrlString:self.string];
    if (params && [params isKindOfClass:[NSDictionary class]]) {
        for (NSString *key in params.allKeys) {
            NSURLQueryItem *item = [NSURLQueryItem queryItemWithName:key value:params[key]];
            [queryItems addObject:item];
        }
    }
    return queryItems;
}

- (NSDictionary *)paramsWithUrlString:(NSString *)urlStr {
    if (urlStr && urlStr.length && [urlStr rangeOfString:@"?"].length == 1) {
        NSArray *array = [urlStr componentsSeparatedByString:@"?"];
        if (array && array.count == 2) {
            NSString *paramsStr = array.lastObject;
            if (paramsStr.length) {
                NSMutableDictionary *paramsDict = [NSMutableDictionary dictionary];
                NSArray *paramArray = [paramsStr componentsSeparatedByString:@"&"];
                for (NSString *param in paramArray) {
                    if (param && param.length) {
                        NSArray *parArr = [param componentsSeparatedByString:@"="];
                        if (parArr.count == 2) {
                            paramsDict[parArr[0]] = parArr[1];
                        }
                    }
                }
                return paramsDict;
            }
        }
    }
    return nil;
}

#pragma mark - rangeOf
- (NSRange)xa_rangeOfScheme {
    return [self xa_rangeOfString:self.scheme];
}

- (NSRange)xa_rangeOfUser {
    return [self xa_rangeOfString:self.percentEncodedUser];
}

- (NSRange)xa_rangeOfPassword {
    return [self xa_rangeOfString:self.percentEncodedPassword];
}

- (NSRange)xa_rangeOfHost {
    return [self xa_rangeOfString:self.percentEncodedHost];
}

- (NSRange)xa_rangeOfPort {
    return [self xa_rangeOfString:self.port.stringValue];
}

- (NSRange)xa_rangeOfPath {
    return [self xa_rangeOfString:self.percentEncodedPath];
}

- (NSRange)xa_rangeOfQuery {
    return [self xa_rangeOfString:self.percentEncodedQuery];
}

- (NSRange)xa_rangeOfFragment {
    return [self xa_rangeOfString:self.percentEncodedFragment];
}

- (NSRange)xa_rangeOfString:(NSString *)string {
    if (string && [string isKindOfClass:[NSString class]] && string.length) {
        return [self.string rangeOfString:string];
    } else {
        return NSMakeRange(NSNotFound, 0);
    }
}

#pragma mark - Method Resolution
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    Class class = [self class];
    SEL swizzledSelector = NSSelectorFromString([NSString stringWithFormat:@"xa_%@", NSStringFromSelector(sel)]);
    Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
    
    class_addMethod(class, sel, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod));
    
    return YES;
}

@end

XAURLQueryItem.h

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface XAURLQueryItem : NSObject <NSSecureCoding, NSCopying> {
@private
    NSString *_name;
    NSString *_value;
}
- (instancetype)initWithName:(NSString *)name value:(nullable NSString *)value;
+ (instancetype)queryItemWithName:(NSString *)name value:(nullable NSString *)value;
@property (readonly) NSString *name;
@property (nullable, readonly) NSString *value;

@end

NS_ASSUME_NONNULL_END

XAURLQueryItem.m

#import "XAURLQueryItem.h"
#import <objc/runtime.h>

@implementation XAURLQueryItem

- (instancetype)initWithName:(NSString *)name value:(NSString *)value {
    if ((self = [super init])) {
        _name = name;
        _value = value;
    }
    return self;
}

+ (instancetype)queryItemWithName:(NSString *)name value:(NSString *)value {
    return [[XAURLQueryItem alloc] initWithName:name value:value];
}

#pragma mark - NSSecureCoding
+ (BOOL)supportsSecureCoding {
    return YES;
}

- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    if ((self = [super init])) {
        _name = [aDecoder decodeObjectOfClass:[NSString class] forKey:@"name"];
        _value = [aDecoder decodeObjectOfClass:[NSString class] forKey:@"value"];
    }
    return self;
}

- (void)encodeWithCoder:(NSCoder *)aCoder {
    [aCoder encodeObject:_name forKey:@"name"];
    [aCoder encodeObject:_value forKey:@"value"];
}

#pragma mark - NSCopying
- (id)copyWithZone:(NSZone *)zone {
    return [XAURLQueryItem allocWithZone:zone];
}

#pragma mark - Runtime Injection

__asm(
      ".section        __DATA,__objc_classrefs,regular,no_dead_strip\n"
#if TARGET_RT_64_BIT
      ".align          3\n"
      "L_OBJC_CLASS_NSURLQueryItem:\n"
      ".quad           _OBJC_CLASS_$_NSURLQueryItem\n"
#else
      ".align          2\n"
      "_OBJC_CLASS_NSURLQueryItem:\n"
      ".long           _OBJC_CLASS_$_NSURLQueryItem\n"
#endif
      ".weak_reference _OBJC_CLASS_$_NSURLQueryItem\n"
      );

__attribute__((constructor)) static void XAURLQueryItemPatchEntry(void) {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        @autoreleasepool {
            // >= iOS8
            if (objc_getClass("NSURLQueryItem")) {
                return;
            }
            
            Class *urlQueryItem = NULL;
            
#if TARGET_CPU_ARM
            __asm("movw %0, :lower16:(_OBJC_CLASS_NSURLQueryItem-(LPC0+4))\n"
                  "movt %0, :upper16:(_OBJC_CLASS_NSURLQueryItem-(LPC0+4))\n"
                  "LPC0: add %0, pc" : "=r"(urlQueryItem));
#elif TARGET_CPU_ARM64
            __asm("adrp %0, L_OBJC_CLASS_NSURLQueryItem@PAGE\n"
                  "add  %0, %0, L_OBJC_CLASS_NSURLQueryItem@PAGEOFF" : "=r"(urlQueryItem));
#elif TARGET_CPU_X86_64
            __asm("leaq L_OBJC_CLASS_NSURLQueryItem(%%rip), %0" : "=r"(urlQueryItem));
#elif TARGET_CPU_X86
            void *pc = NULL;
            __asm("calll L0\n"
                  "L0: popl %0\n"
                  "leal _OBJC_CLASS_NSURLQueryItem-L0(%0), %1" : "=r"(pc), "=r"(urlQueryItem));
#else 
#error Unsupported CPU
#endif
            
            if (urlQueryItem && !*urlQueryItem) {
                Class class = objc_allocateClassPair([XAURLQueryItem class], "NSURLQueryItem", 0);
                *urlQueryItem = class;
            }
        }
    });
}

@end
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 因为要结局swift3.0中引用snapKit的问题,看到一篇介绍Xcode8,swift3变化的文章,觉得很详细...
    uniapp阅读 4,385评论 0 12
  • (一) 森格是一位普普通通的贫民——既不是达官贵人,也不是富家公子,就是一名贫民。 森格生活在大漠,他...
    一车阅读 384评论 1 1
  • 我是一个拖延症患者重度的拖延症患者,无论什么事情我都会想办法把他拖到最后一个去。好像早一刻完成,我都会心中有愧。 ...
    天之巅海无涯阅读 37评论 0 0
  • 很早以前的一个梦,已经讲给解梦师听过了。可是还是很爱这个梦,于是再次记录在这里。时间间隔有些细节不太记得,但是整体...
    红线芊芊阅读 108评论 0 0