Как отсортировать данные из JSON по дате и суммы, номинальной по дате в Swift? - PullRequest
0 голосов
/ 01 апреля 2019

Я хочу отсортировать данные из массива json по дате и по номинальному количеству данных, которые соответствуют дате.Я использую swiftyjson и alamofire.

вот мой код:

import UIKit
import Alamofire
import SwiftyJSON

    class HomeViewController: UIViewController{
          override func viewDidLoad() {
            super.viewDidLoad()
             fetchData()

             let createOrder = UserDefaults.standard.stringArray(forKey: "createOrder")
             let nominal = UserDefaults.standard.stringArray(forKey: "nominal")

             createOrder = createOrder?.sorted{ $0 < $1 } 
             print(createOrder?.description)
           }
         }

    func fetchData(){
            let url = ""

            Alamofire.request(url, method: .get, encoding: URLEncoding.default).responseJSON{
                (response) in
                switch response.result {

                case .success(let value):
                    let json = JSON(value)
                    print(json)

                    let jsonData = json["data"]["transaction"]

                    let data = jsonData.arrayValue.map{ $0["nominal"].string}
                    let createOrder = jsonData.arrayValue.map { $0["order_created"].string}

                    UserDefaults.standard.set(data, forKey: "nominal")
                    UserDefaults.standard.set(createOrder, forKey: "createOrder")

                case .failure(let error):
                    print(error)
                }
            }
        }

вот ответ JSON

"data" : {
    "transaction" : [

{
    "order_created" : "2019-03-30 14:39:05",
    "nominal" : "300000",
},
{
    "order_created" : "2019-03-30 11:26:08",
    "nominal" : "250000",
},
{
    "order_created" : "2019-03-29 10:49:44",
    "nominal" : "200000",
}
]

что было достигнуто:

Optional("[\"2019-03-30 10:49:44\", \"2019-03-30 11:26:08\", \"2019-03-30 14:39:05\"]")

я получаю данные на 2019-03-30: 2 номинала 300000 и 250000

в 2019-03-29 я получаю 1 номинал 200000

я хочу общее номинальное значение на 2019-03-30 - 300000 + 250000 = 550000

, поэтому я должен получить результат: на 2019-03-30 общее число составляет 500000, а общее число 2019-03-29 составляет 200000

, пожалуйста, помогите мне

Ответы [ 2 ]

0 голосов
/ 01 апреля 2019

Вы должны использовать Codable вместо SwiftyJSON, так как он напечатан более явно, но, отвечая на ваш вопрос непосредственно в отношении SwiftyJSON, вы могли бы пойти с этим:

func processedTransactions(_ transactions: JSON) -> [(date: String, nominal: Double)] {
    let processedTransactions = transactions
        .arrayValue
        .reduce(into: [String: Double]()) { (r, t) in
            if let date = t["order_created"].stringValue.split(separator: " ").first {
                let date = String(date)
                r[date] = r[date, default: 0] + (Double(t["nominal"].stringValue) ?? 0)
            }
        }
        .sorted { $0.key > $1.key }
        .map { (date: $0.key, nominal: $0.value) }

    return processedTransactions
}

func handleProcessedTransactions(_ transactions: [(date: String, nominal: Double)]) {
    //do something with your sorted transactions
    for t in transactions {
        print(t.date, t.nominal)
    }
}

Здесь, в processedTransactions(_:), мы в основном перебираем даты и готовим уникальные записи, основанные только на части YYYY-DD-MM.
Одновременно мы ведем учет общей суммы номинальной стоимости, относящейся к определенной дате.
Это сделано в части reduce ( Логика, которую я использовал, является базовой манипуляцией со строками, но можно утверждать, что это ошибка, поэтому не стесняйтесь переписать эту часть )
Затем мы sort это и просто для лучшей читаемости, мы map воссоздаем объект как кортежи (date: String, nominal: String)
Затем мы можем передать это в handleProcessedTransactions(_:) и задействовать на нем требуемую логику.


Кроме того, вы, похоже, используете UserDefaults, поэтому вот как может выглядеть существующая часть вашего кода:

override func viewDidLoad() {
    super.viewDidLoad()
    fetchData()

    let json = UserDefaults.standard.object(forKey: "transactions")
    let transactions = processedTransactions(JSON(json))
    handleProcessedTransactions(transactions)
}

func fetchData() {
    let url = ""
    Alamofire
        .request(url, method: .get, encoding: JSONEncoding.default)
        .responseJSON { (response) in
            switch response.result {
            case .success(let value):
                let json = JSON(value)

                let transactionJSON = json["data"]["transaction"]
                UserDefaults.standard.set(transactionJSON.arrayObject, forKey: "transactions")

                //you probably want to handle it here too
                let transactions = self.processedTransactions(transactionJSON)
                self.handleProcessedTransactions(transactions)
            case .failure(let error):
                print(error)
            }
    }
}
0 голосов
/ 01 апреля 2019

Первый Почему вы используете SwiftyJSON и там встроено Codable protocol? Учебник

Второй : Идея состоит в том, чтобы сгруппировать ваши данные по yyyy-MM-dd, как это 2019-03-30, это потребует дополнительной работы. я создаю декодируемую версию для вашего ответа вы можете проверить

import Foundation

struct JsonData: Decodable {
    let data: Transactions
}
struct Transactions: Decodable {
    let transaction: [Transaction]
}

struct Transaction: Decodable {
    let nominal: String
    let orderCreated : Date

    enum CodingKeys: String, CodingKey {
        case orderCreated = "order_created"
        case nominal
    }
}

extension JsonData {
    init(data: Data) throws {
        self = try DatedDecoder().decode(JsonData.self, from: data)
    }
}

fileprivate func DatedDecoder() -> JSONDecoder {
    let decoder = JSONDecoder()
    decoder.dateDecodingStrategy = .formatted(DateFormatter.yyyyMMddHHmmss)
    return decoder
}


extension DateFormatter {


    // for modeling json data
    static let yyyyMMddHHmmss : DateFormatter = {
        let dateFormater = DateFormatter()
        dateFormater.dateFormat = "yyyy-MM-dd HH:mm:ss"
        dateFormater.timeZone = TimeZone(abbreviation: "UTC")
        dateFormater.locale = Locale(identifier: "en_US_POSIX")
        return dateFormater
    }()

    // for grouping
    static let yyyyMMdd: DateFormatter = {
        let dateFormater = DateFormatter()
        dateFormater.dateFormat = "yyyy-MM-dd"
        dateFormater.timeZone = TimeZone(abbreviation: "UTC")
        dateFormater.locale = Locale(identifier: "en_US_POSIX")
        return dateFormater
    }()
}

И вы можете использовать это

 // (data) is response of your API call

 if let jsonData = try? JsonData(data: data){

            let predicate: (Transaction) -> String = { transaction in
                return DateFormatter.yyyyMMdd.string(from: transaction.orderCreated)
            }

            let groupingByDate : [String:[Transaction]] = Dictionary(grouping: jsonData.data.transaction, by: predicate)

            let items =  groupingByDate.map { item -> (date:String , value:Int) in
               return (item.key , item.value.map({Int(String($0.nominal))!}).reduce(0, +))
            }
            //  Final Items array is tuple with date and total nominal  [(date: "2019-03-30", value: 550000), (date: "2019-03-29", value: 200000)
            print(items)
        }

Ответ Json API для вышеуказанного решения

{
    "data" : {
        "transaction" : [

            {
                "order_created" : "2019-03-30 14:39:05",
                "nominal" : "300000"
            },
            {
                "order_created" : "2019-03-30 11:26:08",
                "nominal" : "250000"
            },
            {
                "order_created" : "2019-03-29 10:49:44",
                "nominal" : "200000"
            }
        ]
    }
}
...