Surfaceview

SurfaceView

SurfaceView和View的区别:
View 主要适用于主动更新的情况,而 surfaceView 主要适用于被动更新,例如频繁的刷新。
View 在主线程中对画面进行刷新,而 surfaceView 通常会通过一个子线程来进行页面的刷新
View 在绘图时没有使用双缓冲机制,而 surfaceView 在底层实现机制上就已经实现了双缓冲机制。
总结就是,如果你的自定义 View 需要频繁刷新,或者刷新时数据处理量很大,考虑用 SurfaceView 来替代 View。

双缓冲:

在这里插入图片描述

surfaceview如何播放视频+surfaceview发送弹幕

在这里插入图片描述

实现代码

package com.example.day05_test;

import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.PixelFormat;
import android.graphics.PorterDuff;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

import java.io.IOException;
import java.util.ArrayList;

public class MainActivity extends AppCompatActivity implements SurfaceHolder.Callback {
    SurfaceView vido;
    SurfaceView danmu;
    SurfaceHolder holder_void;
    SurfaceHolder holder_danmu;
    MediaPlayer mediaPlayer = new MediaPlayer();
    EditText et;
    Button bt;
    int y = 0;
    ArrayList<Barrage> list = new ArrayList<>();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mediaPlayer = new MediaPlayer();
        bt = findViewById(R.id.bt);
        et = findViewById(R.id.et);
        //视频
        vido =findViewById(R.id.vido);
        holder_void = vido.getHolder();
        holder_void.addCallback(this);
        //弹幕
        danmu =findViewById(R.id.danmu);
        holder_danmu = danmu.getHolder();
        holder_danmu.addCallback(this);
        //变成半透明
        danmu.setZOrderOnTop(true);
        holder_danmu.setFormat(PixelFormat.TRANSLUCENT);
        //自定义方法
        initMedia();
        //按钮点击事件
        bt.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                String text = et.getText().toString();
                int height = danmu.getHeight();
                y = (int) (Math.random()*height);
                Barrage barrage = new Barrage(0, y, text);
                list.add(barrage);
            }
        });
    }

    private void initMedia() {
        mediaPlayer.reset();
        try {
            mediaPlayer.setDataSource("http://wvideo.spriteapp.cn/video/2019/0708/5d232436d1fc5_wpd.mp4");
            mediaPlayer.prepareAsync();
            mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
                @Override
                public void onPrepared(MediaPlayer mp) {
                    mediaPlayer.start();
                }
            });
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void surfaceCreated(SurfaceHolder holder) {
                if (holder == holder_void){
                    mediaPlayer.setDisplay(holder);
                }else {
                    new MyThread().start();
                }
    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

    }

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {

    }
    class MyThread extends Thread{
        @Override
        public void run() {
            Paint paint = new Paint();
            paint.setTextSize(60);
            while (true){
                Canvas canvas = holder_danmu.lockCanvas();
                if (canvas == null){
                    break;
                }
                canvas.drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR);
                for (int i = 0; i < list.size(); i++) {
                    int a = (int) (Math.random()*255);
                    int b = (int) (Math.random()*255);
                    int c = (int) (Math.random()*255);
                    paint.setColor(Color.rgb(a,b,c));
                    int x = list.get(i).getX();
                    list.get(i).setX(x+=10);
                    canvas.drawText(list.get(i).getText(),list.get(i).getX(),list.get(i).getY(),paint);
                }
                holder_danmu.unlockCanvasAndPost(canvas);
            }
        }
    }
}

布局代码

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" android:orientation="vertical">
    <FrameLayout
        android:layout_weight="8"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <SurfaceView
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:id="@+id/vido"
            />
        <SurfaceView
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:id="@+id/danmu"
            />
    </FrameLayout>
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        <EditText
            android:id="@+id/et"
            android:layout_width="0dp"
            android:layout_height="wrap_content" android:layout_weight="8"/>
        <Button
            android:layout_width="0dp"
            android:layout_height="wrap_content" android:layout_weight="2"
            android:text="发送"
            android:id="@+id/bt"
            />
    </LinearLayout>
</LinearLayout>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

MX_XXS

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值