sql >> Databasteknik >  >> RDS >> Oracle

Hanterar stora JSON-data som returneras av webb-API

Ditt problem är att du kör en Oracle-fråga som returnerar ett mycket stort antal resultat och sedan laddar hela resultatuppsättningen i minnet innan du serialiserar den till HttpResponseMessage .

För att minska din minnesanvändning bör du hitta och eliminera alla fall där hela uppsättningen resultat från frågan läses in i en tillfällig mellanliggande representation (t.ex. en DataTable eller JSON-sträng), och istället strömma ut data med en DataReader . Detta undviker att dra in allt i minnet på en gång enligt det här svaret .

För det första, från din spårning, verkar det som att du har Aktivera webbläsarlänk kontrollerade. Eftersom detta uppenbarligen försöker cachelagra hela svaret i en MemoryStream , vill du inaktivera det enligt beskrivningen i FilePathResult kastade en OutOfMemoryException med stor fil .

Därefter kan du strömma innehållet i en IDataReader direkt till JSON med Json.NET med följande klass och omvandlare:

[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();
    }
}

Ändra sedan din kod så att den ser ut ungefär som:

    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;
    }

Detta undviker DataSet i minnet representation av resultaten.

För övrigt räknar jag med linjen

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

istället bör vara:

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

Jag tror att du försöker ta bort det avslutande kommatecken i frågan, som kommer att finnas om och bara om inconditions.Length > 0

Observera - jag är inte en Oracle-utvecklare och jag har inte Oracle installerat. För att testa hånade jag OracleClient klasser som använder en underliggande OleDbConnection och det fungerade bra.



  1. Hur man håller kontakten vid liv i java

  2. Kontrollera om ett objekt inte finns i en annan tabell

  3. letar efter en liten variant av GROUP BY

  4. Returnera en procentandel av en resultatuppsättning i SQL Server