Notes:
本来计划花个1、2天就把这个整理下,结果,越来越忙
,那索性就做个长远计划吧,花个1、2个月把这个整理完,也当成自己学习swift的一个笔记吧...
Time:
2016-04-23 ~ 2016-04-24
Location:
Shanghai
Content:
First Day:
- 《一个Swift项目的网络变迁 -- 陈乘方》
- 《Objective-C和Swift的跨平台开发 -- Tinyfool》
- 《逆向工程:从Objective-C到Swift -- 包涵卿》
- 《拥抱Swift3.0与未来展望 -- 图拉鼎》
- 《Swift语言的设计取舍及跨语言调用 -- 董一凡》
- 《从数学函数角度理解函数式编程 -- 丁峰》
Second Day:
- 《Swift设计模式 -- 老镇》
- 《Redux in Swift——新一代的iOS应用架构 -- 方志刚》
- 《Asynchronous Programming in Swift,Swift异步编程 -- 刘冠杉》
- 《读Swift源码,理解Monad -- 唐巧》
- 《How to parse float numbers-the really hard way -- 傅若愚》
Summarize
- 说直白点,这次会议持续2天,11个分享内容,看着面还是挺广的,从设计模式到语言层面再到应用和逆向,基本几个大的方向该有的都有了,不过听了2天云里雾里,回头就记得一句话"函数式编程"...
- 不过这种会议本身不能指望几个小时内学完人家的全部功力(唐巧大哥学了一年monad了,做了100多页PPT才做了这个不到一小时的分享),但是能找到几个方向就已经算是有收获了,哈哈.
- Swift目前的项目还没有用起来,这次分享的东西很多东西,估计近期都是不太可能接触或者使用的,所以赶紧总结下,免得过几个月,回头忘光了...
Detail
应用:
《一个Swift项目的网络变迁 -- 陈乘方》
polen:
这个就是把大象装进冰箱需要几步的问题(把OC项目改为Swift项目需要几步):
1. 把OC的代码照搬翻译成Swift
2. 开始"更Swift"地写代码:使用Enum Associated Values, Generic,Protocol Extension
//Enum Associated Values 枚举 和 关联值
enum Trade {
case Buy(stock: String, amount: Int)
case Sell(stock: String, amount: Int)
}
let trade = Trade.Buy(stock: "APPL", amount: 500)
if case let Trade.Buy(stock, amount) = trade {
print("buy \(amount) of \(stock)")
}
Enum深入学习-> http://www.tuicool.com/articles/b2YveqQ
//Generic 泛型
//泛型函数可以适应各种类型,这个是泛型版本的swapTwoInts函数,命名为swapTwoValues:
func swapTwoValues<T>(inout a: T, inout b: T) {
let temporaryA = a
a = b
b = temporaryA
}
Generic深入学习:http://www.tuicool.com/articles/YbYz6j
//Protocol Extension
protocol ExerciseType: CustomStringConvertible {
var name: String { get }
var caloriesBurned: Double { get }
var minutes: Double { get }
}
// 结构体遵守 ErerciseType 协议, 但是也可以有自己的属性和方法, 这也是一种继承和封装
struct EllipticalTrainer: ExerciseType {
let name = "Elliptical Machine"
let caloriesBurned: Double
let minutes: Double
}
struct Treadmill: ExerciseType {
let name = "Treadmill"
let caloriesBurned: Double
let minutes: Double
let distancesInMiles: Double
}
extension Treadmill {
var description: String {
return "Treadmill(\(caloriesBurned) calories and \(distancesInMiles) miles in \(minutes) minutes)"
}
}
扩展学习:面向协议编程的WWDC视频
Session 408: Protocol-Oriented Programming in Swift
3.回调方式改变
- success和fail合并到一个回调里面(completion)
(polen:一方面代码看起来更舒服,另外进行链式调用更加可行) - 定义了一个APIResult
- 参照Alamofire修改接口,把代码写的更漂亮
enum APIResult<T> {
case Success(T)
case Failure(NSError, AnyObject?)
}
4.网络model改变
- ObjectMapper代替Mantle
- Struct 类型替换了Class类型
- 尽可能减少optional 类型的属性
(polen:这个是真心实践出的收货)
5. 数据的处理通用组件
- 对数据处理过程的抽象
- 提供通用函数作为组件
- 组件函数的参数和返回值符合flatMap参数定义
6. 其他包括网络结果的链式调用等等等等...
这个App叫Enjoy,很实在的讲了他们一步步修改,一步步完善的经历,踩了很多坑,但是最终也成功走下来了,总体感觉作为首场演讲,还算不错,中规中矩,都是比较实用的方法或者技术,相当接地气 👍
《Asynchronous Programming in Swift,Swift异步编程 -- 刘冠杉》
一位阿里音乐的哥们,讲的时候,我在和旁边的同学聊天,很遗憾听几句
他是现场编程讲解型的,技术看着不错,但是现场效果貌似不是很好...
逆向:
《逆向工程:从Objective-C到Swift -- 包涵卿》
干货啊,干货
自己没怎么玩过逆向,其实把一个APP逆向还是比较简单的,classdump执行几条命令,基本人家的.h文件你能都看到了
包涵卿这哥们这次将的是,直接对Swift语言层面的逆向,相对更深入一些,不过现场没拍照,没记录,除了记得几个寄存器,大部分又忘掉了
可能很多人iOS写多了,底层的东西就慢慢忘记了(包括我)
我这里自己简单总结下,
如何快速当个iOS hacker:
1.学习下xcrun命令
打开终端,输入xcrun clang -v
xcrun clang -v
如果出现截图这个东西
okey,说明你已经会了xcrun了,嗯,我们继续
(补充: 如果有人真要问这个是干嘛,那我告诉你就是xcode run的意思,就是代替了xcode,以后你可以直接用这个命令打包,编译...)
2.编译代码
用vim也好,自己写也好,先新建一个helloword.c
#include <stdio.h>
int main(int argc,char *argv[]){
printf("birdies");
return 1;
}
然后使用我们的xcrun命令,编译我们的helloword
xcrun clang helloword.c
3. 学习汇编
接着上一步,执行如下命令生成汇编代码
xcrun clang -S -o - helloword.c | open -f
//以点 .开头的行。这些就是汇编指令。其它的则是实际的 x86_64 汇编代码
.section __TEXT,__text,regular,pure_instructions
.macosx_version_min 10, 11
.globl _main
.align 4, 0x90
_main: ## @main
.cfi_startproc
## BB#0:
pushq %rbp
Ltmp0:
.cfi_def_cfa_offset 16
Ltmp1:
.cfi_offset %rbp, -16
movq %rsp, %rbp
Ltmp2:
.cfi_def_cfa_register %rbp
subq $32, %rsp
leaq L_.str(%rip), %rax
movl $0, -4(%rbp)
movl %edi, -8(%rbp)
movq %rsi, -16(%rbp)
movq %rax, %rdi
movb $0, %al
callq _printf
movl $1, %ecx
movl %eax, -20(%rbp) ## 4-byte Spill
movl %ecx, %eax
addq $32, %rsp
popq %rbp
retq
.cfi_endproc
.section __TEXT,__cstring,cstring_literals
L_.str: ## @.str
.asciz "birdies"
.subsections_via_symbols
来,我们来分析一下这一段代码:
/* 以下内容是本文最干货的地方 **/
//以点 .开头的行。这些就是汇编指令。其它的则是实际的 x86_64 汇编代码
.section __TEXT,__text,regular,pure_instructions
.macosx_version_min 10, 11
.globl _main //globl说明main是个外部符号
.align 4, 0x90//对齐方式
//以上是指令,不是汇编代码,.section 说明接下来要执行哪一段
_main: ## @main
.cfi_startproc //与cfi_endproc相对应,标出main函数开始和结束的地方
## BB#0:
//rbp是个寄存器,ABI会先把rbp的值存到栈中,回头用的时候再取,避免被改掉
//rbp: 基础指针寄存器(base pointer register)
pushq %rbp
//下面这两条CFI指令用于输出一些调试用的堆栈信息,告诉编译器一些基础指针或者堆栈在哪里
Ltmp0:
.cfi_def_cfa_offset 16
Ltmp1:
.cfi_offset %rbp, -16
//rsp->rbp 把局部变量放到栈上(RSP:堆栈指针,通过这个指针存取堆栈存储器数据)
movq %rsp, %rbp
Ltmp2:
.cfi_def_cfa_register %rbp
//将栈指针移动32字节,移动到函数会调用的地方
subq $32, %rsp
//printf()函数调用, rax是个累加器,leaq会将 L_.str的指针加载到 rax寄存器中。
leaq L_.str(%rip), %rax
movl $0, -4(%rbp)
movl %edi, -8(%rbp)
movq %rsi, -16(%rbp)
movq %rax, %rdi
movb $0, %al
callq _printf
movl $1, %ecx
movl %eax, -20(%rbp) ## 4-byte Spill
movl %ecx, %eax
addq $32, %rsp
popq %rbp
retq
.cfi_endproc
.section __TEXT,__cstring,cstring_literals
L_.str: ## @.str
.asciz "birdies"
.subsections_via_symbols
关于这个的详细分析在这里
https://www.objc.io/issues/6-build-tools/mach-o-executables/
http://objccn.io/issue-6-3/
扩展:
iOS程序main函数之前发生了什么
http://blog.sunnyxx.com/2014/08/30/objc-pre-main/
2016-04-28 00:48:20 先写到这,明天要发版本,后面慢慢补充...
语言层面:
《拥抱Swift3.0与未来展望 -- 图拉鼎》
《Swift语言的设计取舍及跨语言调用 -- 董一凡》
函数式编程:
《从数学函数角度理解函数式编程 -- 丁峰》
《读Swift源码,理解Monad -- 唐巧》
《How to parse float numbers-the really hard way -- 傅若愚》
《Objective-C和Swift的跨平台开发 -- Tinyfool》
《Swift设计模式 -- 老镇》
《Redux in Swift——新一代的iOS应用架构 -- 方志刚》