티스토리 뷰
https://www.test.com/users?gender=male&age=20&job=designer
위와 같은 get 요청에서
gender=male
age=20
job=designer
같은 파라미터를 구하려면 어떻게 해야할까요?!?
저는 정규표현식을 이용했습니다.
Swift에서 정규표현식(Regular Expression)을 이용하기 포스팅에 차마 이어쓰지 못하겠어서(😱) 새롭게 씁니다.
하지만 자세한 설명은 저기에 있습니다. 여기에는 간략하게!
결론
이렇게 extension을 추가하고
extension String { | |
func matchingKeywords(regexPattern: String) -> [String] { | |
if self.isEmpty { | |
return [] | |
} | |
do { | |
let regex = try NSRegularExpression(pattern: regexPattern) | |
let results = regex.matches(in: self, range: NSRange(location: 0, length: self.count)) | |
let keywords = results | |
.compactMap { Range($0.range, in: self) } | |
.map { String(self[$0]) } | |
return keywords | |
} catch { | |
return [] | |
} | |
} | |
} |
정규표현식 패턴으로 아래의 것을 쓰면 됩니다.
[^&?]+=[^&?]+
테스트를 돌리면 성공하는 것을 볼 수 있습니다.
func test_find_get_parameters() { | |
// given | |
let sampleURL = "https://www.test.com/users?gender=male&age=20&job=designer" | |
// when | |
let texts = sampleURL.matchingKeywords(regexPattern: "[^&?]+=[^&?]+") | |
// then | |
XCTAssertEqual(texts.count, 3) | |
XCTAssertEqual(texts[0], "gender=male") | |
XCTAssertEqual(texts[1], "age=20") | |
XCTAssertEqual(texts[2], "job=designer") | |
} |
과정
저렇게 테스트가 성공할 수 있는 과정을 살펴보겠습니다.
우선
[^?&] 이거 부터 살펴보자면
대괄호[]는 한 문자를 의미하고
대괄호 안에 ^를 쓰면 부정(not)의 의미를 가집니다.
즉 [^?&]는 ?랑 &가 아닌 한 문자를 의미합니다.
그래서 이 사이트에서 테스트 해보면 (참고로 이 사이트는 swift 랭기지 설정을 할 수 없습니다)
&랑 ?가 아닌 한문자들로 총 56개가 매칭되는 것을 볼 수 있습니다.

그 다음, [^?&]+를 살펴보겠습니다.
+는 정규표현식에서 1개~여러개를 의미합니다.
?랑 &가 아닌 1개~여러개 문자들이 선택되는 것을 볼 수 있습니다. 총 4개가 매칭된다고 나옵니다

그 다음, [^&?]+=[^&?]+ 를 살펴보겠습니다.
"?랑 &가 아닌 1개~여러개 문자들 = ?랑 &가 아닌 1개~여러개 문자들"
이렇게 된 패턴을 찾는 것입니다.
총 3개가 잘 매칭되는 것을 볼 수 있습니다.

추가로 이렇게 딕셔너리로 만들어서 value값을 꺼내는 식으로 활용할 수 있을 것 같네요 :-)
func test_find_get_parameters() { | |
// given | |
let sampleURL = "https://www.test.com/users?gender=male&age=20&job=designer" | |
// when | |
let texts = sampleURL.matchingKeywords(regexPattern: "[^&?]+=[^&?]+") | |
// then | |
XCTAssertEqual(texts.count, 3) | |
XCTAssertEqual(texts[0], "gender=male") | |
XCTAssertEqual(texts[1], "age=20") | |
XCTAssertEqual(texts[2], "job=designer") | |
var dic: [String: String] = [:] | |
texts.forEach { | |
let array = $0.split(separator: "=").map { String($0) } | |
if array.count == 2 { | |
dic[array[0]] = array[1] | |
} | |
} | |
XCTAssertEqual(dic["gender"], "male") | |
XCTAssertEqual(dic["age"], "20") | |
XCTAssertEqual(dic["job"], "designer") | |
} |
[ 추가 ]
정규표현식말고 이런 extension 만들어서 사용할 수도 있어요-!!
extension URL {
public var parameters: [String: Any?] {
var dic: [String: Any?] = [:]
guard let components = URLComponents(url: self, resolvingAgainstBaseURL: false) else { return dic }
guard let queryItems = components.queryItems else { return dic }
for item in queryItems {
dic[item.name] = item.value
}
return dic
}
}
'🍏 > Swift' 카테고리의 다른 글
[Swift] if case 예제 (0) | 2021.01.20 |
---|---|
[Swift] 전처리문 (#if DEBUG, #if os) (1) | 2020.08.09 |
[Date] Swift에서 Date를 이용하는 여러 케이스들 모음 (0) | 2020.02.01 |
[Enum] Enum에서 불가능한 상황들(?) (0) | 2019.06.06 |
[Swift] Swift5의 Result Type 사용하기 (3) | 2019.04.29 |
- Total
- Today
- Yesterday
- 장고 URL querystring
- Flutter Spacer
- 구글 Geocoding API
- flutter build mode
- Flutter 로딩
- ipad multitasking
- cocoapod
- Flutter getter setter
- SerializerMethodField
- Sketch 누끼
- 플러터 얼럿
- 장고 Custom Management Command
- Dart Factory
- Django FCM
- flutter dynamic link
- Flutter Text Gradient
- METAL
- PencilKit
- Django Heroku Scheduler
- Watch App for iOS App vs Watch App
- ribs
- Python Type Hint
- drf custom error
- Flutter Clipboard
- flutter 앱 출시
- flutter deep link
- github actions
- 플러터 싱글톤
- Django Firebase Cloud Messaging
- DRF APIException
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |