sql >> Databáze >  >> RDS >> Oracle

Práce s velkými daty JSON vrácenými webovým rozhraním API

Váš problém spočívá v tom, že spouštíte dotaz Oracle, který vrací velmi velký počet výsledků, a poté načtete celou sadu výsledků do paměti, než ji serializujete do HttpResponseMessage .

Chcete-li snížit využití paměti, měli byste najít a odstranit všechny případy, kdy je celá sada výsledků z dotazu načtena do dočasné přechodné reprezentace (např. DataTable nebo řetězec JSON) a místo toho streamujte data pomocí DataReader . Tím se zabrání stahování všeho do paměti najednou podle tato odpověď .

Za prvé, z vašeho sledování se zdá, že máte Povolit odkaz na prohlížeč kontrolovány. Protože se to zjevně pokouší uložit do mezipaměti celou odpověď v MemoryStream , budete jej chtít deaktivovat, jak je vysvětleno v FilePathResult vyvolal výjimku OutOfMemoryException s velkým souborem .

Dále můžete streamovat obsah IDataReader přímo do JSON pomocí Json.NET s následující třídou a převodníkem:

[JsonConverter(typeof(OracleDataTableJsonResponseConverter))]
public sealed class OracleDataTableJsonResponse
{
    public string ConnectionString { get; private set; }
    public string QueryString { get; private set; }
    public OracleParameter[] Parameters { get; private set; }

    public OracleDataTableJsonResponse(string connStr, string strQuery, OracleParameter[] prms)
    {
        this.ConnectionString = connStr;
        this.QueryString = strQuery;
        this.Parameters = prms;
    }
}

class OracleDataTableJsonResponseConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(OracleDataTableJsonResponse);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException("OracleDataTableJsonResponse is only for writing JSON.  To read, deserialize into a DataTable");
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var response = (OracleDataTableJsonResponse)value;

        using (var dbconn = new OracleConnection(response.ConnectionString))
        {
            dbconn.Open();
            using (var selectCommand = new OracleCommand(response.QueryString, dbconn))
            {
                if (response.Parameters != null)
                    selectCommand.Parameters.AddRange(response.Parameters);
                using (var reader = selectCommand.ExecuteReader())
                {
                    writer.WriteDataTable(reader, serializer);
                }
            }
        }
    }
}

public static class JsonExtensions
{
    public static void WriteDataTable(this JsonWriter writer, IDataReader reader, JsonSerializer serializer)
    {
        if (writer == null || reader == null || serializer == null)
            throw new ArgumentNullException();
        writer.WriteStartArray();
        while (reader.Read())
        {
            writer.WriteStartObject();
            for (int i = 0; i < reader.FieldCount; i++)
            {
                writer.WritePropertyName(reader.GetName(i));
                serializer.Serialize(writer, reader[i]);
            }
            writer.WriteEndObject();
        }
        writer.WriteEndArray();
    }
}

Poté upravte svůj kód tak, aby vypadal nějak takto:

    public HttpResponseMessage Getdetails([FromUri] string[] id)
    {
        var prms = new List<OracleParameter>();
        var connStr = ConfigurationManager.ConnectionStrings["PDataConnection"].ConnectionString;
        var inconditions = id.Distinct().ToArray();
        var strQuery = @"SELECT 
                       STCD_PRIO_CATEGORY_DESCR.DESCR AS CATEGORY, 
                       STCD_PRIO_CATEGORY_DESCR.SESSION_NUM AS SESSION_NUMBER, 
                       Trunc(STCD_PRIO_CATEGORY_DESCR.START_DATE) AS SESSION_START_DATE, 
                       STCD_PRIO_CATEGORY_DESCR.START_DATE AS SESSION_START_TIME , 
                       Trunc(STCD_PRIO_CATEGORY_DESCR.END_DATE) AS SESSION_END_DATE, 
                         FROM 
                         STCD_PRIO_CATEGORY_DESCR, 
                         WHERE 
                        STCD_PRIO_CATEGORY_DESCR.STD_REF IN(";
        var sb = new StringBuilder(strQuery);
        for (int x = 0; x < inconditions.Length; x++)
        {
            sb.Append(":p" + x + ",");
            var p = new OracleParameter(":p" + x, OracleDbType.NVarchar2);
            p.Value = inconditions[x];
            prms.Add(p);
        }
        if (sb.Length > 0)// Should this be inconditions.Length > 0  ?
            sb.Length--;
        strQuery = sb.Append(")").ToString();

        var returnObject = new { data = new OracleDataTableJsonResponse(connStr, strQuery, prms.ToArray()) };
        var response = Request.CreateResponse(HttpStatusCode.OK, returnObject, MediaTypeHeaderValue.Parse("application/json"));
        ContentDispositionHeaderValue contentDisposition = null;
        if (ContentDispositionHeaderValue.TryParse("inline; filename=ProvantisStudyData.json", out contentDisposition))
        {
            response.Content.Headers.ContentDisposition = contentDisposition;
        }
        return response;
    }

Tím se vyhnete uložení DataSet v paměti reprezentace výsledků.

Mimochodem počítám čár

        if (sb.Length > 0)
            sb.Length--;

místo toho by mělo být:

        if (inconditions.Length > 0)
            sb.Length--;

Domnívám se, že se pokoušíte odloupnout koncovou čárku v dotazu, která bude přítomna právě tehdy, když inconditions.Length > 0

Upozornění - nejsem vývojář Oracle a nemám nainstalovaný Oracle. Pro testování jsem si vymodeloval OracleClient třídy pomocí základního OleDbConnection a fungovalo to dobře.



  1. DNA vs moderní metody zálohování:Budoucnost ukládání dat

  2. Jak najít nejprohranější za 1 den, 1 měsíc na základě historických dat

  3. Ukládání přihlašovacích údajů k MS SQL Serveru v databázi MySQL

  4. Pokus o uložení proměnných do databázové tabulky MYSQL pomocí obrázků