欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Swift?中的?RegexBuilder學(xué)習(xí)指南

 更新時間:2023年04月07日 10:47:43   作者:馮志浩  
這篇文章主要為大家介紹了Swift中的RegexBuilder學(xué)習(xí)指南,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

前言

在我們?nèi)粘5捻椖块_發(fā)中,經(jīng)常會碰到和正則表達式打交道的時候。比如用戶密碼,通常會要求同時包含小寫字母、大寫字母、數(shù)字,并且長度不少于 8 位,以此來提高密碼的安全性。

在 Swift 中,我們可以用正則表達式的字面量方式來進行實現(xiàn)。

Regex 字面量

Regex 字面量實現(xiàn)代碼:

let regex = /^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)[a-zA-Z\d]{8,}$/
let text = "Aa11111111"
print(text.matches(of: regex).first?.output) // Optional("Aa11111111")

通過上述代碼可以看到,//通過兩個斜線就可以來生成正則的字面量。用字面量的方式確實可以使代碼很簡潔,但簡潔的代價就是很難看懂,對后面的代碼維護也造成了很大的困難。

就像網(wǎng)上盛傳的一句梗一樣:“我有一個問題,所以我寫了一個正則表達式?,F(xiàn)在,我有了兩個問題。”??

對于 Regex 難懂且難維護的問題,Swift 的開發(fā)團隊給出的方案就是:RegexBuilder。

RegexBuilder - 像寫代碼一樣寫正則

假設(shè)我們有一個字符串"name: John Appleseed, user_id: 100",想要提取其中user_id的值。 首先第一步,先導(dǎo)入 RegexBuilder:

import RegexBuilder

接著,通過結(jié)構(gòu)體 Regex 來構(gòu)建正則語句:

let regex = Regex {
    "user_id:" // 1
    OneOrMore(.whitespace) // 2
    Capture(.localizedInteger(locale: Locale(identifier: "zh-CN"))) // 3
}

第一行代碼匹配的是固定字符串:"user_id",第二行代碼匹配的是一個或者多個空格,第三行代碼則是匹配的整型數(shù)字。

localizedInteger 會將匹配到的數(shù)字自動轉(zhuǎn)為整型,比如下面的例子:

let input = "user_id:  100.11"
let regex = Regex {
    Capture(.localizedInteger(locale: Locale(identifier: "zh-CN")))
}
if let match = input.firstMatch(of: regex) {
    print("Matched: \(match.0)") // Matched:  100.11
    print("User ID: \(match.1)") // User ID: 100
}

雖然匹配的是 100.11,但輸出的仍然是 100。

最后,就可以通過 macth 的相關(guān)函數(shù)來進行數(shù)據(jù)提取了:

if let match = input.firstMatch(of: regex) {
    print("Matched: \(match.0)")
    print("User ID: \(match.1)")
}

RegexRepetitionBehavior

該結(jié)構(gòu)體是用來定義匹配的重復(fù)行為的,它有三個值:

  • edger:會盡可能多的去匹配輸入的字符,必要的時候會回溯。默認為edger
  • reluctant:會盡可能少的去匹配輸入的字符,它會根據(jù)你的需求來一點點增大匹配區(qū)域,以完成匹配。
  • possessive:會盡可能多的去匹配輸入的字符,不會回溯。

比如下面這個例子:

let testSuiteTestInputs = [    "2022-06-06 09:41:00.001",    "2022-06-06 09:41:00.001.",    "2022-06-06 09:41:00.001."]
let regex = Regex {
    Capture(OneOrMore(.any))
    Optionally(".")
}
for line in testSuiteTestInputs {
    if let (dateTime) = line.wholeMatch(of: regex)?.output {
        print("Matched: \(dateTime)\"")
    }
}

因為這三條數(shù)據(jù)最后的.是不一定有的,所以我們的正則有一個 Optionally(".")。但匹配出來的 dateTime 還是會帶 .。因為 edger 會匹配所有的字符包含最后的點在內(nèi),這樣 Optionally(".") 根本不會起作用。

改成 Capture(OneOrMore(.any, .reluctant))則會修復(fù)這個問題。因為 reluctant 它是匹配盡可能少的輸入,所以最后的Optionally(".")會執(zhí)行。

在 Swift 5.7 中,F(xiàn)oundation 框架也對 RegexBuilder 進行適配。所以對于 Date、URL等類型,我們可以借助 Foundation 的強大功能來進行解析。

Foundation 的支持

假如,我們在做一個金融相關(guān)的 APP,為了兼容一些老數(shù)據(jù),需要將一些字符串類型的數(shù)據(jù)轉(zhuǎn)為結(jié)構(gòu)體。

這是我們的字符串?dāng)?shù)據(jù):

let statement = """
CREDIT    2022/03/03    張三     ¥2,000,000.00
DEBIT     03/03/2022    Tom      $2,000,000.00
DEBIT

這是我們需要轉(zhuǎn)的結(jié)構(gòu)體:

struct Trade {
    let type: String
    let date: Date
    let name: String
    let count: Decimal
}

下面這個就是我們需要編寫的 Regex:

let regex = Regex {
    Capture {
        /CREDIT|DEBIT/
    }
    OneOrMore(.whitespace)
    Capture {
        One(.date(.numeric, locale: Locale(identifier: "zh_CN"), timeZone: .gmt))
    }
    OneOrMore(.whitespace)
    Capture {
        OneOrMore(.word)
    }
    OneOrMore(.whitespace)
    Capture {
        One(.localizedCurrency(code: "CNY", locale: Locale(identifier: "zh_CN")))
    }
}

首先,我們需要匹配固定的字符串:CREDIT/DEBIT,接著是匹配一個或者多個空格。

接下來就是 Foundation 的重頭戲了,對于日期類型的字符串,我們并不需要寫一些匹配年月日規(guī)則的正則,只需要借助 Foundation 內(nèi)嵌的功能即可。這樣做不僅省去了我們自己編寫的時間,更重要的是:官方寫的要比我們自己寫的更能保證代碼的正確性。

需要注意的是,Apple 推薦我們顯式的寫出 locale 屬性,而不是下面這種跟隨系統(tǒng)寫法 :

?

One(.date(.numeric, locale: Locale.current, timeZone: TimeZone.current))

因為這種寫法會帶來多種預(yù)期,并不能保證數(shù)據(jù)的確定性。

匹配完日期,接著就是對空格和用戶名的匹配。最后,是對交易金額的匹配,金額也是 Foundation 提供的函數(shù)來進行的匹配。

測試代碼:

let result = statement.matches(of: regex)
var trades = [Trade]()
result.forEach { match in
    let (_, type, date, name, count) = match.output
    trades.append(Trade(type: String(type), date: date, name: String(name), count: count))
}
print(trades) 
// [SwiftDemo.Trade(type: "CREDIT", date: 2022-03-03 00:00:00 +0000, name: "張三", count: 2000000), SwiftDemo.Trade(type: "DEBIT", date: 2022-03-05 00:00:00 +0000, name: "李三", count: 33.27)]

通過打印可以得知,輸出的結(jié)果并不符合預(yù)期,漏掉了 Tom 那條數(shù)據(jù)。漏掉的原因可以通過代碼一眼得知:因為對日期和金額我們顯式的指定了是中國的格式,顯然03/03/2022 這種格式是不符合年月日的格式的。這也體現(xiàn)了顯式指定格式的好處:方便排查問題。

我們只要將日期格式轉(zhuǎn)為年月日格式,再將 $ 轉(zhuǎn)為 ¥ 即可讓正則正確匹配。

首先,我們需要根據(jù) currency 來來返回正確的 Date 類型:

func pickStrategy(_ currency: Substring) -> Date.ParseStrategy {
  switch currency {
  case "$": return .date(.numeric, locale: Locale(identifier: "en_US"), timeZone: .gmt)
  case "¥": return .date(.numeric, locale: Locale(identifier: "zh_CN"), timeZone: .gmt)
  default: fatalError("We found another one!")
  }
}

接著,編寫正則表達式來獲取相應(yīng)的字符串字段:

let regex1 = #/
  (?<date>     \d{2} / \d{2} / \d{4})
  (?<name>   \P{currencySymbol}+)
  (?<currency> \p{currencySymbol})
/#

注:#//#格式為 Swift 中運行時正則表達式的格式。

最后,再調(diào)用 replace 函數(shù)來進行符合正則的字符替換:

statement.replace(regex1) { match -> String in
    print(match.currency)
    let date = try! Date(String(match.date), strategy: pickStrategy(match.currency))
    // ISO 8601, it's the only way to be sure
    let newDate = date.formatted(.iso8601.year().month().day())
    return newDate + match.name + "¥"
  }
statement = statement.replacingOccurrences(of: "-", with: "/")

這樣,我們就能解析出符合我們需求的 Trade 類型的數(shù)據(jù)了。

總結(jié)

  • RegexBuilder 會使代碼更加易讀易維護
  • RegexRepetitionBehavior 的三個值的區(qū)別
  • 盡可能多的使用 Foundation 提供的函數(shù)來解析數(shù)據(jù)
  • 使用 Foundation 時要指定格式解析數(shù)據(jù),這樣可以保證數(shù)據(jù)的唯一性

參考鏈接

以上就是Swift 中的 RegexBuilder學(xué)習(xí)指南的詳細內(nèi)容,更多關(guān)于Swift RegexBuilder的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • swift4 使用DrawerController實現(xiàn)側(cè)滑菜單功能的示例代碼

    swift4 使用DrawerController實現(xiàn)側(cè)滑菜單功能的示例代碼

    這篇文章主要介紹了swift4 使用DrawerController實現(xiàn)側(cè)滑功能的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-06-06
  • Swift3.0 GCD定時器的使用DEMO

    Swift3.0 GCD定時器的使用DEMO

    這篇文章主要介紹了Swift3.0 GCD定時器的使用,實現(xiàn)倒計時,UIDatePicker的使用, 仿寫一個活動倒計時的DEMO,需要的朋友可以參考下
    2017-02-02
  • Swift繪制漸變色的方法

    Swift繪制漸變色的方法

    這篇文章主要為大家詳細介紹了Swift繪制漸變色的方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • 簡單理解插入排序算法及Swift版的代碼示例

    簡單理解插入排序算法及Swift版的代碼示例

    插入排序算法可以在已排序的序列中將要插入的元素和原有元素保持有序,這里我們來簡單理解插入排序算法及Swift版的代碼示例,需要的朋友可以參考下
    2016-07-07
  • swift?cell自定義左滑手勢處理方法

    swift?cell自定義左滑手勢處理方法

    這篇文章主要介紹了swift?cell自定義左滑手勢處理,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • Swift縮放并填充圖片功能的實現(xiàn)

    Swift縮放并填充圖片功能的實現(xiàn)

    最近有一個需求,就是將圖片先等比例縮放到指定大小,然后將空余出來空間填充為黑色,返回指定大小的圖片。本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2021-11-11
  • SwiftUI使用Paths和AnimatableData實現(xiàn)酷炫的顏色切換動畫

    SwiftUI使用Paths和AnimatableData實現(xiàn)酷炫的顏色切換動畫

    這篇文章主要介紹了SwiftUI使用Paths和AnimatableData實現(xiàn)酷炫的顏色切換動畫,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2020-05-05
  • 深入解析Swift語言編程中的可選鏈

    深入解析Swift語言編程中的可選鏈

    這篇文章主要介紹了深入解析Swift語言編程中的可選鏈,是Swift入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下
    2015-11-11
  • Swift data范圍截取問題解決方案

    Swift data范圍截取問題解決方案

    這篇文章主要介紹了Swift data范圍截取問題解決方案,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-09-09
  • Swift學(xué)習(xí)筆記之構(gòu)造器重載

    Swift學(xué)習(xí)筆記之構(gòu)造器重載

    Swift 將為所有屬性已提供默認值的且自身沒有定義任何構(gòu)造器的結(jié)構(gòu)體或基類,提供一個默認的構(gòu)造器。這個默認構(gòu)造器將簡單的創(chuàng)建一個所有屬性值都設(shè)置為默認值的實例。
    2014-11-11

最新評論