百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

分享你最喜欢的技巧和提示(Xcode,Objective-C,Swift,C...等等)

yuyutoo 2025-02-03 15:54 4 浏览 0 评论

笔者分享总结如下(本篇会不定期进行更新) :

Objective-C

1.让Xcode的控制台支持LLDB类型的打印

这有什么用?

怎么说尼,笔者认为这个还是比较有用滴,为什么有用尼?

因为在Xcode断点调试的时候, 在控制台输入 po self.view.frame 或者 po id 类型的时候就死翘翘了。不信? 看如下图 :

进入正题

打开终端输入三条命令:

1. touch ~/.lldbinit

2. echo display @import UIKit >> ~/.lldbinit

3. echo target stop-hook add -o \"target stop-hook disable\" >> ~/.lldbinit

输完命令后没有任何提示? 好吧, 那恭喜你成功了~! 然后, 最关键的一步来了, 那就是…

重新运行项目(不用重启Xcode也可以),看如下图~~

就代表成功啦

那么现在我们继续在控制台输入po self.view.frame

成功了!如果po指令是一个id类型也可以正常打印。是不是感觉方便很多呀? 反正我是这么觉得。至于有没有用就看个人需要咯~~!

如何删除?

好吧, 那么问题来了, 我用命令创建, 如果不想玩了怎么办尼??

其实很简答, 看第一条命令touch ~/.lldbinit,就是在根目录下创建了一个隐藏文件.lldbinit,然后删除这个文件就搞定啦。打开终端然后,在终端输入 : rm ~/.lldbinit 命令即可.

2.用宏定义检测block是否可用!

#define?BLOCK_EXEC(block,?...)?if?(block)?{?block(__VA_ARGS__);?};????

//?宏定义之前的用法??
/*?
if?(completionBlock)???
{???
????completionBlock(arg1,?arg2);???
}???
??*/
????
//?宏定义之后的用法??
BLOCK_EXEC(completionBlock,?arg1,?arg2);

3.用@ 来包含C字符串 或者非OC对象

NSString?*propertyAttributesString?=
????@(property_getAttributes(class_getProperty([NSObject?class],?"description")));
//?T@"NSString",R,C

4.AmIBeingDebugged(from mattt)

Nolan O’Brien brings the AmIBeingDebugged function to our attention from from this Technical Q&A document:

#include?#include?#include?#include?#include?static?Bool?AmIBeingDebugged(void)?{
????int?mib[4];
????struct?kinfo_proc?info;
????size_t?size?=?sizeof(info);
????info.kp_proc.p_flag?=?0;
????
????mib[0]?=?CTL_KERN;
????mib[1]?=?KERN_PROC;
????mib[2]?=?KERN_PROC_PID;
????mib[3]?=?getpid;
????
????sysctl(mib,?sizeof(mib)?/?sizeof(*mib),?&info,?&size,?NULL,?0);
????return?(info.kp_proc.p_flag?&?P_TRACED)?!=?0;
}

5.给SDK头文件加权限

如果您是从DMG安装Xcode的,看看这个技术通过Joar Wingfors,以避免通过保留所有权,权限和硬链接意外修改SDK头:

$ sudo ditto /Volumes/Xcode/Xcode.app /Applications/Xcode.app

6.检查void *实例变量(from mattt)

对于逆向工程的目的,但是这是可以看的对象实例变量。它通常很容易用valueForKey这样获取。

还有一个情况下,它不能用valueForKey获取,虽然:当这个变量是void *类型。

@interface?MPMoviePlayerController?:?NSObject?{
????void?*_internal;????//?4?=?0x4
????BOOL?_readyForDisplay;??//?8?=?0x8
}

用底层方式来访问

id?internal?=?*((const?id*)(void*)((uintptr_t)moviePlayerController?+?sizeof(Class)));

不要使用这段代码,它的非常危险的。仅使用于逆向工程!

7.使用ARC和不使用ARC(from 夏夏)

//使用ARC和不使用ARC
#if?__has_feature(objc_arc)
//compiling?with?ARC
#else
//?compiling?without?ARC
#endif

8.读取本地图片(from 夏夏)

#define?LOADIMAGE(file,ext)?[UIImage?imageWithContentsOfFile:[NSBundle?mainBundle]pathForResource:file?ofType:ext]

//定义UIImage对象
#define?IMAGE(A)?[UIImage?imageWithContentsOfFile:[NSBundle?mainBundle]?pathForResource:A?ofType:nil]

9.一个通用回调的简单示例(from 灰灰)

.h文件

#import?@interface?UIViewController?(LHYBlock)

#pragma?mark?-?block

@property?(nonatomic,?copy)?void?(^viewControllerActionBlock)(UIViewController?*vc,?NSUInteger?type,?NSDictionary?*dict);

#pragma?mark?-?viewControllerAction

/**
?*??View?事件的block回调
?*
?*??@param?viewControllerActionBlock?block的参数有view本身,状态码,键值对。
?*/
-?(void)viewControllerAction:(void?(^)(UIViewController?*vc,?NSUInteger?type,?NSDictionary?*dict))viewControllerActionBlock;

@end

.m 文件

#import?"UIViewController+LHYBlock.h"
#import?@implementation?UIViewController?(LHYBlock)
#pragma?mark?-?runtime?associate

-?(void)setViewControllerActionBlock:(void?(^)(UIViewController?*vc,?NSUInteger?type,?NSDictionary?*dict))viewControllerActionBlock?{
????objc_setAssociatedObject(self,?@selector(viewControllerActionBlock),?viewControllerActionBlock,?OBJC_ASSOCIATION_COPY);
}

-?(void?(^)(UIViewController?*,?NSUInteger,?NSDictionary?*))viewControllerActionBlock?{
????return?objc_getAssociatedObject(self,?@selector(viewControllerActionBlock));
}

#pragma?mark?-?block

-?(void)viewControllerAction:(void?(^)(UIViewController?*vc,?NSUInteger?type,?NSDictionary?*dict))viewControllerActionBlock?{
????self.viewControllerActionBlock?=?nil;
????self.viewControllerActionBlock?=?[viewControllerActionBlock?copy];
}

#pragma?mark?-
@end

import这个类 , 就能用block, 参数都是通用的本身,状态码,字典.(灰神提供)

10.iOS图片内存优化(博文)内存优化经验(from 灰灰)

解决步骤:instrument调试后,发现没被释放的全是imageIO,差不多就知道了,把读图的方式,从[UIImage imageNamed:@”“],改成imageWithContentsOfFile,就可以了。

问题原因:imageNamed读取图片的方法,会缓存在内存中,所以较大的图片,还是用imageWithContentsOfFile。

Tip1:.xcassets里的图片无法用imageWithContentsOfFile读取;

Tip2:imageWithContentsOfFile读取图片需要加文件后缀名如png,jpg等;

11.自定义弱关联对象(weak associated objects)

不幸的是,关联对象不支持弱引用。幸运的是,很容易实现。

你只需要一个简单的类包装与弱引用一个对象.

@interface?WeakObjectContainter?:?NSObject
@property?(nonatomic,?readonly,?weak)?id?object;
@end

@implementation?WeakObjectContainter
-?(instancetype)initWithObject:(id)object?{
????self?=?[super?init];
????if?(!self)?{
????????return?nil;
????}
????
????_object?=?object;
????
????return?self;
}
@end

设置与获取

//?设置弱引用关联
objc_setAssociatedObject(self,?&MyKey,?[[WeakObjectContainter?alloc]?initWithObject:object],?OBJC_ASSOCIATION_RETAIN_NONATOMIC);

//获取弱引用关联
id?object?=?[objc_getAssociatedObject(self,?&MyKey)?object];

12.在控制台里打印controller的层级

在控制台里使用po [UIViewController _printHierarchy]命令即可打印出controller的层级,一目了然.大家都去玩玩吧~~1

13.在控制台里打印view的层级

在控制台里使用po [[[UIApplication sharedApplication] keyWindow] recursiveDescription]命令即可打印出view的层级,一目了然。

当然,可能对于某一些人来说打印window下的所有view层级,会觉得眼花缭乱。

但是,也可以打印指定某一个view的层级。

po?[view?recursiveDescription]

14.在debug模式下的控制台里使用po命令打印对象的属性和值

添加分类,加上代码即可。不用导入头文件,即可在控制台里使用po命令打印出model的属性和值

#import?"NSObject+ZXPDebugDescription.h"
#import?@implementation?NSObject?(ZXPDebugDescription)

+?(void)load?{
????method_exchangeImplementations(class_getInstanceMethod([self?class],?@selector(debugDescription)),?class_getInstanceMethod([self?class],?@selector(zxp_swizzleDebugDescription)));
}

-?(NSString?*)zxp_swizzleDebugDescription?{

????//一把情况下,如果不是entity或者model的子类就不需要打印属性,?比如系统的class.~.?这个按照个人需求而定
????if?(![self?isKindOfClass:[ZXPBaseEntity?class]]?||?![self?isKindOfClass:[ZXPBaseModel?class]])?{
????????????return?[self?zxp_swizzleDebugDescription];
????}
????//?以上代码是判断是否model或者entity
????
????NSMutableDictionary?*dictionary?=?[NSMutableDictionary?dictionary];
????
????uint?count;
????objc_property_t?*properties?=?class_copyPropertyList([self?class],?&count);
????
????for?(int?i?=?0;?i

15.给category添加属性的小技巧

这是运用到了对象关联, 如果不会的请看这篇文章: 时空传送门

.h 文件

#import?@interface?NSObject?(ZXPDebugDescription)

@property?(copy,nonatomic)?NSString?*zxp_testString;

@end

.m 文件

#import?"NSObject+ZXPDebugDescription.h"
#import?@implementation?NSObject?(ZXPDebugDescription)

-?(void)setZxp_testString:(NSString?*)zxp_testString?{
????objc_setAssociatedObject(self,?@selector(zxp_testString),?zxp_testString,?OBJC_ASSOCIATION_COPY_NONATOMIC);
}

-?(NSString?*)zxp_testString?{
????return?objc_getAssociatedObject(self,?@selector(zxp_testString));
}
@end

16.AutoLayout框架介绍(ZXPAutoLayout)

iOS原生的自动布局(NSLayoutConstraint)非常繁琐, 影响开发进度和可读性也不利于维护, 正所谓工欲善其事必先利其器 , 有一个良好的自动布局框架, 则会让我们事半功倍. 而ZXPAutoLayout则是解决这一问题和诞生 . 采用新颖的链式语法, 扩展性,可读性,维护成本也较低.并致力打造最好用,最简洁,最方便,最轻巧的自动布局。

以下一个简单示例。ZXPAutoLayout详细教程点此-github地址点此

//设置一个背景为半透明红色的view,上下左右四边都距离superview的距离为10
????UIView?*bgView?=?[UIView?new];
????[self.view?addSubview:bgView];
????bgView.backgroundColor?=?[[UIColor?redColor]?colorWithAlphaComponent:.5];
????[bgView?zxp_addConstraints:^(ZXPAutoLayoutMaker?*layout)?{
????????//上下左右四边都距离superview的距离为10
????????layout.edgeInsets(UIEdgeInsetsMake(10,?10,?10,?10));
????????
????????//也可以如下这行代码来设置,但要同时设置top,left,bottom,right.推荐以上写法,比较简洁.
????????//layout.topSpace(10).leftSpace(10).bottomSpace(10).rightSpace(10);
????}];

17.动态调用block(黑魔法)

//定义一个block
id?(^testBlock)(NSString?*string,NSArray?*array)?=?^id(NSString?*string,NSArray?*array)?{
????????????NSLog(@"param:--%@--%@",string,array);
????????????return?string;
????????};
????????
????????//?_Block_signature??是iOS的私有api
????????const?char?*?_Block_signature(void?*);
????????const?char?*?signature?=?_Block_signature((__bridge?void?*)(testBlock));
????????
????????NSMethodSignature?*methodSignature?=?[NSMethodSignature?signatureWithObjCTypes:signature];
????????NSInvocation?*invocation?=?[NSInvocation?invocationWithMethodSignature:methodSignature];
????????[invocation?setTarget:testBlock];
????????
????????NSString?*string?=?@"string";
????????[invocation?setArgument:&string?atIndex:1];
????????
????????NSArray?*array?=?@[@"xx",@"oo"];
????????[invocation?setArgument:&array?atIndex:2];
????????
????????[invocation?invoke];
????????
????????id?returnValue;
????????[invocation?getReturnValue:&returnValue];
????????NSLog(@"returnValue?:?%@",returnValue);

相关推荐

自卑的人容易患抑郁症吗?(自卑会导致抑郁吗)

Filephoto[Photo/IC]Lowself-esteemmakesusfeelbadaboutourselves.Butdidyouknowthatovert...

中考典型同(近)义词组(同义词考题)

中考典型同(近)义词组...

WPF 消息传递简明教程(wpf messagebox.show)

...

BroadcastReceiver的原理和使用(broadcast-suppression)

一、使用中注意的几点1.动态注册、静态注册的优先级在AndroidManifest.xml中静态注册的receiver比在代码中用registerReceiver动态注册的优先级要低。发送方在send...

Arduino通过串口透传ESP 13板与java程序交互

ESP13---是一个无线板子,配置通过热点通信Arduino通过串口透传ESP13板与java程序交互...

zookeeper的Leader选举源码解析(zookeeper角色选举角色包括)

作者:京东物流梁吉超zookeeper是一个分布式服务框架,主要解决分布式应用中常见的多种数据问题,例如集群管理,状态同步等。为解决这些问题zookeeper需要Leader选举进行保障数据的强一致...

接待外国人英文口语(接待外国友人的英语口语对话)

接待外国人英文口语询问访客身份:  MayIhaveyourname,please?  请问您贵姓?  Whatcompanyareyoufrom?  您是哪个公司的?  Could...

一文深入理解AP架构Nacos注册原理

Nacos简介Nacos是一款阿里巴巴开源用于管理分布式微服务的中间件,能够帮助开发人员快速实现动态服务发现、服务配置、服务元数据及流量管理等。这篇文章主要剖析一下Nacos作为注册中心时其服务注册与...

Android面试宝典之终极大招(android面试及答案)

以下内容来自兆隆IT云学院就业部,根据多年成功就业服务经验,以及职业素养课程部分内容,归纳总结:18.请描述一下Intent和IntentFilter。Android中通过Intent...

除了Crontab,Swoole Timer也可以实现定时任务的

一般的定时器是怎么实现的呢?我总结如下:1.使用Crontab工具,写一个shell脚本,在脚本中调用PHP文件,然后定期执行该脚本;2.ignore_user_abort()和set_time_li...

Spark源码阅读:DataFrame.collect 作业提交流程思维导图

本文分为两个部分:作业提交流程思维导图关键函数列表作业提交流程思维导图...

使用Xamarin和Visual Studio开发Android可穿戴设备应用

搭建开发环境我们需要做的第一件事情是安装必要的工具。因此,你需要首先安装VisualStudio。如果您使用的是VisualStudio2010,2012或2013,那么请确保它是一个专业版本或...

Android开发者必知的5个开源库(android 开发相关源码精编解析)

过去的时间里,Android开发逐步走向成熟,一个个与Android相关的开发工具也层出不穷。不过,在面对各种新鲜事物时,不要忘了那些我们每天使用的大量开源库。在这里,向大家介绍的就是,在这个任劳任怨...

Android事件总线还能怎么玩?(android实现事件处理的步骤)

顾名思义,AndroidEventBus是一个Android平台的事件总线框架,它简化了Activity、Fragment、Service等组件之间的交互,很大程度上降低了它们之间的耦合,使我们的代码...

Android 开发中文引导-应用小部件

应用小部件是可以嵌入其它应用(例如主屏幕)并收到定期更新的微型应用视图。这些视图在用户界面中被叫做小部件,并可以用应用小部件提供者发布。可以容纳其他应用部件的应用组件叫做应用部件的宿主(1)。下面的截...

取消回复欢迎 发表评论: