news 2026/6/23 1:30:23

【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/6/23 7:40:27

【EF Core】通过 DbContext 选项扩展框架

本来老周计划在 10 月 1 日或 2 日写这篇水文的&#xff0c;没打算出去玩&#xff08;确实没啥好玩&#xff09;。不过因为买的运动相机到手&#xff0c;急着想试试效果&#xff0c;于是就备了些干粮&#xff0c;骑着山地车在外面鬼混了一天。10 月 2 日&#xff0c;家里来了三…

作者头像 李华
网站建设 2026/6/23 0:43:36

新用户免费试用EmotiVoice 1000个token

EmotiVoice&#xff1a;用1000个免费Token开启高表现力语音合成之旅 在虚拟主播的直播间里&#xff0c;一句“太开心了&#xff01;”如果只是平平无奇地念出来&#xff0c;观众很难被感染&#xff1b;而在智能助手中&#xff0c;当用户情绪低落时&#xff0c;机械冷漠的回应只…

作者头像 李华
网站建设 2026/6/15 13:57:35

免费视频增强神器:3步将模糊视频升级4K超清画质

免费视频增强神器&#xff1a;3步将模糊视频升级4K超清画质 【免费下载链接】SeedVR-7B 项目地址: https://ai.gitcode.com/hf_mirrors/ByteDance-Seed/SeedVR-7B 想要让那些模糊的家庭录像、珍贵回忆重获新生吗&#xff1f;字节跳动SeedVR视频增强工具为你带来专业级的…

作者头像 李华
网站建设 2026/6/23 18:41:36

dp 总结 1

shout out to professor Adzlpxsn.upd at oct 16th 2025, 修复了时间复杂度分析的重大失误.基本的, 状态, 转移, 方程状态一句话概况即为当前的属性.比如说, 贝贝现在是 3030 岁, 发了 00 张专辑, 我们就可以说 &#xfffd;300f 30​0.这里我们说 3030 和 00 是不同的信息, 所…

作者头像 李华
网站建设 2026/6/23 18:36:08

5大核心参数精准调优:从理论到实践的Faiss HNSW索引优化指南

5大核心参数精准调优&#xff1a;从理论到实践的Faiss HNSW索引优化指南 【免费下载链接】faiss A library for efficient similarity search and clustering of dense vectors. 项目地址: https://gitcode.com/GitHub_Trending/fa/faiss 面对海量向量数据的检索挑战&am…

作者头像 李华
网站建设 2026/6/23 15:22:20

LeetCode 最小覆盖子串:滑动窗口 + 哈希表高效解法

引言&#xff1a;为什么这道题是算法面试高频题&#xff1f;“最小覆盖子串”&#xff08;LeetCode 76&#xff09;是字符串处理领域的经典难题&#xff0c;也是大厂面试中高频出现的算法题。它的核心考点是滑动窗口&#xff08;双指针&#xff09; 与哈希表的结合运用&#xf…

作者头像 李华