为啥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) {
        
    }

}
相关推荐
muyouking119 小时前
Tauri Android 开发踩坑实录:从 Gradle 版本冲突到离线构建成功
android·rust
哔哩哔哩技术9 小时前
B站多模态精细画质分析模型在 ICCV2025 大赛获得佳绩
音视频开发
Jerry11 小时前
Compose 为元素赋予动画特效
android
Jeled11 小时前
协程工具类
android·android studio
阿兰哥14 小时前
【调试篇5】TransactionTooLargeException 原理解析
android·性能优化·源码
爱吃水蜜桃的奥特曼15 小时前
玩Android Flutter版本,通过项目了解Flutter项目快速搭建开发
android·flutter
太过平凡的小蚂蚁16 小时前
Android 版本特性完全解析:从6.0到16.0的实用指南
android
杨筱毅16 小时前
【底层机制】【Android】深入理解UI体系与绘制机制
android·底层机制
介一安全16 小时前
【Frida Android】基础篇8:Java层Hook基础——调用带对象参数的方法
android·网络安全·逆向·安全性测试·frida
puyaCheer16 小时前
Android 13 启动的时候会显示一下logo,很不友好
android·gitee