t***@ejcms.com
t***@ejcms.com
  • 发布:2022-11-30 23:23
  • 更新:2023-04-23 11:10
  • 阅读:269

【报Bug】createInnerAudioContext 和 getBackgroundAudioManager 内存泄漏

分类:uni-app

产品分类: uniapp/App

PC开发环境操作系统: Windows

PC开发环境操作系统版本号: 10

HBuilderX类型: 正式

HBuilderX版本号: 3.6.10

手机系统: Android

手机系统版本号: Android 9.0

手机厂商: 哪吒汽车U

手机机型: 车机屏

页面类型: vue

vue版本: vue2

打包方式: 云端

项目创建方式: HBuilderX

示例代码:
template 代码
<view class="player">  
    <view class="progress">  
        <view class="current-time">{{ slider_time }}</view>  
        <view class="slider"><slider :value="slider_index" :max="song_time" activeColor="#0190ff" backgroundColor="#dddddd" block-color="#0190ff" block-size="12" /></view>  
        <view class="max-time">{{ slider_maxTime }}</view>  
    </view>  
    <view class="control">  
        <view class="col">  
            <view class="previous-btn" @click="changePrevious()"><view class="icon-previous"></view></view>  
        </view>  
        <view class="col">  
            <view class="outer" v-if="song_loading"></view>  
            <view class="inner" v-if="song_loading"></view>  
            <view class="play-btn" v-else @click="changePlayState()"><view :class="[song_playState ? 'icon-pause' : 'icon-play']"></view></view>  
        </view>  
        <view class="col">  
            <view class="next-btn" @click="changeNext()"><view class="icon-next"></view></view>  
        </view>  
    </view>  
</view>
script 代码
var innerAudioContext;  
export default {  
    data() {  
        return {  
            musicList: [  
                { song: '扬州三月', url: 'https://static.cdn.ejcms.com/1.mp3' },  
                { song: 'The Right Path', url: 'https://static.cdn.ejcms.com/2.mp3' },  
                { song: '故乡的原风景', url: 'https://static.cdn.ejcms.com/3.mp3' },  
                { song: '夏夜', url: 'https://static.cdn.ejcms.com/4.mp3' },  
                { song: 'Going Home', url: 'https://static.cdn.ejcms.com/5.mp3' }  
            ],  

            song_index: 0,  
            song_url: '',  
            song_loading: false,  
            song_playState: false,  
            song_time: 0,  

            slider_index: 0,  
            slider_time: '',  
            slider_maxTime: ''  
        };  
    },  
    onLoad() {  
        this.playAudio(this.musicList[0]);  
    },  
    methods: {  
        initPlay() {  
            let _this = this;  
            // 销毁当前实例  
            if (innerAudioContext) {  
                innerAudioContext.destroy();  
            }  

            // 创建音频  
            innerAudioContext = uni.createInnerAudioContext();  

            // 音频的数据链接,用于直接播放。  
            innerAudioContext.src = _this.song_url;  

            // 开始播放的位置(单位:s),默认 0  
            innerAudioContext.startTime = 0;  

            // 是否自动开始播放,默认 false  
            innerAudioContext.autoplay = true;  

            // 音频进入可以播放状态,但不保证后面可以流畅播放  
            innerAudioContext.onCanplay(res => {  
                _this.song_loading = false;  
            });  

            // 音频播放事件  
            innerAudioContext.onPlay(() => {  
                _this.song_playState = true;  
            });  

            // 音频暂停事件  
            innerAudioContext.onPause(() => {  
                _this.song_playState = false;  
            });  

            // 音频停止事件  
            innerAudioContext.onStop(res => {  
                innerAudioContext.destroy();  
                // console.log(res);  
            });  

            // 音频自然播放结束事件  
            innerAudioContext.onEnded(res => {  
                innerAudioContext.destroy();  
                _this.song_playState = false;  
                _this.changeNext();  
            });  

            // 音频播放进度更新事件  
            innerAudioContext.onTimeUpdate(() => {  
                let { currentTime, duration } = innerAudioContext;  

                _this.slider_index = currentTime;  
                _this.song_time = duration;  
                _this.slider_time = _this.formatTime(currentTime);  
                _this.slider_maxTime = _this.formatTime(duration);  
            });  

            // 音频播放错误事件  
            innerAudioContext.onError(res => {  
                _this.song_playState = false;  
                console.log('onError', res);  
            });  

            // 音频加载中事件,当音频因为数据不足,需要停下来加载时会触发  
            innerAudioContext.onWaiting(res => {  
                _this.song_loading = true;  
            });  
        },  
        // 上一曲  
        changePrevious() {  
            if (this.song_index > 0) {  
                this.song_index--;  
            } else {  
                this.song_index = 0;  
            }  
            this.playAudio(this.musicList[this.song_index]);  
        },  
        // 播放状态  
        changePlayState() {  
            if (this.song_playState) {  
                innerAudioContext.pause();  
            } else {  
                innerAudioContext.play();  
            }  
        },  
        // 下一曲  
        changeNext() {  
            if (this.musicList.length > this.song_index + 1) {  
                this.song_index++;  
            } else {  
                this.song_index = 0;  
            }  
            this.playAudio(this.musicList[this.song_index]);  
        },  
        playAudio(data) {  
            this.slider_index = 0;  
            this.song_url = data.url;  
            this.initPlay();  
        },  
        formatTime(num) {  
            num = num.toFixed(0);  
            let second = num % 60;  
            if (second < 10) {  
                second = '0' + second;  
            }  
            let min = Math.floor(num / 60);  
            if (min < 10) {  
                min = '0' + min;  
            }  
            return min + ':' + second;  
        }  
    }  
};
style 代码
uni-app,  
page {  
    background-image: linear-gradient(35deg, #b6e8fc, #ecccfb, #b6e8fc);  
}  
.player {  
    margin-top: calc(100vh - 150px);  
    width: 100%;  
    height: 150px;  
    background-image: linear-gradient(150deg, #e7f5ff, #ffffff, #f1eeff);  
    border-top-right-radius: 30px;  
    border-top-left-radius: 30px;  
    position: relative;  
    .progress {  
        height: 50px;  
        display: flex;  
        justify-content: space-between;  
        align-items: center;  
        .current-time {  
            width: 60px;  
            text-align: right;  
            color: #898989;  
        }  
        .slider {  
            width: calc(100% - 120px);  
        }  
        .max-time {  
            width: 60px;  
            text-align: left;  
            color: #898989;  
        }  
    }  
    .control {  
        margin: 0 auto;  
        width: 80%;  
        height: 80px;  
        .col {  
            float: left;  
            position: relative;  
            display: block;  
            box-sizing: border-box;  
            width: 33.3333336%;  
            height: 100px;  
            line-height: 100px;  
            display: flex;  
            justify-content: center;  
            .outer {  
                border: 5px solid rgba(0, 148, 255, 0.9);  
                opacity: 0.9;  
                width: 68px;  
                height: 68px;  
                border-top-color: transparent;  
                border-bottom-color: transparent;  
                border-radius: 50%;  
                animation: spinRight 0.5s linear infinite normal;  
                animation-delay: 0;  
                margin: 0 auto;  
            }  
            .inner {  
                border: 5px solid rgba(0, 148, 255, 0.9);  
                opacity: 0.9;  
                width: 38px;  
                height: 38px;  
                border-left-color: transparent;  
                border-bottom-color: transparent;  
                border-radius: 50%;  
                animation: spinLeft 0.5s linear infinite normal;  
                animation-delay: 0;  
                margin: 0 auto;  
                top: -63px;  
                display: block;  
                position: relative;  
            }  
            .play-btn,  
            .previous-btn,  
            .next-btn {  
                width: 68px;  
                height: 68px;  
                line-height: 68px;  
                border-radius: 50%;  
                display: flex;  
                align-items: center;  
                justify-content: center;  
            }  
            .play-btn {  
                background-color: #0190ff;  
            }  
        }  
    }  
}  
.icon-play {  
    width: 26px;  
    height: 26px;  
    background-image: url('');  
}  
.icon-pause {  
    width: 26px;  
    height: 26px;  
    background-image: url('');  
}  
.icon-previous {  
    width: 22px;  
    height: 22px;  
    background-image: url('');  
}  
.icon-next {  
    width: 22px;  
    height: 22px;  
    background-image: url('');  
}

操作步骤:

音频自动一首一首播放时

预期结果:

无内存泄漏

实际结果:

内存泄漏,小程序开发工具上崩溃重启;机车上直接卡死,任何按钮与导航都无反应

bug描述:

createInnerAudioContextgetBackgroundAudioManager都有会内存泄漏

createInnerAudioContext.destroy()也没有垃圾回收机制,

代码示例是从完整的代码中拷贝出来的,完整代码中使用到了storeStorage;代码示例情况也是一样,也会内存泄漏。

2022-11-30 23:23 负责人:DCloud_Android_zl 分享
已邀请:
7***@qq.com

7***@qq.com

您好 最后找到解决方案了没

2***@qq.com

2***@qq.com

我这也发现这问题了 切换的时候卡死更快 安卓手机。。。 有解决方案么?

要回复问题请先登录注册