1.单个控制器监听进入后台
在该控制器的-viewDidLoad方法中,添加代码监听notification,也可以使用监听APP返回方法
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(someMethod:)
name:UIApplicationDidBecomeActiveNotification object:nil];
监听方法的具体实现
-(void)someMethod:(NSNotification *)noti
{
//从该控制器进入后台时需要执行的代码
}
恢复到前台
- (void)applicationDidBecomeActive:(UIApplication *)application{
NSLog(@"---applicationDidBecomeActive----");
}
2.十六进制字符串转换成字节数组.
目标字符串 NSString *str = @"0200107580FD7590FD750FD7590FD75A0FD";
调用方法:NSData *temp = [string hexToBytes]; 转换后
byte数组 Byte byteArr[] = { 0x02, 0x00, 0x10, ... , 0xFD };
3、对于项目中的第三库一定要进行再次封装,包括网络、刷新、提示、模型转换等所有能封装的部分,一直用MJExtension来做字典转模型,突然想用YYModel了,项目中替换很麻烦,刷新也一样,一直用第三方框架,突然想自己写,改起来麻烦的不要不要的;
4、熟悉一下测试的几种方案,例如交叉测试(一个功能正在运行,另一个功能运行对它的影响,例如 扫码时打开灯光,突然退到后台,再回来查看,可能灯光已经熄灭,按钮还没改变状态)等,这样写程序时才知道往哪些方面考虑
5、多写点代码块,写起代码来会很快,我把自定义Cell都封装了代码块,用xib布局cell就没有纯手写这么快了,改起来还麻烦,这就是代码块的好处;
6、主控制器因为代码比较多,结构一定要清晰,才方便寻找,插入的类按 服务工具类+MVC 划分,属性按修饰符划分,下面代码按功能划分,如下
#import "ViewController.h"
// 工具和服务类
#import "Header.h"
#import "Tool1.h"
#import "Tool2.h"
// Model
#import "Model1.h"
#import "Model2.h"
#import "Model3.h"
// View
#import "View1.h"
#import "View2.h"
#import "View3.h"
// Controller
#import "ViewController1.h"
#import "ViewController2.h"
#import "ViewController3.h"
@interface ViewController ()
// 按修饰符划分
@property (nonatomic, assign) NSInteger num1;
@property (nonatomic, assign) NSInteger num2;
@property (nonatomic, strong, nonnull) NSMutableArray<NSString *> *object3;@property (nonatomic, strong, nonnull) NSMutableArray<NSNumber *> *object4;
@property (nonatomic, copy, nullable) NSString *object5;
@property (nonatomic, copy, nullable) NSString *object6;
@property (nonatomic, weak) UILabel *object1;
@property (nonatomic, weak) UILabel *object2;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
}
#pragma mark - 初始化View
- (void)initView{
}
#pragma mark - 系统方法,界面显示到销毁
#pragma mark - 代理
#pragma mark - 按钮点击,通知,定时器
#pragma mark - 辅助方法#pragma mark - 懒加载
@end
7、定时器不是马上开始的,多久触发一次事件,多久才开始,记得在退出页面的时候释放定时器,否则控制器不会释放;
8、如果错误提示中出现了duplicate这样的字眼,很可能就是引入了.m文件
9、UIView的tag不能为0;
10、字典转JSON字符串;
NSData *data = [NSJSONSerialization dataWithJSONObject:params options:NSJSONWritingPrettyPrinted error:nil];
NSString *paramStr = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
11、有时在动画过程中,需要避免用户重复操作,否则很容易崩溃。
建议动画过程中设置[[UIApplication sharedApplication] beginIgnoringInteractionEvents];
允许用户操作[[UIApplication sharedApplication] endIgnoringInteractionEvents];
12、如果延时执行事件会被多次触发,那是一件很危险的事情,需要取消前面的延时执行事件
[selfperformSelector:<#(nonnull SEL)#>
withObject:<#(nullable id)#>afterDelay:<#(NSTimeInterval)#>];// 延时执行
[NSObject cancelPreviousPerformRequestsWithTarget:<#(nonnull id)#>
selector:<#(nonnull SEL)#>object:<#(nullable id)#>]// 取消延时执行
13、测试某部分代码的运行时间
NSTimeInterval beginTime = CFAbsoluteTimeGetCurrent();
......// 执行代码
NSTimeInterval endTime = CFAbsoluteTimeGetCurrent();
time = endTime-beginTime;// 运行时间
14、对某个控件截图
UIGraphicsBeginImageContextWithOptions(view.bounds.size,NO,0);
[view.layer renderInContext:UIGraphicsGetCurrentContext()];
UIImage *viewImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
15、添加毛玻璃效果
UIBlurEffect *effect = [UIBlurEffect effectWithStyle:UIBlurEffectStyleLight];
UIVisualEffectView *effectView = [[UIVisualEffectView alloc]initWithEffect:effect];
effectView.contentView = 控件;
16、添加长按手势,在手势开始时才执行方法,避免方法被调用两次
- (void)longPress:(UILongPressGestureRecognizer *)longPressGesture{
if (UIGestureRecognizerStateBegan != longPressGesture.state) {
return;
}
... // 执行方法
}
17、输入框有值时才能点击return key
textField.enablesReturnKeyAutomatically = YES;
18、isKindOfClass判断对象是否是一个类的成员,或者是派生自该类的成员isMemberOfClass确定对象是否是当前类的成员;
19、tableView设置cell的分割线从屏幕左侧边缘开始
cell.separatorInset = UIEdgeInsetsMake(0, 0, 0, 0);
20、tableView当内容不够时,去掉底部的分割线
self.tableView.tableFooterView = [[UIView alloc]init];
21、UITableView设置为Plain的样式时,你又有多组时,组头就会默认有悬浮效果,停留在上边,如果不想组头悬浮在上边,可以将样式设为Grouped,把足部设置很小,解决这问题;
UITableView *tableView = [[UITableView alloc] initWithFrame:CGRectMake(x,y,w,h) style:UITableViewStyleGrouped];
tableView.sectionFooterHeight = 0.0001;
22、设置UITableViewCell之间的间距,在自定义cell中重写setFrame方法
-(void)setFrame:(CGRect)frame{
frame.origin.y += 5;
frame.size.height -= 5;
[super setFrame:frame];
}
23、修改UISearchController上searchBar的风格,遍历子控件(很有用),找到合适的, 设置想要的子控件的颜色和分格;
UIImageView *barImageView = self.searchController.searchBar.subviews[0].subviews[0];
barImageView.layer.borderColor = [UIColor lightGrayColor];
barImageView.layer.borderWidth = 1;
UIView *textView = self.searchController.searchBar.subviews[0].subviews[1];
textView.backgroundColor = [UIColor whiteColor];
24、监测WKWebView的加载进度
[wkWebView addObserver:self forKeyPath:@"estimatedProgress" options:NSKeyValueObservingOptionNew context:nil];
25、一个控件获取在某个控件上的坐标点的四种方式
// 获取View在window上的坐标点的四种写法
// 使用 convertRect:toRect 方法
CGRect rect = [view convertRect:view.bounds toRect:window];
CGRect rect = [view.superView convertRect:view.frame toRect:window];
// 使用 convertRect:from 方法
CGRect rect = [window convertRect:view.boundsfrom:view];
CGRect rect = [window convertRect:view.framefrom:view.superView];
26、iOS11获取最上面的window
// iOS 11 以前
UIView *windowView = [[UIApplication sharedApplication].windows lastObject];
// iOS 11 以后
UIView *windowView = [[UIApplication sharedApplication].windows firstObject];
27、某个控件有双击和单击时,设置双击失败时,才触发单击
[oneTap requireGestureRecognizerToFail:doubleTap];
28、设置图片捏合缩放,双击放大
// 首先把imageView添加到scrollview中
- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView{
return self.imageView;
// 在代理返回当前imageView现
}
- (void)scrollViewDidZoom:(UIScrollView *)scrollView{
// 保证绕着中心点缩放
CGSize boundsSize = self.scrollView.bounds.size;
CGRect frameToCenter = self.imageView.frame;
if (frameToCenter.size.width < boundsSize.width) {
frameToCenter.origin.x = floorf((boundsSize.width - frameToCenter.size.width) * 0.5f);
} else {
frameToCenter.origin.x = 0;
}
if (frameToCenter.size.height < boundsSize.height) {
frameToCenter.origin.y = floorf((boundsSize.height - frameToCenter.size.height) * 0.5f);
}else {
frameToCenter.origin.y = 0;
}
if (!CGRectEqualToRect(self.imageView.frame, frameToCenter)) {
self.imageView.frame = frameToCenter;
}
}
-(void)doubleTap:(UITapGestureRecognizer *)tapBgRecognizer{
CGFloat scale = 4; // 最大缩放比例
if (self.self.imageView.frame.frame.size.width < kScreenWidth * scale) {
CGPoint point = [tapBgRecognizer locationInView:self.imageView.frame];
CGFloat xSize = kScreenWidth / scale;
CGFloat ySize = kScreenHeight / scale;
CGRect zoomRect = CGRectMake(point.x - xSize * 0.5f, point.y - ySize * 0.5f, xSize, ySize);
self.scrollView.maximumZoomScale = scale;
// 以点击点为中心缩放到最大
[self.scrollView zoomToRect:zoomRect animated:YES];
} else {
[UIView animateWithDuration:0.25 animations:^{
self.scrollView.zoomScale = 1.0;
self.scrollView.contentSize = self.scrollView.bounds.size;
self.self.imageView.frame.frame = self.originFrame; }];
}
}
29、寻找当前控件的控制器
-(UIViewController*)ht_viewController{
for(UIView *next = self; next; next = next.superview){
UIResponder *nextResponder = [next nextResponder];
if([nextResponder isKindOfClass:[UIViewController class]]{
return (UIViewController *)nextResponder;
}
}
return nil;
}
30、UITextField的inputView属性是指第一响应的不是键盘,而是赋值给inputView的那个view, inputAccessoryView 是指往键盘上添加另一个view;
31、设置键盘在UIScrollView拖动动时消失
scrollview.keyboardDismissMode = UIScrollViewKeyboardDismissModeOnDrag
32、设置UITableViewCell之间的分割线颜色;
self.tableView.separatorColor = [UIColor redColor];
33、iOS9以后点击状态栏,UIScrollView可返回顶部;
34、显示状态栏的网络请求菊花;
// 显示菊花-----NO为关闭菊花
[UIApplication sharedApplication].networkActivityIndicatorVisible = YES;
35、让UILabel的字体适应指定的宽度,当宽度大时,字体不变, 宽度小时, 字体变小适应宽度
label.adjustsFontSizeToFitWidth = YES;
36、监听拖动进度条时的状态
监听拖动进度条时的状态
[slider addTarget:self action:@selector(sliderChange:event:) forControlEvents:UIControlEventValueChanged];
- (void)sliderChange:(UISlider *)slider event:(UIEvent *)event{
UITouch *touch = [[event allTouches] anyObject];
// 根据状态来做相应的事情,避免拖动时一直调用某些事件
switch (touch.phase) {
case UITouchPhaseBegan: // 开始
case UITouchPhaseMoved: // 拖动
case UITouchPhaseEnded: // 结束
default:
break;
}
}
37、让UITableView的某一行滚到底部;
[self.tableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];
38、以Modal的形式push出一个界面;
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationCurve:UIViewAnimationCurveEaseInOut];
[UIView setAnimationDuration:0.75];
[self.navigationController pushViewController:vc animated:NO];
[UIView setAnimationTransition:UIViewAnimationTransitionFlipFromRight forView:self.navigationController.view cache:NO];
[UIView commitAnimations];
39、阻止设备自动锁屏[UIApplication sharedApplication].idleTimerDisabled = YES;在后台不管用,退出当前页面时,记得设为NO
40、为了避免循环引用,在block中我们一般都用弱引用,但是block中的弱引用对象可能会提前释放,造成崩溃,我们需要在block中强引用一下这个弱对象;
__weak __typeof(self)weakSelf = self;
view.callback = ^(ViewStatus status) {
// 强引用这个对象,避免执行到一半 self释放,造成崩溃;
__strong __typeof(weakSelf)strongSelf = weakSelf;
[strongSelf doSomething];
}
41、如果一个参数中需要包含多个枚举值,用NS_OPTIONS,不要用NS_ENUM
// 位运算保证任意组合的枚举值进行或运算能得到唯一的值
typedefNS_OPTIONS(NSUInteger,TestName){
TestNameXiaoHua=1<<0,// 小花
TestNameXiaoBai=1<<1,// 小白
TestNameXiaoHei=1<<2// 小黑
};
[self eat:TestNameXiaoHua | TestNameXiaoBai];// 让小花和小白有饭吃;
-(void)eat:(TestName)name{
if( (name & TestNameXiaoHua) || (name & TestNameXiaoBai) ){
NSLog(@"有饭吃");
}
}
42、快速生成一个带值的可变字典
NSMutableDictionary *mutDic = @{@"key":@"value"}.mutableCopy;
43、忽略编译器警告
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wgnu" // 设置要忽略的类型 这里是GNU警告
// 代码
#pragma clang diagnostic pop
44、为了避免多线程访问数据库,造成数据混乱,让读写方法都在同一个队列中进行;
static const void * const IOKey = &IOKey;
// 开辟一个单例队列
dispatch_queue_t NTESDispatchIOQueue(){
static dispatch_queue_t queue;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// dispatch_queue_create("", DISPATCH_QUEUE_SERIAL) queue = dispatch_queue_create("io.queue", 0);
dispatch_queue_set_specific(queue, IOKey, (void *)IOKey, NULL); // 设置队列的的标记 });
return queue;
}
typedef void(^dispatch_block)(void);
void io_sync_safe(dispatch_block block){
// 如果是自己设置的队列,执行block
if (dispatch_get_specific(IOKey)) {
block();
} else// 如果不是自己设置的队列,先创建队列,再执行block {
dispatch_sync(NTESDispatchIOQueue(), ^() {
block();
});
}
45、监听UITextField值的改变,可以使用这个方法
[_textViewaddTarget:self.action:@selector(textFieldDidChangeValue:)forControlEvents:UIControlEventEditingChanged]
而不要使用这个代理,因为可能监听不到
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string { },
46、把C++算法库封装在OC文件里形成静态库时,记得把OC的.m文件改成.mm文件
47、如果视图里面存在唯一一个UIScrollView或其子类View时,会主动设置相应的内边距,避免被导航栏遮住,如果我们的导航栏不透明,原点会从我们的导航栏下方算起,导致上方留白,解决这问题:
if (@available(iOS 11.0, *)) {
self.tableView.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever; //UIScrollView也适用
}else {
self.automaticallyAdjustsScrollViewInsets = NO;
}
48、如果数组中的元素是唯一的,或者要查询某个数据是否在该数组中,不要使用NSArray, 而是使用集合NSSet,集合采用的是哈希表,查询速度更快;
49、动画里的万能神器CAShapeLayer,性能特别好,能实现很多神奇的效果,做什么动画前优先考虑它,尤其做股票财经、健康APP类的,经常需要画图,如果遇到性能问题,试试它;
50、swift中为了避免循环引用,我们使用weak或者unowned来解决,但它们是有区别的; unowned更像OC里的unsafe_unretained,当引用对象释放了以后,它仍然会保持对引用对象的一个无效引用,如果尝试调用方法和成员属性的话,程序就会崩溃; weak则会在引用的内容被释放后,它会自动地变成 nil; 因此我们在使用它时,如果你引用对象不会释放, 使用unowned,写起来方便点; 如果你引用对象会被释放,请使用weak,例如网络请求;
51、在swift中用强制解包!一定要非常小心,尽量少用,一不小心程序就崩溃了;
52、在swift中数组和字典都属于值类型,相当于int类型,跟oc不一样,当你把数组赋值给另一个数组,修改数组的值不会影响另一个数组;
53、__bridge 用于OC指针与c语言中的 void *互相转换,虽然 id 和void *能够相互转换。但转换为void *,其安全性与赋值给__unsafe_unretained修饰符相近,甚至会更低。如果管理时不注意赋值对象的所有者,就会因悬垂指针而导致程序崩溃。具体细节,参考 __bridge 的那些事儿
54、void* 类型指针:通用变体类型指针;可以不经转换,赋给其他指针,函数指针除外;malloc返回的就是void*类型。
NULL指针:是一个标准规定的宏定义;#define NULL ((void *) 0) 用来表示空指针常量;
零指针:指针值为0,零值指针,没有存储任何内存地址的指针;可以使任意一种指针类型,eg:void * ;int * ;double *;
空指针:指针赋值为0;0*7;3-3等之后,指针即变成空指针;即:空指针不指向任何实际的对象或者函数;NULL指针和零指针都是空指针。
野指针:指向垃圾内存的指针;(1)指针变量没有初始化(2)指针被delete或者free之后没有置为空(3)指针操作超越了变量的范围
悬垂指针:指向曾经存放对象的内存,但是该对象已经不存在了;delete操作完成后的指针就是悬垂指针,此时需要将指针置为0变为零值指针;
55、WKWebView 与 JS 交互,需要添加 [userController addScriptMessageHandler:self name:@"callFunction"]; callFunction 可自定义,与 JS 保持一致,才可交互。
于控制器中添加 evaluateJavaScript:方法
56.关于 -> 语法 与 .语法
self -> _topview 这个是把 self 当成了个结构体指针;
self.topview 是把self 当成了一个对象,或者说是结构体变量;
另外,self-> 不会触发 set方法,self. 会触发。
. 左侧可以是结构体变量,也可以是对象;-> 左侧肯定是当成 结构体指针。
像这种用法, k 就是个结构体变量,p 是取了k 的地址,所以p 是个 astrct 类型指针,他就可以用->
成员变量 + 读写方法 = 类属性
简单直接的理解方式:.语法是去访问类属性,->是访问成员变量。 . 语法用于寻址,在c 语言中,左侧必须是个类型变量。而 - > 用于间接寻址。比如下图
验证上面所说
第一行 &k 和 &(k.a)的地址完全一样,是因为在这个结构体中,于内存存储的时候,最开始的部分存放的就是a,然后紧接着就是b,当拿到这个结构题类型的变量K, 实际上就是拿到存储的首地址。
而第二行的 p ,其实是 指针p 自身的地址,而指针指向的是 k。
57.Implicit declaration of function XXX is invalid in C99
** 和CNCopySupportedInterfaces CNCopyCurrentNetworkInfo 等相关的错 需要 #import <SystemConfiguration/CaptiveNetwork.h>
58.关于 armv7、armv7s、arm64、i386、x86_64 的区别
arm64:iPhone6s | iphone6s plus|iPhone6| iPhone6 plus|iPhone5S | iPad Air| iPad mini2(iPad mini with Retina Display)
armv7s:iPhone5|iPhone5C|iPad4(iPad with Retina Display)
armv7:iPhone4|iPhone4S|iPad|iPad2|iPad3(The New iPad)|iPad mini|iPod Touch 3G|iPod Touch4
i386 是针对intel通用微处理器32位处理器
x86_64是针对x86架构的64位处理器
模拟器32位处理器测试需要i386架构;
模拟器64位处理器测试需要x86_64架构;
真机32位处理器需要armv7,或者armv7s架构;
真机64位处理器需要arm64架构;
58.界面出来之前 xib 中的 frame 都不要去使用。可以在viewDidAppear 方法里设置🤷♂️
59.使用CocoaPods创建自己的私有库,首次使用 pod lib lint 可能遇到的问题
这个问题是pod依赖的组件fourflusher与xcode版本不匹配造成的,可以使用如下命令:
1.sudo gem uninstall fourflusher
2.sudo gem install fourflusher
必要情况下还需要更新pod
sudo gem update cocoapods
[!] TestLib did not pass validation, due to 3 warnings (but you can use `--allow-warnings` to ignore them).
pod验证发现3个及以上的warning就会报这个错,如果只是验证一下工程,能确保对外发布之前能修复,可以使用 --allow-warnings
pod lib lint --allow-warnings
如果验证通过,会看到 TestLib passed validation ,到这一步既完成验证。
60.WKWebView 背景透明
[self.webView setOpaque:NO];
[self.webView setBackgroundColor:[UIColor clearColor]];
[self.webView.scrollView setBackgroundColor:[UIColor clearColor]];
61.ARC 情况下堆、栈注意点
如图第二个Block处不使用Copy发生崩溃,通过打印可以发现2个Block所处区域不同,前者 blk0 在堆区,而后面的blk1在栈,容易被系统回收,在ARC 的情况下一般就直接拷贝到堆上去。
62.App 或 Mac 程序内使用WebView播放网络视频,禁止自动全屏:
目前在做一个Mac程序的监控设备播放,由JavaScript控制播放,发现程序在加载页面后会自动弹出播放器并进行全屏播放,碰巧视频是采用如 .m3u8 格式,结果在退出全屏后,除非重新加载页面,否则视频无法播放;
WKWebViewConfiguration *configuration = [[WKWebViewConfiguration alloc] init];
configuration.allowsInlineMediaPlayback = true;
self.webView =[[WKWebView alloc] initWithFrame:CGRectMake(0,20,W,H-20)configuration: configuration];
如果使用的是UIWebView:
self.webView.allowsInlineMediaPlayback = YES;
如果是在Xib中创建,需要勾选Inline Playback选项:
63.手机连接打印机
let printVC = UIPrintInteractionController.init()
let printInfo = UIPrintInfo.printInfo()
printInfo.outputType = .general
printInfo.duplex = .longEdge
printVC.showsPaperSelectionForLoadedPapers = true
printVC.printInfo = printInfo
printVC.printFormatter = webView.viewPrintFormatter()
printVC.present(animated: true, completionHandler: nil)
64.关于可变数组添加对象
假设有临时变量可变数组A,可变数组B,可变数组B中有若干元素,当可变数组A执行 AddObject 添加数组B后,仅是添加可变数组B的指针,即数组B的首地址,当操作数组B后,如移除数组元素,数组A中的数据也会随之改变。
同理。数组A在添加字符串C后,改变字符串C的值,数组A中的元素也会改变。
65.dealloc 方法被调用后,不代表对象的内存立刻被回收。详情可以了解一下runloop内部autoRelase的释放机制.
66.ARC情况下,关于控制器dealloc未执行,导致没有移除通知的预防措施
假设 A、B、C三个类,都继承自 D 类,即D类为基类,层级关系为A push B,B push C,由此可以知道,程序会当C出栈时,会执行dealloc方法,然后D类执行dealloc,接着B执行dealloc,D又执行dealloc,最后是A执行dealloc,D再执行dealloc。故而我们可以根据这一特性来设计。
首先定义一个全局的NSCache 来存储 某个类class,其内部监听所有KVO的Key,即 class 作为NSCache 的key, 所监听 key的数组作为Value,也就是一个类有多个通知的情况,然后在基类的delloc中,检查是否存在这样的key-value,如存在,for循环Remove。
解释:假设因为C类中的block强弱引用问题,或NSTime未释放等原因导致deallo未执行,通知未移除。而当B类出栈时若正常执行dealloc,程序会自动调用基类的dealloc,其方法内检查到NSCache中有通知未移除,就移除通知。以达到防止内存泄漏的目的。
67. arrarWithArray 等效于 alloc init 的方式创建数组
68. 关于copy与mutableCopy
69. 不可变字符串相当于常量,初始化后,只有当程序结束运行才会被释放,并不会因为出了函数作用域或者控制器被销毁而释放
验证:开始执行for循环,循环创建1亿个String对象,CPU与内存开始上涨
循环结束,CPU使用率下降,内存占有率不变
70. WWDC 2020 ,苹果宣布当用户退款成功时,无论哪种内购类型,开发者都能收到退款通知!
当Apple受理了用户(玩家)的退款申请后,即允许退款后,苹果服务器会发送通知,(商家)通过处理退款信息以响应退款通知,达到因用户(退款)而采取的相应行动,如扣除相关订阅服务或游戏道具等。
详细说明:https://developer.apple.com/documentation/storekit/in-app_purchase/handling_refund_notifications
在苹果后台可以配置一个退款通知的回调地址(一个App配置一条链接):
配置的回调链接必须满足条件:
满足应用传输安全要求(使用https)
URL 最长 255 字符
注意:这里的 https 是指苹果的 App Transport Security (ATS),其中有协议的要求,比如使用 Transport Layer Security (TLS) protocol 1.2 版本,具体见苹果文档:Preventing Insecure Network Connections | Apple Developer Documentation。
苹果把回调的通知分为2种类型:
退款通知类型
取消通知类型
其中新增加的退款通知类型是针对:
消耗型
非消耗型
非续期订阅
取消通知类型是针对:
自动续期订阅
退款通知流程
退款通知的内容:
苹果返回的通知内容为 JSON 对象数据,所有的退款订单的通知是在 unified_receipt 里的 latest_receipt_info 数组中:
在 unified_receipt 里的 latest_receipt_info 是一个数组,其中包含的最近的100次应用内购买交易:
数据中每个退款订单的主要字段:
详细的返回字段见官方文档:
responseBody | Apple Developer Documentation
unified_receipt | Apple Developer Documentation
responseBody.Latest_receipt_info | Apple Developer Documentation
退款通知的响应
您的服务器应发送HTTP状态代码,以指示服务器到服务器的通知接收是否成功:
如果回调接收成功,则发送 HTTP 200。您的服务器不需要返回数据。
如果回调接收不成功,请发送 HTTP 50x 或 40x 让 App Store 重试该通知。App Store在一段时间内尝试重试该通知,但在连续失败尝试后最终停止(3次)。
注意事项:
当您使用包含退款交易的收据 transaction_data 向苹果服务器校验 verifyReceipt 时,JSON响应中不存在退款交易,自动续订订阅除外。
收到 REFUND 通知时,您有责任为每笔退款交易存储,监控并采取适当的措施。(因为苹果只通知一次,暂时无法在苹果后台查询退款的订单。也不能由开发者主动去苹果服务器查询。)
自动续订订阅通知
这个取消通知之前就一直有,所以这里不重复了,需要的自行搜索。
自动续订订阅的相关文档:
Handling Subscriptions Billing | Apple Developer Documentation
In-App Purchases and Using Server-to-Server Notifications - WWDC 2019 - Videos - Apple Developer
Subscription Offers Best Practices - WWDC 2019 - Videos - Apple Developer
最后有2点,1.退款通知目前并不能在沙盒中测试;2.苹果后台也无法看到退款的订单详情;
71. 命令行修改App版本号
cd 进入工程所在目录后
修改 Version 版本号 - 如改成 3.0.0
/usr/libexec/PlistBuddy -c "Set :CFBundleShortVersionString 3.0.0" Info.plist
修改 Build 版本号 - 如改成 3.0.202103051700
/usr/libexec/PlistBuddy -c "Set :CFBundleVersion 3.0.202103051700" Info.plist
72. 真机调试下查看沙盒文件
连接真机设备,打开Xcode ,选择Window -->Device and Simulator ,快捷键 Command + Shift + 2;
找到对应的APP,选择设置图标,点击Download Container 下载到桌面
完成后就会在桌面上看见如下文件,右击 ——> 选择“显示包内容”
如果要想替换真机中的沙盒文件也是如上操作,在已下载好的文件中修改完成后。打开Xcode,Command + Shift + 2。选择Replace Container,选择修改后的沙盒文件,点击Open。
73. 获取不到内购套餐产品信息的情况
1.检查App的Bundle ID是否与苹果商店中配置的信息统一;
2.检查产品ID是否填写错误;
3.检查是否同意App付费协议,及填写银行卡等税务信息;
当在苹果商店配置好内购套餐后,即便不提交审核(二进制文件缺失),或者审核失败,App在沙盒环境下,仍然能获取、购买App内购套餐;
74. 简单的线程死锁
75.并行与串行的区别
在子线程中启动定时器,需要使用Runloop
76.OC的动态消息机制
调用某个对象的私有方法
77.字符转换
将普通话转换为拉丁符号,且带音标,包括希伯来语、阿拉伯语、泰语等多种语言。对处理除英语以外的语言和非拉丁文字时非常有用
78.将时间戳转换为NSDate 或 NSString (如今日0点时间戳1639929600 -> 2021-12-20 00:00:00)
79.在字典A中SetValue一个从字典B转换成的字符串B‘,然后将字典A转成字符串A’导致Web端解析失败的问题
因为往字典中Set 字符串pkgStr,会自动为Value添加引号,导致最后将字典infoData转成字符串时,会出现 "webInfo":"{xxxx 的情况,导致Web解析失败。
解决方法,直接在 infoData 中Set另一个字典 pkgInfo,不需要转成字符串后再写入
这样就不会出现 Web 解析失败的问题;