视频课程地址:http://www.jikexueyuan.com/course/1420.html

快捷键的使用只是操作提高速度,还需要学习更多的设计方面的技巧。视频学习确实速度够快,就像看电影,不容暂停分心才能跟上进度。

  • 常用快捷键和自定义快捷键
  • 快速导出图片资源
  • 使用快捷键控制定位视图

Original post on Ray Wenderlich: http://www.raywenderlich.com/13418/how-to-play-record-edit-videos-in-ios

This post is a step by step instruction. Now I copy add some explanation just in the code. Hope I can understand the every step meaning and easy to read.

Edit in AVFoundation

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292

//
// MergeVideo.m
// VideoPlayRecord
//
// Created by Abdul Azeem Khan on 5/9/12.
// Copyright (c) 2012 DataInvent. All rights reserved.
//

#import "MergeVideo.h"

@implementation MergeVideo
@synthesize ActivityView;
@synthesize firstAsset, secondAsset, audioAsset;


- (void)viewDidLoad {

[super viewDidLoad];
NSLog(@"Loaded");
// Do any additional setup after loading the view, typically from a nib.
}

- (void)didReceiveMemoryWarning {

// Releases the view if it doesn't have a superview.
[super didReceiveMemoryWarning];

// Release any cached data, images, etc that aren't in use.
}


#pragma mark - View lifecycle
#pragma mark - Load assests methods

- (IBAction)LoadAssetOne:(id)sender {
if ([UIImagePickerController isSourceTypeAvailable:
UIImagePickerControllerSourceTypeSavedPhotosAlbum] == NO)
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error" message:@"No Saved Album Found" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];
}else{
isSelectingAssetOne = TRUE;
[self startMediaBrowserFromViewController: self
usingDelegate: self];
}
}
- (IBAction)LoadAssetTwo:(id)sender {
if ([UIImagePickerController isSourceTypeAvailable:
UIImagePickerControllerSourceTypeSavedPhotosAlbum] == NO)
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error" message:@"No Saved Album Found" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];

}else{
isSelectingAssetOne = FALSE;
[self startMediaBrowserFromViewController: self
usingDelegate: self];
}
}

- (BOOL) startMediaBrowserFromViewController: (UIViewController*) controller
usingDelegate: (id <UIImagePickerControllerDelegate,
UINavigationControllerDelegate>) delegate {
if (([UIImagePickerController isSourceTypeAvailable:
UIImagePickerControllerSourceTypeSavedPhotosAlbum] == NO)
|| (delegate == nil)
|| (controller == nil))
return NO;
UIImagePickerController *mediaUI = [[UIImagePickerController alloc] init];
mediaUI.sourceType = UIImagePickerControllerSourceTypeSavedPhotosAlbum;

mediaUI.mediaTypes = [[NSArray alloc] initWithObjects: (NSString *) kUTTypeMovie, nil];

// Hides the controls for moving & scaling pictures, or for
// trimming movies. To instead show the controls, use YES.
mediaUI.allowsEditing = YES;

mediaUI.delegate = delegate;

[self presentViewController: mediaUI animated: YES completion:nil];
return YES;
}

- (IBAction)LoadAudio:(id)sender {
MPMediaPickerController *mediaPicker = [[MPMediaPickerController alloc] initWithMediaTypes: MPMediaTypeAny];
mediaPicker.delegate = self;
mediaPicker.prompt = @"Select Music";
[self presentViewController:mediaPicker animated:YES completion:nil];
}


#pragma mark - Conform MPMediaPickerControllerDelegate
- (void) mediaPicker: (MPMediaPickerController *) mediaPicker didPickMediaItems: (MPMediaItemCollection *) mediaItemCollection
{
NSArray * SelectedSong = [mediaItemCollection items];
if([SelectedSong count]>0){
MPMediaItem * SongItem = [SelectedSong objectAtIndex:0];
NSURL *SongURL = [SongItem valueForProperty: MPMediaItemPropertyAssetURL];

audioAsset = [AVAsset assetWithURL:SongURL];
NSLog(@"Audio Loaded");
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Asset Loaded" message:@"Audio Loaded" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];
}

[self dismissViewControllerAnimated: YES completion:nil];
}

- (void) mediaPickerDidCancel: (MPMediaPickerController *) mediaPicker {

[self dismissViewControllerAnimated: YES completion:nil];

}


#pragma mark - Conform UIImagePickerControllerDelegate
// For responding to the user tapping Cancel.
- (void) imagePickerControllerDidCancel: (UIImagePickerController *) picker {

[self dismissViewControllerAnimated: YES completion:nil];
}


// For responding to the user accepting a newly-captured picture or movie
- (void) imagePickerController: (UIImagePickerController *) picker
didFinishPickingMediaWithInfo: (NSDictionary *) info {

NSString *mediaType = [info objectForKey: UIImagePickerControllerMediaType];
[self dismissModalViewControllerAnimated:NO];

// Handle a movie capture
if (CFStringCompare ((__bridge_retained CFStringRef) mediaType, kUTTypeMovie, 0)
== kCFCompareEqualTo) {
if(isSelectingAssetOne){
NSLog(@"Video One Loaded");
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Asset Loaded" message:@"Video One Loaded" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];
firstAsset = [AVAsset assetWithURL:[info objectForKey:UIImagePickerControllerMediaURL]];
}else{
NSLog(@"Video two Loaded");
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Asset Loaded" message:@"Video Two Loaded" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];
secondAsset = [AVAsset assetWithURL:[info objectForKey:UIImagePickerControllerMediaURL]];
}
}
}


#pragma mark - MergeAndSave methods

- (IBAction)MergeAndSave:(id)sender {

if(firstAsset !=nil && secondAsset!=nil) {

[ActivityView startAnimating];

// 1 - Create AVMutableComposition Object.This object will hold our multiple AVMutableCompositionTrack.
// AVMutableComposition is a mutable subclass of AVComposition you use when you want to create a new composition from existing assets. You can add and remove tracks, and you can add, remove, and scale time ranges.
AVMutableComposition* mixComposition = [[AVMutableComposition alloc] init];

// 2 - VIDEO TRACK
// AVMutableCompositionTrack lets you for insert, remove, and scale track segments without affecting their low-level representation (that is, the operations you perform are non-destructive on the original).
AVMutableCompositionTrack *firstTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid];
[firstTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, firstAsset.duration) ofTrack:[[firstAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:kCMTimeZero error:nil];

AVMutableCompositionTrack *secondTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid];
[secondTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, secondAsset.duration) ofTrack:[[secondAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:firstAsset.duration error:nil];

// 3 - AUDIO TRACK
if(audioAsset!=nil) {

AVMutableCompositionTrack *AudioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];
[AudioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, CMTimeAdd(firstAsset.duration, secondAsset.duration)) ofTrack:[[audioAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0] atTime:kCMTimeZero error:nil];
}

// 4 - Set AVMutableVideoCompositionLayerInstruction
// AVMutableVideoCompositionLayerInstruction used to modify the transform, cropping, and opacity ramps to apply to a given track in a composition.
AVMutableVideoCompositionInstruction * MainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];
MainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, CMTimeAdd(firstAsset.duration, secondAsset.duration));

//FIXING ORIENTATION//
AVMutableVideoCompositionLayerInstruction *FirstlayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:firstTrack];
AVAssetTrack *FirstAssetTrack = [[firstAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];
UIImageOrientation FirstAssetOrientation_ = UIImageOrientationUp;
BOOL isFirstAssetPortrait_ = NO;
CGAffineTransform firstTransform = FirstAssetTrack.preferredTransform;
if(firstTransform.a == 0 && firstTransform.b == 1.0 && firstTransform.c == -1.0 && firstTransform.d == 0) {FirstAssetOrientation_= UIImageOrientationRight; isFirstAssetPortrait_ = YES;}
if(firstTransform.a == 0 && firstTransform.b == -1.0 && firstTransform.c == 1.0 && firstTransform.d == 0) {FirstAssetOrientation_ = UIImageOrientationLeft; isFirstAssetPortrait_ = YES;}
if(firstTransform.a == 1.0 && firstTransform.b == 0 && firstTransform.c == 0 && firstTransform.d == 1.0) {FirstAssetOrientation_ = UIImageOrientationUp;}
if(firstTransform.a == -1.0 && firstTransform.b == 0 && firstTransform.c == 0 && firstTransform.d == -1.0) {FirstAssetOrientation_ = UIImageOrientationDown;}
CGFloat FirstAssetScaleToFitRatio = 320.0/FirstAssetTrack.naturalSize.width;
if(isFirstAssetPortrait_){
FirstAssetScaleToFitRatio = 320.0/FirstAssetTrack.naturalSize.height;
CGAffineTransform FirstAssetScaleFactor = CGAffineTransformMakeScale(FirstAssetScaleToFitRatio,FirstAssetScaleToFitRatio);
[FirstlayerInstruction setTransform:CGAffineTransformConcat(FirstAssetTrack.preferredTransform, FirstAssetScaleFactor) atTime:kCMTimeZero];
}else{
CGAffineTransform FirstAssetScaleFactor = CGAffineTransformMakeScale(FirstAssetScaleToFitRatio,FirstAssetScaleToFitRatio);
[FirstlayerInstruction setTransform:CGAffineTransformConcat(CGAffineTransformConcat(FirstAssetTrack.preferredTransform, FirstAssetScaleFactor),CGAffineTransformMakeTranslation(0, 160)) atTime:kCMTimeZero];
}
[FirstlayerInstruction setOpacity:0.0 atTime:firstAsset.duration];

AVMutableVideoCompositionLayerInstruction *SecondlayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:secondTrack];
AVAssetTrack *SecondAssetTrack = [[secondAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];
UIImageOrientation SecondAssetOrientation_ = UIImageOrientationUp;
BOOL isSecondAssetPortrait_ = NO;
CGAffineTransform secondTransform = SecondAssetTrack.preferredTransform;
if(secondTransform.a == 0 && secondTransform.b == 1.0 && secondTransform.c == -1.0 && secondTransform.d == 0) {SecondAssetOrientation_= UIImageOrientationRight; isSecondAssetPortrait_ = YES;}
if(secondTransform.a == 0 && secondTransform.b == -1.0 && secondTransform.c == 1.0 && secondTransform.d == 0) {SecondAssetOrientation_ = UIImageOrientationLeft; isSecondAssetPortrait_ = YES;}
if(secondTransform.a == 1.0 && secondTransform.b == 0 && secondTransform.c == 0 && secondTransform.d == 1.0) {SecondAssetOrientation_ = UIImageOrientationUp;}
if(secondTransform.a == -1.0 && secondTransform.b == 0 && secondTransform.c == 0 && secondTransform.d == -1.0) {SecondAssetOrientation_ = UIImageOrientationDown;}
CGFloat SecondAssetScaleToFitRatio = 320.0/SecondAssetTrack.naturalSize.width;
if(isSecondAssetPortrait_){
SecondAssetScaleToFitRatio = 320.0/SecondAssetTrack.naturalSize.height;
CGAffineTransform SecondAssetScaleFactor = CGAffineTransformMakeScale(SecondAssetScaleToFitRatio,SecondAssetScaleToFitRatio);
[SecondlayerInstruction setTransform:CGAffineTransformConcat(SecondAssetTrack.preferredTransform, SecondAssetScaleFactor) atTime:firstAsset.duration];
}else{
;
CGAffineTransform SecondAssetScaleFactor = CGAffineTransformMakeScale(SecondAssetScaleToFitRatio,SecondAssetScaleToFitRatio);
[SecondlayerInstruction setTransform:CGAffineTransformConcat(CGAffineTransformConcat(SecondAssetTrack.preferredTransform, SecondAssetScaleFactor),CGAffineTransformMakeTranslation(0, 160)) atTime:firstAsset.duration];
}


MainInstruction.layerInstructions = [NSArray arrayWithObjects:FirstlayerInstruction,SecondlayerInstruction,nil];;


// 5 - Mergo AVMutableVideoCompositionLayerInstruction into AVMutableVideoComposition
// AVMutableComposition is a mutable subclass of AVComposition you use when you want to create a new composition from existing assets. You can add and remove tracks, and you can add, remove, and scale time ranges.
AVMutableVideoComposition *MainCompositionInst = [AVMutableVideoComposition videoComposition];
MainCompositionInst.instructions = [NSArray arrayWithObject:MainInstruction];
MainCompositionInst.frameDuration = CMTimeMake(1, 30);
MainCompositionInst.renderSize = CGSizeMake(320.0, 480.0);

// 6 - Get path
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentsDirectory = [paths objectAtIndex:0];
NSString *myPathDocs = [documentsDirectory stringByAppendingPathComponent:[NSString stringWithFormat:@"mergeVideo-%d.mov",arc4random() % 1000]];

NSURL *url = [NSURL fileURLWithPath:myPathDocs];

// 7 - Create exporter with AVAssetExportSession
// An AVAssetExportSession object transcodes the contents of an AVAsset source object to create an output of the form described by a specified export preset.
AVAssetExportSession *exporter = [[AVAssetExportSession alloc] initWithAsset:mixComposition presetName:AVAssetExportPresetHighestQuality];
exporter.outputURL=url;
exporter.outputFileType = AVFileTypeQuickTimeMovie;
exporter.videoComposition = MainCompositionInst;
exporter.shouldOptimizeForNetworkUse = YES;
[exporter exportAsynchronouslyWithCompletionHandler:^
{
dispatch_async(dispatch_get_main_queue(), ^{
[self exportDidFinish:exporter];
});
}];
}
}

// exportDidFinish method: save the mergo video to Photos Album
- (void)exportDidFinish:(AVAssetExportSession*)session {

if(session.status == AVAssetExportSessionStatusCompleted) {

NSURL *outputURL = session.outputURL;
// 8 - Output video
// An instance of ALAssetsLibrary provides access to the videos and photos that are under the control of the Photos application.
ALAssetsLibrary *library = [[ALAssetsLibrary alloc] init];

if ([library videoAtPathIsCompatibleWithSavedPhotosAlbum:outputURL]) {

[library writeVideoAtPathToSavedPhotosAlbum:outputURL
completionBlock:^(NSURL *assetURL, NSError *error){
dispatch_async(dispatch_get_main_queue(), ^{
if (error) {
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error" message:@"Video Saving Failed" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles: nil, nil];
[alert show];
}else{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Video Saved" message:@"Saved To Photo Album" delegate:self cancelButtonTitle:@"Ok" otherButtonTitles: nil];
[alert show];
}

});

}];
}
}

audioAsset = nil;
firstAsset = nil;
secondAsset = nil;
[ActivityView stopAnimating];
}
@end

今天主要是和项目的团队沟通进度,Tonyfool的分享很有料。

Tonyfool 分享

演讲能力

忽悠能力低于水平,不会表达,就很吃亏。

藏拙,不添加有问题的的功能。iPhone 前几版都少了很多功能的,基于产品迭代和

演讲追求哇效应,出乎观众的意料。

别人都有的功能的,就跳过不讲。卖亮点功能,融入场景故事中去演示,如 Apple keynote。

iOS 强调 UE。内置 app 已经用的很舒服很易用,谁便一个产品经理不能画出其产品流畅图。

系统化解决问题

代码调优和 Debug,如何系统化解决新的问题:猜可能是那几个方面出了问题,全面分析完了,再去全面的验证调试解决。而不是想到一个可能一个想法就去试。

摘自 sunnyxx 的文章:一个丝滑的全屏滑动返回手势

status bar

UIApplication 全局的 status bar,牵一发还得动全身,不过 Apple 在 iOS7 之后为 vc 控制自己的 status bar 提供了下面几个方法,终于让这个全局变量变成了局部变量:

1
2
3
- (UIStatusBarStyle)preferredStatusBarStyle NS_AVAILABLE_IOS(7_0);
- (BOOL)prefersStatusBarHidden NS_AVAILABLE_IOS(7_0);
- (UIStatusBarAnimation)preferredStatusBarUpdateAnimation NS_AVAILABLE_IOS(7_0);

fullscreen-pop-gesture

一个 UINavigationController 管理了串行的 N 个 UIViewController 栈式的 push 和 pop,而 UINavigationBar 由 UINavigationController 管理,这就导致了 UIViewController 无法控制自己上面的 bar 单独的隐藏或显示。
但是对 UINavigationBar 的控制,依然是全局的,可能 Apple 觉得 App 不应该有这种奇怪的页面结构?

解决这个问题的方法也不难,在滑动返回的后要出现的那个 view controller 中写下面的代码:

1
2
3
4
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
[self.navigationController setNavigationBarHidden:YES animated:animated];
}

fullscreen-pop-gesture

Xcode tips

  • Secondary click : Jump to definition
  • Show Relative Items(^1): Callers
    Show Relative Items

一下看了9/10两课,有点烧脑也好,走出舒适区也好, 困了也好,反正看完睡了2小时。现在刚起来写笔记,稍后实践知识点加到项目中。没开字幕,因为我的阅读速度还不如听力。

Tinyfool:看英文视频不看字幕。

ibuick:学习知识后,在另外一个小 App 去运用

Scroll View

概念和使用都很简单,理解成画布、画稿、放大镜即可。

Multithreading

典型的用法:

1
2
3
4

let session = NSURLSession(NSURLSessionConfiguration.defaultSessionConfiguration())
if let url = NSURL(string: “http://url”) { let request = NSURLRequest(URL: url) let task = session.downloadTaskWithRequest(request) {
(localURL, response, error) in  dispatch_async(dispatch_get_main_queue()) { /* I want to do something in the UI here, can I? */ } } task.resume() }

View Controller Lifecycle

  • Instantiated (from storyboard usually)
  • awakeFromNib
  • segue preparation happens
  • outlets get set
  • viewDidLoad

These pairs will be called each time your Controller’s view goes on/off screen …

  • viewWillAppear and viewDidAppear
  • viewWillDisappear and viewDidDisappear

These “geometry changed” methods might be called at any time after viewDidLoad …

  • viewWillLayoutSubviews (… then autolayout happens, then …) viewDidLayoutSubviews

If memory gets low, you might get …

  • didReceiveMemoryWarning

Autolayout

You’ve seen a lot of Autolayout already

  • Using the dashed blue lines to try to tell Xcode what you intend
  • Ctrl-Dragging between views to create relationships (spacing, etc.)
  • The “Pin” and “Arrange” popovers in the lower right of the storyboard
  • Reset to Suggested Constraints (if the blue lines were enough to unambiguously set constraints)
  • Document Outline (see all constraints, resolve misplacements and even conflicts)
  • Size Inspector (look at (and edit!) the details of the constraints on the selected view)
  • Clicking on a constraint to select it then bring up Attributes Inspector (to edit its details)

Mastering Autolayout requires experience

You just have to do it to learn it

Autolayout can be done from code too

Though you’re probably better off doing it in the storyboard wherever possible The demo today will show a simple case of doing Autolayout from code

Demo:

  • Notice auto layout issues and resove them.
  • Preview Storyboard in assistant editor
  • Remove magic nubmers in constraints of size inspector: such as use standard value or 0.

More:

What does “Use standard value and Constrain to Margins” mean in Auto Layout?

Auto Layout Concepts

The fundamental building block in Auto Layout is the constraint. Constraints express rules for the layout of elements in your interface;

Constraint Basics

You can think of a constraint as a mathematical representation of a human-expressable statement. If you’re defining the position of a button, for example, you might want to say “the left edge should be 20 points from the left edge of its containing view.” More formally, this translates to button.left = (container.left + 20), which in turn is an expression of the form y = m*x + b, where:

y and x are attributes of views.

m and b are floating point values.

An attribute is one of left, right, top, bottom, leading, trailing, width, height, centerX, centerY, and baseline.

Auto Layout Tutorial Part 1: Getting Started

fastlane

fastlane lets you define and run your deployment pipelines for different environments. It helps you unify your app’s release process and automate the whole process. fastlane connects all fastlane tools and third party tools, like CocoaPods and xctool.

fastlane 是一款自动化 App 提交流程工具,也拆分了开了每一个工具,也可以直接按照下面的工作流一步提交 App Store。

  • snapshot
    把你的iOS应用程序的本地化语言自动截屏

  • deliver
    一行命令上传截图,元数据和应用程序到 App Store

  • frameit
    快速把你的截图到正确的设备框架

  • gym
    快速打包 iOS 应用程序成 ipa

感兴趣的话,赶紧去官网 https://fastlane.tools/#https://github.com/KrauseFx/fastlane 试一试。希望能够节约到你宝贵的时间。

fastlane workflow

Video Tutorial

  1. Add first View Controller Segue to new View Controller
  2. Add Bar Button Item on first View Controller and set target-action
  3. code in action:method add self.porformSegueWithIdentifier
  4. setToolbarHidden(_ hidden: Bool, animated animated: Bool)
  5. Or set in interface builder

Show and Hide Bottom by code
Show and Hide Bottom set in interface builder

Demo Project on GitHub

More in Segue

Simpler way:

Adding a Segue Between Scenes in a Storyboard

Another way:View Controller segue to View Controller,
your can something more in target-action method.

IOS7, Segue and storyboards - How to create without a button?

design_patterns

总体感觉设计模式课程偏理论,操作起来可能要反复实践后,才会知道项目用哪个设计模式比较合适。

我目前还是学习了解的还是六大基本设计模式,希望以后开发过程中能够思考设计模式的选择。

@李建忠:设计模式是程序员从初阶往中阶迈进很重要的素养。关键在于要去思考设计的方案,为什么这么做?还有哪些设计方法?彼此的优劣?而不是功能跑通就了事。 在这样日积月累的思考和反复实践中,设计功力就会提高。

1. 设计模式简介和 MVC

简单介绍了设计模式的含义,和一些常用的 iOS 设计模式。

xdrt81y 的 关于iOS六大基本设计模式,文章简单明了可以参考一下。

Introducing iOS Design Patterns in Swift – Part 1/2 结合项目介绍了 Swift 中的设计模式。

使用设计模式,苹果文档介绍。

Wikipedia:

In software engineering, a design pattern is a general reusable solution to a commonly occurring problem within a given context in software design.
It is a description or template for how to solve a problem that can be used in many different situations.
Patterns are formalized best practices that the programmer can use to solve common problems when designing an application or system.

2. Target-Action、selector 和 Nib

详细介绍了 Nib 加载

3. Storyboard 和原型设计

Storyboard

我还是很喜欢 Storyboard,比代码直观高效,方便看到原型图和交互关系。

拆分 Storyboard,避免多人同时修改。

推荐:WWDC15 215 What’s New in Storyboards

参考:iOS 9: Staying Organized with Storyboard References

  • instantiateInitialViewController()
  • destination of a segue

Storyboard references are powerful. Not only do they make storyboards manageable and modular, they make it very easy to reuse storyboards and even hook into storyboards at arbitrary places.

Storyboard references are only available in iOS 9.

代码见 GitHub

4. 两步创建和模板方法

The best way to predict your future is to create it.

― Abraham Lincoln

两步创建

对象两步创建:alloc init

工厂方法:+ (instancetype)dateWithTimeIntervalSinceNow:(NSTimeInterval)seconds

内省(Introspection)

模板方法

参考:lichwei1983 的文章 模板方法–行为型模式之四

模板方法应用于下列情况:

  • 一次性实现一个算法的不变的部分,并将可变的行为留给子类来实现。
  • 各子类中公共的行为应被提取出来并集中到一个公共父类中以避免代码重复。这是 Opdyke 和 Johnson 所描述过的“重分解以一般化”的一个很好的例子。首先识别现有代码中的不同之处,并且将不同之处分离为新的操作。最后,用一个调用这些新的操作的模板方法来替换这些不同的代码。
  • 控制子类扩展。模板方法只在特定点调用“ h o o k”操作(参见效果一节),这样就只允许在这些点进行扩展。

模板方法是一种代码复用的基本技术。它们在类库中尤为重要,它们提取了类库中的公共行为。模板方法导致一种反向的控制结构,这种结构有时被称为“好莱坞法则”,即“别找我们,我们找你”。这指的是一个父类调用一个子类的操作,而不是相反。

5. 单例模式

  • UIApplication
  • NSNotificationCenter
  • NSUserDefaults
  • NSFileManager
  • UIAccelerometer
  • NSURLSession

参考:刚刚在线的 iOS 设计模式系列:Singleton – 单例模式

单例设计模式确切的说就是一个类只有一个实例,有一个全局的接口来访问这个实例。当第一次载入的时候,它通常使用延时加载的方法创建单一实例。

在一些情况下,一个类只有一个实例是有意义的。例如,这里没有必要有多个登录实例,除非你一次想写入多个日志文件。或者,一个全局的配置类文件:它可以很容易的很安全的执行一个公共资源,这样的一个配置文件,要比同时修改多个配置类文件好很多。

转发整理一些我认为值得学习的小节。

-原文 https://github.com/objc-zen/objc-zen-book

-Gitbook 链接(包含 PDF, mobi, epub 格式): http://yourtion.gitbooks.io/objc-zen-book-cn/ (感谢 yourtion 整理 )

nil 和 BOOL 检查

类似于 Yoda 表达式,nil 检查的方式也是存在争议的。一些 notous 库像这样检查对象是否为 nil:

1
if (nil == myValue) { ...

或许有人会提出这是错的,因为在 nil 作为一个常量的情况下,这样做就像 Yoda 表达式了。 但是一些程序员这么做的原因是为了避免调试的困难,看下面的代码:

1
if (myValue == nil) { ...

如果程序员敲错成这样:

1
if (myValue = nil) { ...

这是合法的语句,但是即使你是一个丰富经验的程序员,即使盯着眼睛瞧上好多遍也很难调试出错误。但是如果把 nil 放在左边,因为它不能被赋值,所以就不会发生这样的错误。 如果程序员这样做,他/她就可以轻松检查出可能的原因,比一遍遍检查敲下的代码要好很多。

为了避免这些奇怪的问题,可以用感叹号来作为运算符。因为 nil 是 解释到 NO,所以没必要在条件语句里面把它和其他值比较。同时,不要直接把它和 YES 比较,因为 YES 的定义是 1, 而 BOOL 是 8 bit的,实际上是 char 类型。

推荐:

1
2
3
if (someObject) { ...
if (![someObject boolValue]) { ...
if (!someObject) { ...

不推荐:

1
2
3
if (someObject == YES) { ... // Wrong
if (myRawValue == YES) { ... // Never do this.
if ([someObject boolValue] == NO) { ...

同时这样也能提高一致性,以及提升可读性。

Read more »
0%