繁体   English   中英

如何在无休止的流式请求中释放responseText的内存?

[英]How to release memory of the responseText in an endless streaming request?

我需要使用xmlHTTPRequest从无尽的运动JPEG数据流中获取数据,并且我听说一个不错的技巧,即responseText可以在不完成请求的情况下填充数据(它永远不会完成)。 现在我遇到了问题。

我的请求是这样绑定与RxJS可观察的。

  postActionGivenDataBinaryStream(url:string, data:string) : Observable<any>{

    this.binary_progress_call_accum = 0 ;
    this.previous_length = 0 ;

    return Observable.create((observer)=>{
        let xhr = new XMLHttpRequest() ;

        xhr.open('POST',url,true) ;
        xhr.setRequestHeader("Content-Type","application/json;charset=utf-8");
        //this way the binary data keeps populate at state 3
        xhr.overrideMimeType('text\/plain; charset=x-user-defined');

        xhr.onreadystatechange = () =>{
            if (xhr.readyState === 4) {
                if (xhr.status === 200) {
                    this.binary_progress_call_accum = 0 ;
                    this.previous_length = 0 ;
                    observer.complete();
                } else {
                    this.binary_progress_call_accum = 0 ;
                    this.previous_length = 0 ;
                    observer.error(xhr.response);
                }
            }
        }
        xhr.onprogress = (event)=>{
            let outstring:string = "";
            //should preordically emit the response text 

            if (this.binary_progress_call_accum > 1) {
                //just send the next piece of data
                let endCount = xhr.responseText.length ;

                //so here try to next a string

                for (let i = this.previous_length ; i < endCount; ++i ){
                    outstring += ("00" + (xhr.responseText.charCodeAt(i) & 0xFF).toString(16)).slice(-2) ;
                }

                observer.next(outstring) ;
                this.previous_length = endCount ;
            }
            else{
                for (let i = 0 ; i < xhr.responseText.length; ++i ){
                    outstring += ("00" + (xhr.responseText.charCodeAt(i) & 0xFF).toString(16)).slice(-2) ;
                }
                observer.next(outstring) ;
                this.previous_length = xhr.responseText.length ;
            }

            this.binary_progress_call_accum += 1;
        };

        xhr.send(data) ;

        //https://stackoverflow.com/a/38622923/921082
        //elegantly abort()
        return () => xhr.abort() ;
    }) ;

}

但是这段代码有严重的问题,我以30秒的间隔将此时间称为可观察到的时间,但是有时会卡住。 当我的间隔可观察到触发时,请求仅延迟30秒! 如果不执行任何操作,则控制台日志将不提供任何内容。

我怀疑这是xhr.abort()完成的时间太长,在完成之前,可观察到的间隔给出了下一个请求,这将与前一个请求发生冲突,从而导致请求-响应延迟。 因此,有什么方法可以释放responseText的内存, responseText无需使用间隔来重新初始化可观察到的xhr请求?

这似乎很困难,我实际上已经解决了。

答案与上面的代码无关,但是代码将该请求视为可观察到的。

我应该使用repeatWhen而不是IntervalObservable 由于IntervalObservable不在乎请求是否真正完成,因此它将导致请求在最后一个请求完成之前触发。 repeatWhen可确保在下一次启动之前最后一个可观测值实际上已完成。

请检查此启发性讨论, https://github.com/ReactiveX/RxJava/issues/448

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM