为啥TextureView比SurfaceView表现还差呢?

从原理上面讲,我们大众的认知就是TextureView比SurfaceView的性能要好。硬的比软的好。但是其实这种是片面的。最近就遇到一个奇怪的现象:在3399上面通过ffmpeg拉rtsp流,然后通过mediacodec解码后渲染。渲染到TextureView上会比较频繁的出现马赛克的现象。但是换用SurfaceView立马就变好了。

以下3点是我用Raccoon小熊搜出来的答案。

TextureView 和 SurfaceView 都有各自的优势和局限性,所以它们的性能表现也会因应用的具体需求和使用场景而异。 在某些情况下,TextureView 的性能可能会比 SurfaceView 差,原因可能有以下几点:

  1. 渲染管道的差异:TextureView 是基于 OpenGL ES 的,它使用图形渲染管道来渲染内容。而 SurfaceView 则使用传统的 Android 渲染管道,这与 Android 的视图系统更加紧密集成。在某些情况下,这可能会导致 SurfaceView 的性能更好。

  2. 线程管理:SurfaceView 使用一个独立的线程来渲染内容,这可以提供更平滑的渲染性能,尤其是在处理复杂动画或游戏时。而 TextureView 则在主线程上渲染内容,这可能会导致性能下降,尤其是在处理大量数据或复杂渲染时。

  3. 硬件加速:虽然 TextureView 支持硬件加速,但在某些情况下,硬件加速可能会导致性能问题,尤其是在低端设备上。SurfaceView 则更多地依赖于软件渲染,这可能在某些情况下会提供更稳定的性能。

需要注意的是,性能差异可能会因设备和应用而异,因此在实际开发中应该根据具体需求和性能测试结果来选择合适的视图。无论选择哪种视图,都应该优化代码以提高性能,并确保在不同设备上进行充分的测试。

于是,我针对上面的3点的结论做了一个实验,在3399上面ffmpeg硬解码居然比软解码帧率要低。看来3399的CPU性能比其他硬件确实要抢。这就证明了标题中的疑惑了。

下面贴出一段出马赛克的代码,换上SurfaceView就好了。

java 复制代码
public class IPCameraPreviewFragment extends Fragment implements TextureView.SurfaceTextureListener{

    public static final String TAG = "IPCameraPreviewFragment";
	public static final boolean DEBUG = true;
	
	private TextureView mPreview;
	private SurfaceTexture mSurfaceTexture;
	private Handler mUiHandler = new Handler();
	private Runnable mRunnable = new Runnable() {
        
        @Override
        public void run() {
            if(mPreview == null || mSurfaceTexture == null) return;
            Play.getInstances().startPreivew(new Surface(mSurfaceTexture));
        }
    };
    private IErrorCallback mErrorCallback = new IErrorCallback() {
        
        @Override
        public void onError(int error) {
            Log.d(TAG, "onError = " + error);
            if(null == mUiHandler || null == mRunnable) return;
            mUiHandler.removeCallbacks(mRunnable);
            mUiHandler.postDelayed(mRunnable, 5000);
        }
    };
	
	public void setDataSource(String source){
	    Play.getInstances().setErrorCallback(mErrorCallback);
		Play.getInstances().setDataSource(source);
	}
	
	@Override
	public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
		View view = inflater.inflate(R.layout.preview_fragment, container,false);
		mPreview =  (TextureView)view.findViewById(R.id.preview);
		mPreview.setSurfaceTextureListener(this);
		return view;
	}
	
    @Override
    public void onSurfaceTextureAvailable(SurfaceTexture surface, int width,
            int height) {
        mSurfaceTexture = surface;
        Play.getInstances().startPreivew(new Surface(surface));
    }

    @Override
    public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width,
            int height) {
        
    }

    @Override
    public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
        mSurfaceTexture = null;
        Play.getInstances().releaseMediaPlay();
        if(null == mUiHandler || null == mRunnable) return false;
        mUiHandler.removeCallbacks(mRunnable);
        mUiHandler = null;
        mRunnable = null;
        return false;
    }

    @Override
    public void onSurfaceTextureUpdated(SurfaceTexture surface) {
        
    }

}
相关推荐
louisgeek28 分钟前
Java 线程池取消的方式
android
Billy_Zuo1 小时前
人工智能机器学习——模型评价及优化
android·人工智能·机器学习
tangweiguo030519871 小时前
Flutter与原生混合开发:实现完美的暗夜模式同步方案
android·flutter
雨白2 小时前
深入理解 Android 触摸事件:以实现 ViewPager 为例
android
shenshizhong2 小时前
看懂鸿蒙系统源码 比较重要的知识点
android·harmonyos
一只修仙的猿4 小时前
再谈性能优化,一次项目优化经历分享
android·性能优化
雮尘6 小时前
Android性能优化之枚举替代
android
2501_915909068 小时前
苹果上架App软件全流程指南:iOS 应用发布步骤、App Store 上架流程、uni-app 打包上传与审核技巧详解
android·ios·小程序·https·uni-app·iphone·webview
2501_915921438 小时前
iOS 文件管理与能耗调试结合实战 如何查看缓存文件、优化电池消耗、分析App使用记录(uni-app开发与性能优化必备指南)
android·ios·缓存·小程序·uni-app·iphone·webview
2501_915918419 小时前
App 苹果 上架全流程解析 iOS 应用发布步骤、App Store 上架流程
android·ios·小程序·https·uni-app·iphone·webview