Swift 部分干貨總結(jié)

DDLog

/// Debug模式日志打印
/// - Parameters:
///   - message: 內(nèi)容
///   - file: 文件名
///   - function: 方法名
///   - line: 行號
public func DDLog(_ message: Any?..., file: String = #file, function: String = #function, line: Int = #line){
    #if DEBUG
    let params = message.compactMap{ "\($0)" }.joined(separator: ", ");
    
    let fmt = DateFormatter.format("yyyy-MM-dd HH:mm:ss.SSSSSSZ");
    fmt.locale = Locale(identifier: "zh_CN")
    
    let dateStr = fmt.string(from: Date())
    print(dateStr, "\((file as NSString).lastPathComponent).\(function)[line \(line)]: \(params)")
    #endif
}

UITableViewCell 復(fù)用方法

@objc public extension UITableViewCell{
    /// [源]自定義 UITableViewCell 獲取方法(兼容OC)
    static func dequeueReusableCell(_ tableView: UITableView, identifier: String, style: UITableViewCell.CellStyle = .default) -> Self {
        var cell = tableView.dequeueReusableCell(withIdentifier: identifier);
        if cell == nil {
            cell = self.init(style: style, reuseIdentifier: identifier);
        }

        cell!.selectionStyle = .none
        cell!.separatorInset = .zero
        cell!.layoutMargins = .zero
        cell!.backgroundColor = .white
        return cell as! Self;
    }
    
    /// [OC簡潔方法]自定義 UITableViewCell 獲取方法
    static func dequeueReusableCell(_ tableView: UITableView) -> Self {
        return dequeueReusableCell(tableView, identifier: String(describing: self), style: .default)
    }
}

@objc public extension UITableViewHeaderFooterView{
    /// [源]自定義 UITableViewHeaderFooterView 獲取方法(兼容OC)
    static func dequeueReusableHeaderFooterView(_ tableView: UITableView, identifier: String) -> Self {
        var view = tableView.dequeueReusableHeaderFooterView(withIdentifier: identifier)
        if view == nil {
            view = self.init(reuseIdentifier: identifier)
        }
        return view as! Self;
    }
    
    /// [OC簡潔方法]自定義 UITableViewHeaderFooterView 獲取方法
    static func dequeueReusableHeaderFooterView(_ tableView: UITableView) -> Self {
        return dequeueReusableHeaderFooterView(tableView, identifier: String(describing: self))
    }
}

UICollectionViewCell 復(fù)用方法

@objc public extension UICollectionViewCell{
    
    /// [源]自定義 UICollectionViewCell 獲取方法(兼容OC)
    static func dequeueReusableCell(_ collectionView: UICollectionView, identifier: String, indexPath: IndexPath) -> Self {
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: identifier, for: indexPath)
        cell.backgroundColor = .white
        return cell as! Self
    }

    /// [OC簡潔方法]自定義 UITableViewCell 獲取方法
    static func dequeueReusableCell(_ collectionView: UICollectionView, indexPath: IndexPath) -> Self {
        return dequeueReusableCell(collectionView, identifier: String(describing: self), indexPath: indexPath)
    }
}

DateFormatter: 日期强重,字符串,時間戳的相互轉(zhuǎn)化

@objc public extension DateFormatter{
    /// 獲取DateFormatter(默認(rèn)格式)
    static func format(_ formatStr: String = kDateFormat) -> DateFormatter {
        let dic = Thread.current.threadDictionary;
        if let formatter = dic.object(forKey: formatStr) as? DateFormatter {
            return formatter
        }
        
        let fmt = DateFormatter();
        fmt.dateFormat = formatStr;
        fmt.locale = .current;
        fmt.locale = Locale(identifier: "zh_CN");
        fmt.timeZone = formatStr.contains("GMT") ? TimeZone(identifier: "GMT") : TimeZone.current;
        dic.setObject(fmt, forKey: formatStr as NSCopying)
        return fmt;
    }
    
    /// Date -> String
    static func stringFromDate(_ date: Date, fmt: String = kDateFormat) -> String {
        let formatter = DateFormatter.format(fmt);
        return formatter.string(from: date);
    }
    
    /// String -> Date
    static func dateFromString(_ dateStr: String, fmt: String = kDateFormat) -> Date? {
        let formatter = DateFormatter.format(fmt);
        let tmp = dateStr.count <= fmt.count ? dateStr : (dateStr as NSString).substring(to: fmt.count)
        let result = formatter.date(from: tmp);
        return result
    }
    
    /// 時間戳字符串 -> 日期字符串
    static func stringFromInterval(_ interval: String, fmt: String = kDateFormat) -> String {
        let date = Date(timeIntervalSince1970: interval.doubleValue)
        return DateFormatter.stringFromDate(date, fmt: fmt);
    }

    /// 日期字符串 -> 時間戳字符串
    static func intervalFromDateStr(_ dateStr: String, fmt: String = kDateFormat) -> String {
        guard let date = DateFormatter.dateFromString(dateStr, fmt: fmt) else {
            return "0" }
        return "\(date.timeIntervalSince1970)";
    }
}

Data

import UIKit

@objc public extension NSData{
    /// NSData -> 轉(zhuǎn)數(shù)組/字典
    var objValue: Any? {
        if JSONSerialization.isValidJSONObject(self) {
           return nil;
        }
        
        do {
            let obj: Any = try JSONSerialization.jsonObject(with: self as Data, options: [])
            return obj;
        } catch {
           print(error)
        }
        return nil;
    }
    /// NSData -> 轉(zhuǎn)字符串
    var stringValue: String {
        if let result = String(data: self as Data, encoding: .utf8) {
            return result
        }
        return ""
    }
    
    ///文件大小
    var fileSize: String {
        let length: CGFloat = CGFloat(self.length)
        
        if length < 1024.0 {
            return String(format: "%.2fB", length*1.0)
        }
        
        if length >= 1024.0 && length < (1024.0*1024.0) {
            return String(format: "%.2fKB", length/1024.0)
        }
                
        if length >= (1024.0*1024.0) && length < (1024.0*1024.0*1024.0) {
            return String(format: "%.2fMB", length/(1024.0*1024.0))
        }
        
        return String(format: "%.2fGB", length/(1024.0*1024.0*1024.0))
    }
}

public extension Data{
    /// 轉(zhuǎn)數(shù)組/字典
    var objValue: Any? {
        return (self as NSData).objValue;
    }
    /// NSData -> 轉(zhuǎn)字符串
    var stringValue: String? {
        return (self as NSData).stringValue;
    }
    
    /// NSData -> 轉(zhuǎn)字符串
    var fileSize: String {
        return (self as NSData).fileSize;
    }
}

Array

public extension Array{
    
    /// ->Data
    var jsonData: Data? {
        return (self as NSArray).jsonData;
    }
    
    /// ->String
    var jsonString: String {
        return (self as NSArray).jsonString;
    }
    ///同 subarrayWithRange:
    func subarray(_ range: NSRange) -> Array {
        return self.subarray(range.location, range.length)
    }
    ///同 subarrayWithRange:
    func subarray(_ loc: Int, _ len: Int) -> Array {
        assert((loc + len) <= self.count);
        return Array(self[loc...len]);
    }
    
}

@objc public extension NSArray{

    /// ->Data
    var jsonData: Data? {
        var data: Data?
        do {
            data = try JSONSerialization.data(withJSONObject: self, options: []);
        } catch {
            print(error)
        }
        return data;
    }
    
    /// ->NSString
    var jsonString: String {
        guard let jsonData = self.jsonData as Data?,
        let jsonString = String(data: jsonData, encoding: .utf8) as String?
        else { return "" }
        return jsonString
    }
}

Dictionary

import UIKit
public extension Dictionary{
    
    /// ->Data
    var jsonData: Data? {
        return (self as NSDictionary).jsonData;
    }
    /// ->NSString
    var jsonString: String {
        return (self as NSDictionary).jsonString;
    }
    var plistData: Data?{
        return (self as NSDictionary).plistData;
    }
}

public extension Dictionary where Key == String, Value == String {
    /// 鍵值翻轉(zhuǎn)
    var reversed: [String : String] {
        var dic = [String : String]()
        for (key, value) in self {
            dic[value] = key
        }
        return dic;
    }
    ///根據(jù)鍵數(shù)值排序
    func valuesByKeySorted() -> [String] {
        let values = keys.sorted {
//            return $0.intValue < $1.intValue
            return $0.compare($1) == .orderedAscending

        }.map { self[$0] ?? ""}
        return values
    }
}

@objc public extension NSDictionary{

    /// ->Data
    var jsonData: Data? {
        var data: Data?
        do {
            data = try JSONSerialization.data(withJSONObject: self, options: []);
        } catch {
            print(error)
        }
        return data;
    }
    
    /// ->NSString
    var jsonString: String {
        guard let jsonData = self.jsonData as Data?,
        let jsonString = String(data: jsonData, encoding: .utf8) as String?
        else { return "" }
        return jsonString
    }
    
    var plistData: Data?{
        let result = try? PropertyListSerialization.data(fromPropertyList: self, format: .binary, options: 0)
        return result
    }
}

URL

import UIKit

public extension URLComponents {
    
    ///查詢參數(shù)
    var queryParameters: [String: String]? {
        guard let queryItems = queryItems else { return nil }

        var dic = [String: String]()
        for item in queryItems {
            dic[item.name] = item.value
        }
        return dic
    }
    ///追加查詢參數(shù)
    mutating func appendingQueryParameters(_ parameters: [String: String]) -> URL {
        queryItems = (queryItems ?? []) + parameters
            .map { URLQueryItem(name: $0, value: $1) }
        return url!
    }
    
    ///查詢對應(yīng)參數(shù)值
    func queryValue(for key: String) -> String? {
        return queryItems?
            .first(where: { $0.name == key })?
            .value
    }
}


public extension URL {
    
    ///查詢參數(shù)
    var queryParameters: [String: String]? {
        guard let components = URLComponents(url: self, resolvingAgainstBaseURL: false)
        else { return nil }
        return components.queryParameters
    }
    ///追加查詢參數(shù)
    func appendingQueryParameters(_ parameters: [String: String]) -> URL {
        var components = URLComponents(url: self, resolvingAgainstBaseURL: true)!
        return components.appendingQueryParameters(parameters)
    }
    
    ///查詢對應(yīng)參數(shù)值
    func queryValue(for key: String) -> String? {
        return URLComponents(string: absoluteString)?.queryValue(for: key)
    }
}

@objc public extension NSURLComponents {
    
    ///查詢參數(shù)
    var queryParameters: [String: String]? {
        return (self as URLComponents).queryParameters
    }
    ///追加查詢參數(shù)
    func appendingQueryParameters(_ parameters: [String: String]) -> URL {
        queryItems = (queryItems ?? []) + parameters
            .map { URLQueryItem(name: $0, value: $1) }
        return url!
    }
    
    ///查詢對應(yīng)參數(shù)值
    func queryValue(for key: String) -> String? {
        return (self as URLComponents).queryValue(for: key)
    }
}

@objc public extension NSURL {
    
    ///查詢參數(shù)
    var queryParameters: [String: String]? {
        return (self as URL).queryParameters
    }
    ///追加查詢參數(shù)
    func appendingQueryParameters(_ parameters: [String: String]) -> URL {
        return (self as URL).appendingQueryParameters(parameters)
    }
    
    ///查詢對應(yīng)參數(shù)值
    func queryValue(for key: String) -> String? {
        return (self as URL).queryValue(for: key)
    }
}

UINavigationController

public extension UINavigationController{
    ///泛型方法: push到特定控制器頁面
    final func pushVC<T: UIViewController>(_ type: T.Type, animated: Bool = true, block: ((T) -> Void)? = nil) {
        let controller = type.init()
//        controller.hidesBottomBarWhenPushed = true
        block?(controller)
        pushViewController(controller, animated: animated);
    }
    ///泛型方法: pop到特定控制器頁面
    final func popToVC<T: UIViewController>(_ type: T.Type, animated: Bool = true) {
        for e in viewControllers {
            if e.isKind(of: type) {
                popToViewController(e, animated: animated)
                return
            }
        }
        popViewController(animated: animated)
    }
    ///泛型方法: 延遲退出
    func popVC(animated: Bool, delay: Double) {
        DispatchQueue.main.asyncAfter(deadline: DispatchTime(floatLiteral: delay)) {
            self.popViewController(animated: animated)
        }
    }
}

UISegmentedControl

@objc public extension UISegmentedControl{
    private struct AssociateKeys {
        static var items   = "UISegmentedControl" + "items"
    }

    /// 控件items
    var items: [String] {
        get {
            if let obj = objc_getAssociatedObject(self, &AssociateKeys.items) as? [String] {
                return obj
            }
            return []
        }
        set {
            objc_setAssociatedObject(self, &AssociateKeys.items, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC);
            updateItems(newValue)
        }
    }
    
    /// 配置新item數(shù)組
    private func updateItems(_ items: [String]) {
        if items.count == 0 {
            return
        }
        
        removeAllSegments()
        for e in items.enumerated() {
            insertSegment(withTitle: e.element, at: e.offset, animated: false)
        }
        selectedSegmentIndex = 0
    }
}

present

@objc public extension UIViewController{
    /// 呈現(xiàn)
    func present(_ animated: Bool = true, completion: (() -> Void)? = nil) {
        guard let keyWindow = UIApplication.shared.keyWindow,
              let rootVC = keyWindow.rootViewController
              else { return }
        self.modalPresentationStyle = .fullScreen

        DispatchQueue.main.async {
            if let alertVC = self as? UIAlertController {
                if alertVC.preferredStyle == .alert {
                    if alertVC.actions.count == 0 {
                        rootVC.present(alertVC, animated: animated, completion: {
                            DispatchQueue.main.after(TimeInterval(kDurationToast), execute: {
                                alertVC.dismiss(animated: animated, completion: completion)
                            })
                        })
                    } else {
                        rootVC.present(alertVC, animated: animated, completion: completion)
                    }
                } else {
                    //防止 ipad 下 sheet 會崩潰的問題
                    if UIDevice.current.userInterfaceIdiom == .pad {
                        if let popoverPresentationController = alertVC.popoverPresentationController {
                            popoverPresentationController.permittedArrowDirections = UIPopoverArrowDirection(rawValue: 0)
                            popoverPresentationController.sourceView = keyWindow;
                            
                            let isEmpty = popoverPresentationController.sourceRect.equalTo(.null) || popoverPresentationController.sourceRect.equalTo(.zero)
                            if isEmpty {
                                popoverPresentationController.sourceRect = CGRect(x: keyWindow.bounds.midX, y: 64, width: 1, height: 1);
                            }
                        }
                    }
                    rootVC.present(alertVC, animated: animated, completion: completion)
                }
            } else {
                rootVC.present(self, animated: animated, completion: completion)
            }
        }
    }
}

備注:
let list = [1,2,4,3,9,8];
let b = list[2...5];// ArraySlice
ArraySlice不能長期保存,可能會有內(nèi)存泄漏简僧,附上官方api的截圖
Important

Long-term storage of ArraySlice instances is discouraged. A slice holds a reference to the entire storage of a larger array, not just to the portion it presents, even after the original array’s lifetime ends. Long-term storage of a slice may therefore prolong the lifetime of elements that are no longer otherwise accessible, which can appear to be memory and object leakage.

警告:不鼓勵/建議長時間的存儲ArraySlice實例
由于一個ArraySlice實例呈現(xiàn)的是某個比較大的數(shù)組上的一個視圖扒秸。如果這個原始數(shù)組已經(jīng)結(jié)束了其生命周期恭应,存儲這個切片實例可能會延長那些不能再被訪問的數(shù)組元素的存活時間绘搞,這就造成了明顯的內(nèi)存和對象泄露够委。為了防止這個影響的發(fā)生,只在臨時性的計算時肃叶,使用ArraySlice蹂随。

//MARK:字典排序
func sortDictiony() -> Void {
    //字典排序
    /*
    字典中$0 來表示閉包的第一個參數(shù),$1 來表示第二個因惭,以此類推岳锁,in 也可以省略
    元組 **(key: String, value: String) 第0個是 key,第一個是 value **
    dict.sorted { (<#(key: String, value: String)#>, <#(key: String, value: String)#>) -> Bool in
         <#code#>
    }
    */
        
    let dict = ["27":"w","15":"t","36":"b"]
    let dicSortedByKey = dict.sorted(by: {$0.0 < $1.0})
    let dicSortedByValue = dict.sorted(by: {$0.1 < $1.1})
    print(dicSortedByKey)
    print(dicSortedByValue)
        
    let dicSortedByKeyNew = dict.sorted(by:<)
    let dicSortedByValueNew = dict.sorted(by:>)
    print(dicSortedByKeyNew)
    print(dicSortedByValueNew)
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蹦魔,一起剝皮案震驚了整個濱河市激率,隨后出現(xiàn)的幾起案子咳燕,更是在濱河造成了極大的恐慌,老刑警劉巖乒躺,帶你破解...
    沈念sama閱讀 219,427評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件招盲,死亡現(xiàn)場離奇詭異,居然都是意外死亡嘉冒,警方通過查閱死者的電腦和手機(jī)曹货,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來健爬,“玉大人控乾,你說我怎么就攤上這事么介∧茸瘢” “怎么了?”我有些...
    開封第一講書人閱讀 165,747評論 0 356
  • 文/不壞的土叔 我叫張陵壤短,是天一觀的道長设拟。 經(jīng)常有香客問我,道長久脯,這世上最難降的妖魔是什么纳胧? 我笑而不...
    開封第一講書人閱讀 58,939評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮帘撰,結(jié)果婚禮上跑慕,老公的妹妹穿的比我還像新娘。我一直安慰自己摧找,他們只是感情好核行,可當(dāng)我...
    茶點故事閱讀 67,955評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蹬耘,像睡著了一般芝雪。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上综苔,一...
    開封第一講書人閱讀 51,737評論 1 305
  • 那天惩系,我揣著相機(jī)與錄音,去河邊找鬼如筛。 笑死堡牡,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的杨刨。 我是一名探鬼主播晤柄,決...
    沈念sama閱讀 40,448評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼拭嫁!你這毒婦竟也來了可免?” 一聲冷哼從身側(cè)響起抓于,我...
    開封第一講書人閱讀 39,352評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎浇借,沒想到半個月后捉撮,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,834評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡妇垢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,992評論 3 338
  • 正文 我和宋清朗相戀三年巾遭,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闯估。...
    茶點故事閱讀 40,133評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡灼舍,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出涨薪,到底是詐尸還是另有隱情骑素,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評論 5 346
  • 正文 年R本政府宣布刚夺,位于F島的核電站献丑,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏侠姑。R本人自食惡果不足惜创橄,卻給世界環(huán)境...
    茶點故事閱讀 41,477評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望莽红。 院中可真熱鬧妥畏,春花似錦、人聲如沸安吁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽柳畔。三九已至馍管,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間薪韩,已是汗流浹背确沸。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留俘陷,地道東北人罗捎。 一個月前我還...
    沈念sama閱讀 48,398評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像拉盾,于是被迫代替她去往敵國和親桨菜。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,077評論 2 355