「 iOS知識小集 」2018 · 第 29 期

知識小集發表於2018-09-10

原文連結

本週我們啟動主題收集,如果你有想了解的 iOS 技術方面的主題,請在我們的 github https://github.com/awesome-tips/iOS-Tips/ 提 issue,也可以在相關的 issue 推薦文章。我們後期會每週整理一個主題。

另外,我們的小程式 猿小薦 正式釋出,我們希望為大家提供一個良好的職位釋出和檢視工具,後期也會不斷加入其它功能,歡迎大家試用。我們在後面的每篇文章中,也會加入一個職位推薦。

上週公眾號釋出的以下文章:

本期知識小集的主要內容包括:

  • 一入IAP深似海
  • iOS 金額字串格式化顯示的方法
  • 幾個第三方框架關於執行緒鎖的封裝小技巧

一入IAP深似海

作者: 高老師很忙

在做 IAP 的時候,我們通常會給 SKMutablePayment 物件的 applicationUsername 傳入一個值,比如說使用者ID的雜湊值,等交易成功後,通過 transaction.payment.applicationUsername 與之前傳入的值進行對比校驗。

最近發現,交易成功回撥返回的 transaction.payment.applicationUsername 有可能返回空,並且測試階段很難發現。解決辦法就是增加異常處理邏輯,當 transaction.payment.applicationUsername 返回空的時候,要加入一些業務相關的邏輯判斷。例如,你的 applicationUsername 傳入的是使用者ID相關的資訊,當 transaction.payment.applicationUsername 返回為空的時候,就要增加判斷髮生購買行為和收到回撥時是否是同一個使用者的邏輯,或者根據你具體的業務而定補充邏輯。

iOS 金額字串格式化顯示的方法

作者: KANGZUBIN

在一些金融類的 App 中,對於表示金額類的字串,通常需要進行格式化後再顯示出來。例如:

  • 0 --> 0.00
  • 123 --> 123.00
  • 123.456 --> 123.46
  • 102000 --> 102,000.00
  • 10204500 --> 10,204,500.00

它的規則如下:

個位數起每隔三位數字新增一個逗號,同時保留兩位小數,也稱為“千分位格式”。

我們一開始採取了一種比較笨拙的處理方式如下:

首先根據小數點 . 將傳入的字串分割為兩部分,整數部分和小數部分(如果沒有小數點,則補 .00,如果有多個小數點則報金額格式錯誤)。對於小數部分,只取前兩位;然後對整數部分字串進行遍歷,從右到左,每三位數前插入一個逗號 ,,最後再把兩部分拼接起來,程式碼大致如下:

- (NSString *)moneyFormat:(NSString *)money {
    if (!money || money.length == 0) {
        return money;
    }

    BOOL hasPoint = NO;
    if ([money rangeOfString:@"."].length > 0) {
        hasPoint = YES;
    }

    NSMutableString *pointMoney = [NSMutableString stringWithString:money];
    if (hasPoint == NO) {
        [pointMoney appendString:@".00"];
    }

    NSArray *moneys = [pointMoney componentsSeparatedByString:@"."];
    if (moneys.count > 2) {
        return pointMoney;
    } else if (moneys.count == 1) {
        return [NSString stringWithFormat:@"%@.00", moneys[0]];
    } else {
        // 整數部分每隔 3 位插入一個逗號
        NSString *frontMoney = [self stringFormatToThreeBit:moneys[0]];
        if ([frontMoney isEqualToString:@""]) {
            frontMoney = @"0";
        }
        // 拼接整數和小數兩部分
        NSString *backMoney = moneys[1];
        if ([backMoney length] == 1) {
            return [NSString stringWithFormat:@"%@.%@0", frontMoney, backMoney];
        } else if ([backMoney length] > 2) {
            return [NSString stringWithFormat:@"%@.%@", frontMoney, [backMoney substringToIndex:2]];
        } else {
            return [NSString stringWithFormat:@"%@.%@", frontMoney, backMoney];
        }
    }
}
複製程式碼

其中,stringFormatToThreeBit: 方法的實現如下:

- (NSString *)stringFormatToThreeBit:(NSString *)string {
    NSString *tempString = [string stringByReplacingOccurrencesOfString:@"," withString:@""];
    NSMutableString *mutableString = [NSMutableString stringWithString:tempString];
    NSInteger n = 2;
    for (NSInteger i = tempString.length - 3; i > 0; i--) {
        n++;
        if (n == 3) {
            [mutableString insertString:@"," atIndex:i];
            n = 0;
        }
    }
    return mutableString;
}
複製程式碼

上述實現看起來非常繁瑣。

其實,蘋果提供了 NSNumberFormatter 用來處理 NSStringNSNumber 之間的轉化,可以滿足基本的數字形式的格式化。我們通過設定 NSNumberFormatternumberStylepositiveFormat 屬性,即可實現上述功能,非常簡潔,程式碼如下:

- (NSString *)formatDecimalNumber:(NSString *)string {
    if (!string || string.length == 0) {
        return string;
    }
    
    NSNumber *number = @([string doubleValue]);
    NSNumberFormatter *formatter = [[NSNumberFormatter alloc] init];
    formatter.numberStyle = kCFNumberFormatterDecimalStyle;
    formatter.positiveFormat = @"###,##0.00";
    
    NSString *amountString = [formatter stringFromNumber:number];
    return amountString;
}
複製程式碼

關於 NSNumberFormatter 更詳細的用法,可以參考這篇文章的介紹:NSNumberFormatter 介紹和用法

幾個第三方框架關於執行緒鎖的封裝小技巧

作者: 陳滿iOS

啟示

第三方庫中經常用到的這個小技巧,例如 YYCacheSDWebImage 等等,雖然各自封裝的具體形式不太一樣。

  • YYCache

YYCache

  • SDWebImage

SDWebImage

  • YYWebImage

YYWebImage

我們可以借鑑到自己的專案中,在適當的位置通過巨集來加鎖解鎖操作。

使用

1. YYCache 版本的巨集封裝

#define Lock() dispatch_semaphore_wait(self->_lock, DISPATCH_TIME_FOREVER)
#define Unlock() dispatch_semaphore_signal(self->_lock)
複製程式碼
  • 運算元據之前,先外面進行加鎖解鎖
- (NSInteger)totalCount {
	Lock();
	int count = [_kv getItemsCount];
	Unlock();
	return count;
}
複製程式碼
  • 鎖裡面再進行真正的資料操作
- (int)getItemsCount {
	return [self _dbGetTotalItemCount];
}
複製程式碼

2. SDWebImage版本的巨集封裝

  • 定義
#define LOCK(lock) dispatch_semaphore_wait(lock, DISPATCH_TIME_FOREVER);
#define UNLOCK(lock) dispatch_semaphore_signal(lock);
複製程式碼
  • 呼叫示例
- (void)setValue:(nullable NSString *)value forHTTPHeaderField:(nullable NSString *)field {
	LOCK(self.headersLock);
	if (value) {
		self.HTTPHeaders[field] = value;
	} else {
		[self.HTTPHeaders removeObjectForKey:field];
	}
	UNLOCK(self.headersLock);
}
複製程式碼

其中,self.headersLock 的定義為:

@property (strong, nonatomic, nonnull) dispatch_semaphore_t headersLock; 
複製程式碼

3. YYWebImage版本的巨集封裝

相對於上面,還有更方便的巨集封裝,把解鎖操作跟加鎖封裝在一塊。

  • 巨集定義
#define LOCK(...) dispatch_semaphore_wait(self->_lock, DISPATCH_TIME_FOREVER); \
__VA_ARGS__; \
dispatch_semaphore_signal(self->_lock);

#define LOCK_VIEW(...) dispatch_semaphore_wait(view->_lock, DISPATCH_TIME_FOREVER); \
__VA_ARGS__; \
dispatch_semaphore_signal(view->_lock);
複製程式碼
  • 使用示例
- (void)didReceiveMemoryWarning:(NSNotification *)notification {
	[_requestQueue cancelAllOperations];
	[_requestQueue addOperationWithBlock: ^{
		_incrBufferCount = -60 - (int)(arc4random() % 120); // about 1~3 seconds to grow back..
		NSNumber *next = @((_curIndex + 1) % _totalFrameCount);
		LOCK(NSArray * keys = _buffer.allKeys;
			for (NSNumber * key in keys) {
				if (![key isEqualToNumber:next]) { // keep the next frame for smoothly animation
					[_buffer removeObjectForKey:key];
				}
			}
		)//LOCK
	}];
}
複製程式碼

關注我們

歡迎關注我們的公眾號:iOS-Tips,也歡迎加入我們的群組討論問題。可以公眾號留言 iosflutterwebpwa小程式 等關鍵詞獲取入群方式。

「 iOS知識小集 」2018 · 第 29 期

相關文章