news 2026/3/11 23:36:26

【Android】基于SurfaceControlViewHost实现跨进程渲染

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【Android】基于SurfaceControlViewHost实现跨进程渲染

1 前言

​ 本文将介绍基于 SurfaceControlViewHost 实现跨进程渲染普通 View 和 GlSurfaceView,力求用最简单的 Demo,介绍 SurfaceControlViewHost 的应用,方便读者轻松扣出核心代码应用到自己的业务中。

​ 核心代码片段如下。

​ 1)服务端

public SurfaceControlViewHost.SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height) {

// 创建SurfaceControlViewHost

Display display = mContext.getSystemService(DisplayManager.class).getDisplay(displayId);

mSurfaceControlViewHost = new SurfaceControlViewHost(mContext, display, hostToken);

// 创建要渲染的View

mView = new CustomView(mContext);

// 将View附加到SurfaceControlViewHost

mSurfaceControlViewHost.setView(mView, width, height);

SurfacePackage surfacePackage = mSurfaceControlViewHost.getSurfacePackage();

return surfacePackage;

}

​ 2)客户端

IBinder hostToken = mSurfaceView.getHostToken();

SurfaceControlViewHost.SurfacePackage surfacePackage = mRemoteRender.getSurfacePackage(0, hostToken, 1000, 2000);

mSurfaceView.setChildSurfacePackage(surfacePackage);

​ 本文案例项目结构如下,完整资源见 → 基于SurfaceControlViewHost实现跨进程渲染。

img

2 AIDL 配置

​ Android 跨进程通信可以使用 AIDL 或 messenger,它们本质都是 Binder,本文使用 AIDL 实现跨进程通信。

​ 1)aidl 文件

// IRemoteRender.aidl

package com.zhyan8.remoterender;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.os.IBinder;

interface IRemoteRender {

SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height);

}

​ 2)gradle 配置

sourceSets {

main {

aidl.srcDirs = ['src/main/aidl']

}

}

buildFeatures.aidl true

​ 3)manifest 配置

​ 客户端配置如下。

<queries>

<package android:name="com.zhyan8.service" />

<package android:name="com.zhyan8.glservice" />

</queries>

​ 服务端配置如下。

<service

android:name=".RemoteRenderService"

android:exported="true">

<intent-filter>

<action android:name="com.zhyan8.remoterender.IRemoteRender"/>

</intent-filter>

</service>

<service

android:name=".RemoteGLRenderService"

android:exported="true">

<intent-filter>

<action android:name="com.zhyan8.remoterender.IRemoteRender"/>

</intent-filter>

</service>

3 客户端

​ MainActivity.java

package com.zhyan8.client;

import android.content.ComponentName;

import android.content.Context;

import android.content.Intent;

import android.content.ServiceConnection;

import android.os.Bundle;

import android.os.IBinder;

import android.os.RemoteException;

import android.util.Log;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.view.SurfaceView;

import android.view.View;

import androidx.appcompat.app.AppCompatActivity;

import com.zhyan8.remoterender.IRemoteRender;

public class MainActivity extends AppCompatActivity {

private static final String TAG = "MainActivity";

private IRemoteRender mRemoteRender;

private IBinder mService;

private SurfaceView mSurfaceView;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mSurfaceView = findViewById(R.id.surface_view);

startService();

}

public void onClickDraw(View view) {

try {

IBinder hostToken = mSurfaceView.getHostToken();

SurfacePackage surfacePackage = mRemoteRender.getSurfacePackage(0, hostToken, 1000, 2000);

mSurfaceView.setChildSurfacePackage(surfacePackage);

} catch (RemoteException e) {

e.printStackTrace();

}

}

@Override

protected void onDestroy() {

super.onDestroy();

unbindService(mConnection);

}

private void startService() {

Log.d(TAG, "startService");

Intent intent = new Intent("com.zhyan8.remoterender.IRemoteRender");

//intent.setPackage("com.zhyan8.service"); // 渲染普通View的服务

intent.setPackage("com.zhyan8.glservice"); // 基于OpenGL ES渲染的服务

bindService(intent, mConnection, Context.BIND_AUTO_CREATE);

}

private void clearBind() {

Log.d(TAG, "clearBind");

if (mService != null) {

mService.unlinkToDeath(mDeathRecipient, 0);

}

mRemoteRender = null;

mService = null;

}

private ServiceConnection mConnection = new ServiceConnection() {

@Override

public void onServiceConnected(ComponentName name, IBinder service) {

Log.d(TAG, "onServiceConnected");

mRemoteRender = IRemoteRender.Stub.asInterface(service);

mService = service;

try {

mService.linkToDeath(mDeathRecipient, 0);

} catch (RemoteException e) {

Log.e(TAG, "e=" + e.getMessage());

}

}

@Override

public void onServiceDisconnected(ComponentName name) {

Log.d(TAG, "onServiceDisconnected");

clearBind();

}

};

private IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {

@Override

public void binderDied() {

Log.d(TAG, "binderDied");

clearBind();

}

};

}

​ activity_main.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical"

android:padding="16dp">

<Button

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="draw"

android:onClick="onClickDraw"/>

<android.view.SurfaceView

android:id="@+id/surface_view"

android:layout_width="1000px"

android:layout_height="2000px"

android:layout_gravity="center"/>

</LinearLayout>

4 跨进程渲染普通 View

​ RemoteRenderService.java

package com.zhyan8.service;

import android.app.Service;

import android.content.Context;

import android.content.Intent;

import android.hardware.display.DisplayManager;

import android.os.Handler;

import android.os.IBinder;

import android.os.Looper;

import android.util.Log;

import android.view.Display;

import android.view.SurfaceControlViewHost;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.view.ViewGroup;

import android.widget.ImageView;

import com.zhyan8.remoterender.IRemoteRender;

import java.util.concurrent.CountDownLatch;

public class RemoteRenderService extends Service {

private static final String TAG = "RemoteRenderService";

private SurfaceControlViewHost mSurfaceControlViewHost;

private ImageView mImageView;

private Handler mHandler = new Handler(Looper.getMainLooper());

@Override

public void onCreate() {

super.onCreate();

Log.i(TAG, "onCreate");

}

@Override

public IBinder onBind(Intent intent) {

Log.i(TAG, "onBind");

return mBinder;

}

@Override

public void onDestroy() {

super.onDestroy();

Log.i(TAG, "onDestroy");

if (mSurfaceControlViewHost != null) {

mSurfaceControlViewHost.release();

}

}

private final IRemoteRender.Stub mBinder = new IRemoteRender.Stub() {

@Override

public SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height) {

Log.i(TAG, "getSurfacePackage, displayId=" + displayId + ", hostToken=" + hostToken + ", width=" + width + ", height=" + height);

final SurfacePackage[] result = new SurfaceControlViewHost.SurfacePackage[1];

final CountDownLatch latch = new CountDownLatch(1);

mHandler.post( () -> {

// 创建SurfaceControlViewHost

Context context = getBaseContext();

Display display = context.getSystemService(DisplayManager.class).getDisplay(displayId);

mSurfaceControlViewHost = new SurfaceControlViewHost(context, display, hostToken);

// 创建要渲染的内容

mImageView = new ImageView(RemoteRenderService.this);

mImageView.setLayoutParams(new ViewGroup.LayoutParams(width, height));

mImageView.setScaleType(ImageView.ScaleType.FIT_XY);

mImageView.setImageResource(R.drawable.girl);

// 将视图附加到SurfaceControlViewHost

mSurfaceControlViewHost.setView(mImageView, width, height);

result[0] = mSurfaceControlViewHost.getSurfacePackage();

latch.countDown();

});

try {

latch.await(); // 等待主线程完成操作

return result[0];

} catch (InterruptedException e) {

Log.i(TAG, "getSurfacePackage, e=" + e.getMessage());

}

return null;

}

};

}

​ 运行效果如下。

img

5 跨进程渲染 GLSurfaceView

​ RemoteGLRenderService.java

package com.zhyan8.glservice;

import android.app.Service;

import android.content.Context;

import android.content.Intent;

import android.hardware.display.DisplayManager;

import android.opengl.GLSurfaceView;

import android.os.Handler;

import android.os.IBinder;

import android.os.Looper;

import android.util.Log;

import android.view.Display;

import android.view.SurfaceControlViewHost;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.view.ViewGroup;

import com.zhyan8.remoterender.IRemoteRender;

import java.util.concurrent.CountDownLatch;

public class RemoteGLRenderService extends Service {

private static final String TAG = "RemoteGLRenderService";

private SurfaceControlViewHost mSurfaceControlViewHost;

private GLSurfaceView mGLSurfaceView;

private Handler mHandler = new Handler(Looper.getMainLooper());

@Override

public void onCreate() {

super.onCreate();

Log.i(TAG, "onCreate");

}

@Override

public IBinder onBind(Intent intent) {

Log.i(TAG, "onBind");

return mBinder;

}

@Override

public void onDestroy() {

Log.i(TAG, "onDestroy");

super.onDestroy();

if (mSurfaceControlViewHost != null) {

mSurfaceControlViewHost.release();

}

}

private final IRemoteRender.Stub mBinder = new IRemoteRender.Stub() {

@Override

public SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height) {

Log.i(TAG, "getSurfacePackage, displayId=" + displayId + ", hostToken=" + hostToken + ", width=" + width + ", height=" + height);

final SurfacePackage[] result = new SurfaceControlViewHost.SurfacePackage[1];

final CountDownLatch latch = new CountDownLatch(1);

mHandler.post( () -> {

// 创建SurfaceControlViewHost

Context context = getBaseContext();

Display display = context.getSystemService(DisplayManager.class).getDisplay(displayId);

mSurfaceControlViewHost = new SurfaceControlViewHost(context, display, hostToken);

// 创建要渲染的内容

mGLSurfaceView = new GLSurfaceView(RemoteGLRenderService.this);

mGLSurfaceView.setEGLContextClientVersion(3);

mGLSurfaceView.setLayoutParams(new ViewGroup.LayoutParams(width, height));

mGLSurfaceView.setRenderer(new MyGLRenderer(RemoteGLRenderService.this));

// 将视图附加到SurfaceControlViewHost

mSurfaceControlViewHost.setView(mGLSurfaceView, width, height);

result[0] = mSurfaceControlViewHost.getSurfacePackage();

latch.countDown();

});

try {

latch.await(); // 等待主线程完成操作

return result[0];

} catch (InterruptedException e) {

Log.i(TAG, "getSurfacePackage, e=" + e.getMessage());

}

return null;

}

};

}

​ MyGLRenderer.java

package com.zhyan8.glservice;

import android.opengl.GLES30;

import android.opengl.GLSurfaceView;

import javax.microedition.khronos.egl.EGLConfig;

import javax.microedition.khronos.opengles.GL10;

import android.content.Context;

import java.nio.FloatBuffer;

public class MyGLRenderer implements GLSurfaceView.Renderer {

private FloatBuffer vertexBuffer;

private FloatBuffer textureBuffer;

private MyGLUtils mGLUtils;

private int mTextureId;

private int mTimeLocation;

private long mStartTime = 0L;

private long mRunTime = 0L;

public MyGLRenderer(Context context) {

mGLUtils = new MyGLUtils(context);

getFloatBuffer();

mStartTime = System.currentTimeMillis();

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/11 22:37:28

面试实战 问题三十四 对称加密 和 非对称加密 spring 拦截器 spring 过滤器

对称加密 和 非对称加密 对称加密 原理&#xff1a;对称加密是一种加密方法&#xff0c;使用相同的密钥进行加密和解密数据。加密过程是通过特定的加密算法&#xff0c;将明文数据按照密钥规则转换为密文&#xff1b;解密过程则是使用相同的密钥将密文还原为明文。这种加密方法…

作者头像 李华
网站建设 2026/3/4 10:41:13

零基础30分钟解锁Qwen3-4B-FP8:从部署到实战的完整能力获取指南

还在为AI模型部署的技术门槛而烦恼&#xff1f;Qwen3-4B-FP8作为高性能轻量级语言模型&#xff0c;仅需消费级GPU就能实现流畅推理&#xff0c;为个人开发者和中小企业提供低成本的AI解决方案。本文将带你从零开始&#xff0c;通过问题导向的递进式学习&#xff0c;快速掌握模型…

作者头像 李华
网站建设 2026/3/10 12:06:46

初探Kubernetes:核心概念解析

k8s 架构K8s 属于经典的主从模型&#xff08;Master-Slave 架构&#xff09;&#xff0c;由 Master 和 Node 节点构成&#xff1a;Master 节点&#xff1a;负责集群的管理&#xff0c;协调集群中的所有活动。例如应用的运行、修改、更新等。Node 节点&#xff1a;为 Kubernetes…

作者头像 李华
网站建设 2026/3/4 14:25:21

终极Visio形状库:5000+专业图形一键调用

终极Visio形状库&#xff1a;5000专业图形一键调用 【免费下载链接】史上最全Visio形状库分享 你是否在使用Microsoft Visio时&#xff0c;发现内置的形状库无法满足你的需求&#xff1f;你是否在寻找一个更全面、更丰富的形状库来提升你的绘图效率&#xff1f;那么&#xff0c…

作者头像 李华
网站建设 2026/3/9 11:37:19

MouseTester:专业鼠标性能评测工具终极指南

MouseTester&#xff1a;专业鼠标性能评测工具终极指南 【免费下载链接】MouseTester 项目地址: https://gitcode.com/gh_mirrors/mo/MouseTester 告别主观猜测&#xff0c;用数据精准评估鼠标表现 在日常使用中&#xff0c;你是否曾因鼠标响应延迟而在关键时刻错失良…

作者头像 李华