Swift JSON文字列からenumへDecodeについて

Swift言語で、下記はシンプルなenumの書き方です。

enum MobileArmor {
    case bigro, zakrello
}

MobileArmorというenum(列挙型)を作成しました。このMobileArmorを変数に入れたい場合下記のように記述します。

let mobileArmor: MobileArmor = MobileArmor.zakrello

mobileArmor変数の中を.zakrelloに設定しています。この変数の状態によってswitch文で処理を変更したい場合下記のように書きます。

let mobileArmor: MobileArmor = .zakrello
switch mobileArmor {
case .zakrello:
    print("zakrello")
    break
case .bigro:
    print("bigro")
}

enumCaseIterableHashable, Codableなどをprotocolをつけることが可能です。

CaseIterableの場合AllCasesが使えるようになります。このAllCasesを使えば下記のコードのようにenumに設定している内容を確認することができます。

enum MobileArmor: CaseIterable {
    case bigro, zakrello
}
MobileArmor.allCases.forEach { print ( $0.self) }
型の内容をallCases.forEachで取得

enum型にHashableを設定した場合は、hashValuehash(into hasher: inout Hasher)が使えますがHashableを設定しなくてもhashValueなどは利用できるようでした。hash(into: &hasher)を設定するためのHasherは、任意のシーケンスを整数にマッピングするために使用できます。combine()とfinalize()をつかうようです。下記に例を書いてみますがあってるのか不明です。

enum MobileArmor: CaseIterable, Hashable {
    case bigro, zakrello
}
let mobileArmor: MobileArmor = .zakrello
var hasher1 : Hasher = Hasher()
hasher1.combine(128)
hasher1.finalize()
mobileArmor.hash(into: &hasher1)

Codableも他と同じく enumに設定できるprotocolです。Codableは下のコードのようにDecodableEncodableが合わさった機能を持っています。

public typealias Codable = Decodable  & Encodable

MobileArmor enum型にStringCodableを設定しました。Codableを指定しているので JSONのDecodeが可能です。

enum MobileArmor: String, Codable, CaseIterable, Hashable {
    case bigro = "ビグロ"
    case zakrello = "ザクレロ"
}

do {
    let amobileArmor = try JSONDecoder().decode(MobileArmor.self, from: "\"ザクレロ\"".data(using: .utf8)! )
    switch amobileArmor {
    case .zakrello:
        print("zakrello")
        break
    case .bigro:
        print("bigro")
    }
} catch { }

ザクレロ」という文字列をMobileArmor型にdecodeできました。JSONDecoderを使わない方法もあると思いますが、手っ取り早いコード例です。Codableが設定されているので、JSONEncoder()を使えば下のコードのように文字列に変換することもできます。

do {
    print( String(data: try JSONEncoder().encode(MobileArmor.bigro), encoding: .utf8)! )
} catch { }

Swift 文字列置換え replacingOccurrences

Swift replacingOccurrences

Swift言語で、C#のString.Replace()などの文字列置換えしたい場合 replacingOccurrences()が利用できます。replacingOccurrencesは、指定した文字部分すべてが別の指定された文字列で置き換えられ新しい文字列を返す関数です。下のコードは、Swift 5.2で試した文字列置換え例です。この関数は、置換えたい文字 ofと置換えたあとの文字withを引数として渡します。returnされる文字列はwithで指定した文字に置換わっています。

let s = String("ウルトラマン").replacingOccurrences(of: "マン", with: "セブン")
print (s)

上記コードは「ウルトラマン」の「マン」を「セブン」に置換えます。変数 sに入るのは「ウルトラセブン」です。

下記のURLではrangereplaceSubrangeを使った例です。

https://ameblo.jp/mojeld/entry-12593491674.html

SwiftUI Webサーバーから文字列取得する getHTTP

SwiftUIでボタンを押すとWebサーバーから文字列を取得する方法。

ObservableObjectを継承したクラスを作成する。

class TNetHTTPClient:ObservableObject {
    @Published var responseString: String
    init() {
        self.responseString = ""
    }
    func get(aurl: String) -> Void {
        guard let url1 = URL(string: aurl) else { return }
        URLSession.shared.dataTask(with: url1) { (data, res, _) in
            guard let data = data else { return }
            DispatchQueue.main.async {
                print(res ?? "")
                if let res = res as? HTTPURLResponse {
                    if res.statusCode == 200 {
                        self.responseString = " \( String(data: data, encoding: .utf8) ?? "" ) "
                    } else {
                        self.responseString = "status code = \(res.statusCode)"
                    }
                }
            }
        }.resume()
    }
}

Viewの構造体を作成しWebサーバーに接続するためのButtonと、HTTPレスポンスした文字列表示するためのTextFieldを配置します。

struct GetHttp: View {
    let http_url: String = "https://mjeld.com/moji.txt"
    @ObservedObject var netHTTPClient1 = TNetHTTPClient()
    var body: some View {
        VStack{
            Text("Get a string from the server.")
            TextField(self.http_url, text: self.$netHTTPClient1.responseString)
                .padding(.horizontal)
            HStack{
                Button("Button1", action: {
                    self.netHTTPClient1.get(aurl: self.http_url)
                })
                Button("Button2", action: {
                    self.netHTTPClient1.responseString = ""
                })
            }
            Spacer()
        }
    }
}

@ObservedObjectで先に作っていたクラスTNetHTTPClientを宣言します。TextFieldのtextは$netHTTPClient1.responseStringを設定しresponseStringが変更した場合TextFieldに反映されます。Button1をタップするとnetHTTPClient1.get(aurl: self.http_url)が実行されます。

URLからデータを取得しTextFieldに表示する
URLからデータを取得しTextFieldに表示する

実行し、Buttonを押すと上のようにWebサーバーの文字列を取得します。Webサーバー側にはUTF8のプレーンテキストファイルを置いています。

https://developer.apple.com/documentation/foundation/urlsession/1411554-datatask