KVC/KVO与AFNetworking

本文介绍了Objective-C中的KVC和KVO机制,包括它们的基本概念、使用方法及注意事项。同时,文章还展示了如何利用AFNetworking框架进行网络请求、图片下载及文件上传等操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

KCV/KVOAFNetworking


一.KVC/KVO


1.KVC的使用


      KVC全称是Key Value Coding(键值对编程),其本质上是对NSObject类的扩展(NSObject类的Category),它提供一种机制来间接访问对象的属性。KVO全称Key Value Observing(键值对观察),是建立KVC之上的模型对象的观察者模式的实现。

  一个对象拥有某些属性。比如说,一个Person对象有一个name和一个address属性。以KVC说法,Person对象分别有一个value对应他的name和address键。 key只是一个字符串,它对应的值可以是任意类型的对象。从最基础的层次上看,KVC有两个方法:一个是设置key的值,另一个是获取key对应的值。

  例如我们从网络获得一个JSON格式的数据,它里面的键值对正好跟一个对象的属性一一对应,那么我们可以非常方便的用JSON格式的数据完成对象属性的赋值,而不需要去处理对象的每一个属性,代码如下所示:


#import <UIKit/UIKit.h>


@interface Person : NSObject


@property (nonatomic, copy) NSString *name;

@property (nonatomic, assign) int age;

@property (nonatomic, assign) double weight;


@end

#import "Person.h"


@implementation Person


- (void)setValue:(id)value forUndefinedKey:(NSString *)key {

    // 遇到没有对应的键时什么都不要做,这一点非常重要

}


- (NSString *)description {

    return [NSString stringWithFormat:@"%@ - %d - %f", self.name, self.age, self.weight];

}


@end

- (void)viewDidLoad {

    [super viewDidLoad];

    

    NSMutableDictionary *dict = [NSMutableDictionary dictionary];

    [dict setObject:@“elean” forKey:@"name"];

    [dict setObject:@“18” forKey:@"age"];

    [dict setObject:@“40” forKey:@"weight"];

    

    Person *person = [[CDPerson alloc] init];

    // 直接用字典中的键值对映射给person的对应属性赋值

    [person setValuesForKeysWithDictionary:dict];

    // 通过KVC的方式修改对象的属性

    [person setValue:@"张三丰" forKey:@"name"];

    [person setValue:@"120" forKey:@"age"];

    

    NSLog(@"%@", person);

}



2.KVO的使用


      Key-Value Observing(KVO)建立在KVC之上,它能够观察一个对象指定属性值的变化,在需要的时候产生对应的通知。


KVO示例代码:

#import <Foundation/Foundation.h>


@interface Account : NSObject 


@property (nonatomic, assign) double balance;


@end


@implementation Account


@end


@interface Person : NSObject 


@property (nonatomic, strong) NSString *name;


@property (nonatomic, assign) int age;


@property (nonatomic, strong) Account *account;


@end


@implementation Person


- (void)setAccount:(Account *)account {

    _account = account;

    [self.account addObserver:self forKeyPath:@"balance"

                      options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:nil];

}


- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {

    if ([keyPath isEqualToString:@"balance"]) {

        //Account *tempAccount = (id) object;

        //NSLog(@"账户余额发生改变: %f", tempAccount.balance);;

        double oldValue = [change[@"old"] doubleValue];

        double newValue = [change[@"new"] doubleValue];

        NSLog(@"账户余额改变: %f", newValue - oldValue);

    }

}


- (NSString *)description {

    return [NSString stringWithFormat:@"%@ : %d : %f", self.name, self.age, self.account.balance];

}


- (void)dealloc {

    [self.account removeObserver:self forKeyPath:@"balance"];

}


@end


int main(int argc, const char * argv[]) {

    @autoreleasepool {

        Person *person = [[Person alloc] init];

        person.name = @“elean”;

        person.age = 18;

        Account *account = [[Account alloc] init];

        account.balance = 1000;

        person.account = account;

        

        NSLog(@"%@", person);

        

        person.account.balance = 800;

    }

    return 0;

}



这里有几点需要注意:

  • valueForKey:方法用于以字符串调用对象的get属性方法,或者读取成员变量的值;与之相对的是setValue:forKey:,它用于以字符串调用对象的set属性方法,或者修改成员变量的值。


  • 对于基本数据类型,KVC方法会对基本数据类型进行封装,基本数据类型封装为NSNumber,其他结构体类型封装为NSValue


  • 在使用KVC时,如果找不到字符串对应的属性和成员变量时会调用valueForUndefinedKey:或者setValue:forUndefinedKey:这两个方法,默认情况下会抛出异常


  • 在默认情况下KVC方法能够直接访问类的私有成员变量,如果我们不想这样,可以重写accessInstanceVariablesDirectly方法,并令其返回NO(默认是返回YES)


  • KVC方法定义在NSKeyValueCoding类别中,该类别附加于NSObject类上,所以所有对象都具有这些方法


  • 在一些特殊的类的对象上调用KVC方法会有特别的效果。对于数组NSArray、集合NSSet,调用valueForKey:会对每个数组和集合成员调用valueForKey:,并返回新的数组或者集合


  • 在KVC中还有一种常用技术,称为键值链(Key Path)。(拓展内容  详情查看代码)键值链是用点将若干键相连的字符串,例如“manufacturer.product.name”。通过在对象上调用valueForKeyPath:或者setValue:forKeyPath:,我们就可以在一条语句中级联调用指定的属性。



二.JSONModel

      JSONModel可以让你快速创建智能的JSON数据模型,可在iOS和MacOS X应用中使用。使用JSONModel时,不需要额外去检查所要的服务器属性是否有返回。JSONModel的initWithDictionary方法会自动去进行检查并处理。


json.png






三.SDWebImage图片的异步下载


 [SDWebImage下载地址链接](https://github.com/rs/SDWebImage)


 [SDWebImage的文档连接](http://cocoadocs.org/docsets/SDWebImage/3.7.2/)


旧版本的SDWebImage使用MRC编写  需要在工程-》Build Phases-》Compile Sources-》SDWebImage相关文件后 输入 -fno-objc-arc






四.AFNetworking

        AFNetworking是一个能够快速使用的iOS和MacOS X下的网络框架,它是构建在Foundation URL Loading System之上的,封装了网络的抽象层,可以方便的使用,是一个模块化架构并拥有丰富API的框架。


HTTP请求和操作

// GET请求

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];

[manager GET:@"http://example.com/resources.json" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {

    NSLog(@"JSON: %@", responseObject);

} failure:^(AFHTTPRequestOperation *operation, NSError *error) {

    NSLog(@"Error: %@", error);

}];




// 带有表单参数的POST请求

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];

NSDictionary *parameters = @{@"foo": @"bar"};

[manager POST:@"http://example.com/resources.json" parameters:parameters success:^(AFHTTPRequestOperation *operation, id responseObject) {

    NSLog(@"JSON: %@", responseObject);

} failure:^(AFHTTPRequestOperation *operation, NSError *error) {

    NSLog(@"Error: %@", error);

}];





// 上传表单和文件

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];

NSDictionary *parameters = @{@"foo": @"bar"};

NSURL *filePath = [NSURL fileURLWithPath:@"file://path/to/image.png"];

[manager POST:@"http://example.com/resources.json" parameters:parameters constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {

    [formData appendPartWithFileURL:filePath name:@"image" error:nil];

} success:^(AFHTTPRequestOperation *operation, id responseObject) {

    NSLog(@"Success: %@", responseObject);

} failure:^(AFHTTPRequestOperation *operation, NSError *error) {

    NSLog(@"Error: %@", error);

}]; 



Session管理


// 创建一个下载任务

NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];

AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration];


NSURL *URL = [NSURL URLWithString:@"http://example.com/download.zip"];

NSURLRequest *request = [NSURLRequest requestWithURL:URL];


NSURLSessionDownloadTask *downloadTask = [manager downloadTaskWithRequest:request progress:nil destination:^NSURL *(NSURL *targetPath, NSURLResponse *response) {

    NSURL *documentsDirectoryURL = [[NSFileManager defaultManager] URLForDirectory:NSDocumentDirectory inDomain:NSUserDomainMask appropriateForURL:nil create:NO error:nil];

    return [documentsDirectoryURL URLByAppendingPathComponent:[response suggestedFilename]];

} completionHandler:^(NSURLResponse *response, NSURL *filePath, NSError *error) {

    NSLog(@"File downloaded to: %@", filePath);

}];

[downloadTask resume];



// 创建一个上传任务

NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];

AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration];


NSURL *URL = [NSURL URLWithString:@"http://example.com/upload"];

NSURLRequest *request = [NSURLRequest requestWithURL:URL];


NSURL *filePath = [NSURL fileURLWithPath:@"file://path/to/image.png"];

NSURLSessionUploadTask *uploadTask = [manager uploadTaskWithRequest:request fromFile:filePath progress:nil completionHandler:^(NSURLResponse *response, id responseObject, NSError *error) {

    if (error) {

        NSLog(@"Error: %@", error);

    } else {

        NSLog(@"Success: %@ %@", response, responseObject);

    }

}];

[uploadTask resume];





// 创建一个带进度的上传任务

NSMutableURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:@"http://example.com/upload" parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {

        [formData appendPartWithFileURL:[NSURL fileURLWithPath:@"file://path/to/image.jpg"] name:@"file" fileName:@"filename.jpg" mimeType:@"image/jpeg" error:nil];

    } error:nil];


AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];

NSProgress *progress = nil;


NSURLSessionUploadTask *uploadTask = [manager uploadTaskWithStreamedRequest:request progress:&progress completionHandler:^(NSURLResponse *response, id responseObject, NSError *error) {

    if (error) {

        NSLog(@"Error: %@", error);

    } else {

        NSLog(@"%@ %@", response, responseObject);

    }

}];


[uploadTask resume];



// 创建请求对象

NSString *URLString = @"http://example.com";

NSDictionary *parameters = @{@"foo": @"bar", @"baz": @[@1, @2, @3]};

[[AFHTTPRequestSerializer serializer] requestWithMethod:@"GET" URLString:URLString parameters:parameters error:nil];

// 网络可达性检测

[[AFNetworkReachabilityManager sharedManager] setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {

    NSLog(@"Reachability: %@", AFStringFromNetworkReachabilityStatus(status));

}];




[[AFNetworkReachabilityManager sharedManager] startMonitoring];

// 基于HTTP请求操作管理器的网络可达性检测

NSURL *baseURL = [NSURL URLWithString:@"http://example.com/"];

AFHTTPRequestOperationManager *manager = [[AFHTTPRequestOperationManager alloc] initWithBaseURL:baseURL];


NSOperationQueue *operationQueue = manager.operationQueue;

[manager.reachabilityManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {

    switch (status) {

        case AFNetworkReachabilityStatusReachableViaWWAN:

        case AFNetworkReachabilityStatusReachableViaWiFi:

            [operationQueue setSuspended:NO];

            break;

        case AFNetworkReachabilityStatusNotReachable:

        default:

            [operationQueue setSuspended:YES];

            break;

    }

}];


[manager.reachabilityManager startMonitoring];




// 批量任务处理

NSMutableArray *mutableOperations = [NSMutableArray array];

for (NSURL *fileURL in filesToUpload) {

    NSURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:@"http://example.com/upload" parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {

        [formData appendPartWithFileURL:fileURL name:@"images[]" error:nil];

    }];


    AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc] initWithRequest:request];


    [mutableOperations addObject:operation];

}


NSArray *operations = [AFURLConnectionOperation batchOfRequestOperations:@[...] progressBlock:^(NSUInteger numberOfFinishedOperations, NSUInteger totalNumberOfOperations) {

    NSLog(@"%lu of %lu complete", numberOfFinishedOperations, totalNumberOfOperations);

} completionBlock:^(NSArray *operations) {

    NSLog(@"All operations in batch complete");

}];

[[NSOperationQueue mainQueue] addOperations:operations waitUntilFinished:NO];



<key>NSAppTransportSecurity</key>

<dict>

<key>NSAllowsArbitraryLoads</key>

<true/>

</dict>


标题基于SpringBoot+Vue的学生交流互助平台研究AI更换标题第1章引言介绍学生交流互助平台的研究背景、意义、现状、方法创新点。1.1研究背景意义分析学生交流互助平台在当前教育环境下的需求及其重要性。1.2国内外研究现状综述国内外在学生交流互助平台方面的研究进展实践应用。1.3研究方法创新点概述本研究采用的方法论、技术路线及预期的创新成果。第2章相关理论阐述SpringBootVue框架的理论基础及在学生交流互助平台中的应用。2.1SpringBoot框架概述介绍SpringBoot框架的核心思想、特点及优势。2.2Vue框架概述阐述Vue框架的基本原理、组件化开发思想及前端的交互机制。2.3SpringBootVue的整合应用探讨SpringBootVue在学生交流互助平台中的整合方式及优势。第3章平台需求分析深入分析学生交流互助平台的功能需求、非功能需求及用户体验要求。3.1功能需求分析详细阐述平台的各项功能需求,如用户管理、信息交流、互助学习等。3.2非功能需求分析对平台的性能、安全性、可扩展性等非功能需求进行分析。3.3用户体验要求从用户角度出发,提出平台在易用性、美观性等方面的要求。第4章平台设计实现具体描述学生交流互助平台的架构设计、功能实现及前后端交互细节。4.1平台架构设计给出平台的整体架构设计,包括前后端分离、微服务架构等思想的应用。4.2功能模块实现详细阐述各个功能模块的实现过程,如用户登录注册、信息发布查看、在线交流等。4.3前后端交互细节介绍前后端数据交互的方式、接口设计及数据传输过程中的安全问题。第5章平台测试优化对平台进行全面的测试,发现并解决潜在问题,同时进行优化以提高性能。5.1测试环境方案介绍测试环境的搭建及所采用的测试方案,包括单元测试、集成测试等。5.2测试结果分析对测试结果进行详细分析,找出问题的根源并
内容概要:本文详细介绍了一个基于灰狼优化算法(GWO)优化的卷积双向长短期记忆神经网络(CNN-BiLSTM)融合注意力机制的多变量多步时间序列预测项目。该项目旨在解决传统时序预测方法难以捕捉非线性、复杂时序依赖关系的问题,通过融合CNN的空间特征提取、BiLSTM的时序建模能力及注意力机制的动态权重调节能力,实现对多变量多步时间序列的精准预测。项目不仅涵盖了数据预处理、模型构建训练、性能评估,还包括了GUI界面的设计实现。此外,文章还讨论了模型的部署、应用领域及其未来改进方向。 适合人群:具备一定编程基础,特别是对深度学习、时间序列预测及优化算法有一定了解的研发人员和数据科学家。 使用场景及目标:①用于智能电网负荷预测、金融市场多资产价格预测、环境气象多参数预报、智能制造设备状态监测预测维护、交通流量预测智慧交通管理、医疗健康多指标预测等领域;②提升多变量多步时间序列预测精度,优化资源调度和风险管控;③实现自动化超参数优化,降低人工调参成本,提高模型训练效率;④增强模型对复杂时序数据特征的学习能力,促进智能决策支持应用。 阅读建议:此资源不仅提供了详细的代码实现和模型架构解析,还深入探讨了模型优化和实际应用中的挑战解决方案。因此,在学习过程中,建议结合理论实践,逐步理解各个模块的功能和实现细节,并尝试在自己的项目中应用这些技术和方法。同时,注意数据预处理的重要性,合理设置模型参数网络结构,控制多步预测误差传播,防范过拟合,规划计算资源训练时间,关注模型的可解释性和透明度,以及持续更新迭代模型,以适应数据分布的变化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值