亲爱的读者们,你们好 !年底将近,分享从过去一年你最喜欢的技巧和建议作为礼物送给新手们。提交你的最喜欢的迅速或objc琐事,实用的提示,意外的发现,实用的解决方法,没用的迷恋,或不论什么其它你认为今年非常酷。

就在以下写下你的评论!

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

objective-c


用宏定义检測block是否可用~!



#define BLOCK_EXEC(block, ...) if (block) { block(__VA_ARGS__); }; // 宏定义之前的使用方法
/*
if (completionBlock)
{
completionBlock(arg1, arg2);
}
*/ // 宏定义之后的使用方法
BLOCK_EXEC(completionBlock, arg1, arg2);

在控制台里支持LLDB类型的打印 :po framepo id类型 打印


open terminal (打开终端输入例如以下三条命令,然后重新启动Xcode里app就可以):

  1. touch ~/.lldbinit
  2. echo display @import UIKit >> ~/.lldbinit
  3. echo target stop-hook add -o \”target stop-hook disable\” >> ~/.lldbinit

博文地址,具体教程点此

@() 来包括C字符串 或者OC对象



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

AmIBeingDebugged(from mattt)


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

#include <assert.h>
#include <stdbool.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/sysctl.h> 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;
}

给SDK头文件加权限


假设您是从DMG安装Xcode的,看看这个技术通过Joar Wingfors,以避免通过保留全部权,权限和硬链接意外改动SDK头:

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

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


对于逆向project的目的,可是这是能够看的对象实例变量。它通常非常easy用valueForKey这样获取。

另一个情况下,它不能用valueForKey获取。尽管:当这个变量是void *类型。

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

用底层方式来訪问

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

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

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


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

读取本地图片(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]

一个通用回调的简单演示样例(from 灰灰)


.h文件

#import <UIKit/UIKit.h>

@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 <objc/runtime.h>
@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, 參数都是通用的本身,状态码。字典.(灰神提供)

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


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

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

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

TIPs2:imageWithContentsOfFile读取图片须要加文件后缀名如png。jpg等;

灰神内存优化链接地址点此

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


不幸的是,关联对象不支持弱引用.

幸运的是, 非常easy实现

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

@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];

在控制台里打印controller的层级


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

在控制台里打印view的层级


在控制台里使用po [[[UIApplication sharedApplication] keyWindow] recursiveDescription]命令就可以打印出view的层级,一目了然.大家都去玩玩吧~~1

当然,可能对于某一些人来说打印window下的全部view层级,会认为眼花缭乱.

可是,也能够打印指定某一个view的层级.

po [view recursiveDescription]

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


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

#import "NSObject+ZXPDebugDescription.h"
#import <objc/runtime.h> @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<count; i++) {
objc_property_t property = properties[i];
NSString *name = @(property_getName(property));
id value = [self valueForKey:name]? :@"nil";
[dictionary setObject:value forKey:name];
} free(properties); return [NSString stringWithFormat:@"<%@: %p> -- %@",[self class],self,dictionary];
} @end

给category加入属性的小技巧


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

.h 文件

#import <Foundation/Foundation.h>

@interface NSObject (ZXPDebugDescription)

@property (copy,nonatomic) NSString *zxp_testString;

@end

.m 文件

#import "NSObject+ZXPDebugDescription.h"
#import <objc/runtime.h> @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

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);
}];

动态调用block(黑魔法)


//定义一个block
id (^testBlock)(NSString *string,NSArray *array) = ^id(NSString *string,NSArray *array) {
NSLog(@"param:--%@--%@",string,array);
return string;
}; // _Block_signature 是iOS的私有api
const void *_Block_signature(void *);
const void *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);

利用断点查找button的action


在控制台里输入br s -r . -s yourProjectName命令, 然后在app里点击一下button, 在让断点往下运行就可以.

ps:yourProjectName是你的project名的名字哦. 经笔者在Xcode7下使用这条命令的时候, 响应非常慢. Xcode6和Xcode5不会有这个问题, 可能是Xcode7的一个小小问题, 也不排除是我环境配置的太复杂而造成的影响~!

这条命令的具体赘述地址附上:点我就对了

自己定义并强化NSLog, 能查找LOG所打印的函数和类


//打印log
#ifdef DEBUG
#define ZXPLog(format, ...) NSLog((@"[函数名:%s]" "[行号:%d] " format), __FUNCTION__, __LINE__, ##__VA_ARGS__);
#else
#define ZXPLog(format, ...);
#endif

『干货』分享你最喜欢的技巧和提示(Xcode,objective-c,swift,c...等等)的更多相关文章

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

    http://www.cocoachina.com/ios/20151231/14846.html 笔者分享总结如下(本篇会不定期进行更新) : Objective-C 1.让Xcode的控制台支持L ...

  2. 分享iOS最喜欢的技巧和提示

    转自:http://blog.csdn.net/biggercoffee/article/details/50394027 Objective-C 1.让Xcode的控制台支持LLDB类型的打印 这有 ...

  3. 关于『Markdown』:第一弹

    关于『Markdown』:第一弹 建议缩放90%食用 声明: 在我之前已有数位大佬发布 "Markdown" 的语法知识点, 在此, 仅整理归类以及补缺, 方便阅读. 感谢 C20 ...

  4. 『摄影欣赏』16幅 Romantic 风格照片欣赏【组图】

    今天,我们将继续分享人类情感的系列文章.爱是人类最重要的感觉,也可能是各种形式的艺术(电影,音乐,书,画等)最常表达的主题 .这里有40个最美丽的爱的照片,将激励和给你一个全新的视觉角度为这种情绪.我 ...

  5. 在Ubuntu 11.10工具栏上用数字显示网速、CPU负荷和内存占用量『译』

    基本上照抄了<How To Display Network Upload / Download Speed On The Panel In Ubuntu 11.04>,只不过我的实践环境是 ...

  6. [日推荐] 『Streeter』极乐商店邀你一起来尬舞啊!-store.dreawer.com

    你听的舞蹈有哪些?Jazz?Poppin?Hiphop?现代舞?民族舞?... 今天推荐一款小程序『Streeter』,几乎包含所有舞蹈视频,偷偷学好了,邀请你来尬舞啊 Streeter 这个Logo ...

  7. 『实践』VirtualBox 5.1.18+Centos 6.8+hadoop 2.7.3搭建hadoop完全分布式集群及基于HDFS的网盘实现

    『实践』VirtualBox 5.1.18+Centos 6.8+hadoop 2.7.3搭建hadoop完全分布式集群及基于HDFS的网盘实现 1.基本设定和软件版本 主机名 ip 对应角色 mas ...

  8. 『jQuery』.html(),.text()和.val()的使用

    『jQuery』.html(),.text()和.val()的使用 2013-04-21 10:25 by 我是文东, 8335 阅读, 0 评论, 收藏, 编辑 本节内容主要介绍的是如何使用jQue ...

  9. 『StabilityGuide』| 10+位阿里技术专家共同发起稳定性知识库开源项目

    我们穿过山和大海,也见过人山人海.我们见过各类故障,也排过千雷万险.这一次,不如我们一起,开启稳定性的探索之旅.让无法解决的问题少一点点,让世界的确定性多一点点. 无论是前端业务的开发者,还是后端架构 ...

随机推荐

  1. js技巧(三)

    1.检测浏览器,search的用法 if(window.navigator.userAgent.search(/firefox/i)!=-1){ alert('ff'); } else if(wind ...

  2. HTML5——移动端的点击、拖拽

    移动端浏览器不支持mouse事件 https://www.cnblogs.com/joyco773/p/6519668.html https://www.cnblogs.com/yjhua/p/525 ...

  3. Jmeter常用的逻辑控制器

    一.ForEach控制器 作用:ForEach Controlle一般和用户自定义变量(User Defined Variables)一起使用,其在用户自定义变量中读取一系列相关的变量.每一个线程下执 ...

  4. R语言学习 - 箱线图(小提琴图、抖动图、区域散点图)

    箱线图 箱线图是能同时反映数据统计量和整体分布,又很漂亮的展示图.在2014年的Nature Method上有2篇Correspondence论述了使用箱线图的好处和一个在线绘制箱线图的工具.就这样都 ...

  5. js异步请求

    目前async / await特性并没有被添加到ES2016标准中,但不代表这些特性将来不会被加入到Javascript中.在我写这篇文章时,它已经到达第三版草案,并且正迅速的发展中.这些特性已经被I ...

  6. Django - 自定义filter

    自定义filter 自定义filter时,使用装饰器fileter 在html中,使用传参方式为: 参数1|函数名:参数2 并且函数和参数之间,不能有空格,如果有空格,会报错. filter和simp ...

  7. Gym - 101670E Forest Picture (CTU Open Contest 2017 模拟)

    题目: https://cn.vjudge.net/problem/1451310/origin 题意&思路: 纯粹模拟. 大体题意是这样的: 1.有人要在一个10-9<=x<=1 ...

  8. 第一章 Linux命令行简介

    1 Linux系统命令操作语法的格式 命令_[参数选项]_[文件或路径]    其中 _ 至少一个空格    如:rm -f /etc/hosts    其中/etc/hosts完整路径不带空格   ...

  9. Linux系统重要的子目录

    更多目录知识  http://blog.51cto.com/yangrong/1288072 /etc/fstab 机自动挂载分区/磁盘,规定哪个分区/设备,挂载到哪里 [root@oldboy ~] ...

  10. python3.x Day3 文件编码

    文件编码: 知识点不多,但及其重要,python2和python3处理机制还有不同点,需要注意. 首先: 编码.数据类型,完全不同的概念. 文件编码:可以遵循开发环境.可以自行设定. 变量值编码:py ...