Android提高之SurfaceView与多线程的混搭实例

   2015-07-09 0
核心提示:这篇文章主要介绍了Android提高之SurfaceView与多线程的混搭,很实用的功能,需要的朋友可以参考下

前文简单介绍了Android中SurfaceView的基本使用,本文就来介绍一下SurfaceView与多线程的混搭。SurfaceView与多线程混搭,是为了防止动画闪烁而实现的一种多线程应用。android的多线程用法与JAVA的多线程用法完全一样,本文不做多线程方面的介绍了。直接讲解SurfaceView与多线程的混合使用,即开一条线程专门读取图片,另外一条线程专门绘图。

本文程序运行截图如下,左边是开单个线程读取并绘图,右边是开两个线程,一个专门读取图片,一个专门绘图:

Android提高之SurfaceView与多线程的混搭实例

对比一下可以看出,右边动画的帧速明显比左边的快,左右两者都没使用Thread.sleep()。为什么要开两个线程一个读一个画,而不去开两个线程像左边那样都“边读边画”呢?因为SurfaceView每次绘图都会锁定Canvas,也就是说同一片区域这次没画完下次就不能画,因此要提高动画播放的效率,就得开一条线程专门画图,开另外一条线程做预处理的工作。

main.xml的源码如下:

<xml version="1.0" encoding="utf-8">
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="fill_parent" android:layout_height="fill_parent"
 android:orientation="vertical">

 <LinearLayout android:id="@+id/LinearLayout01"
 android:layout_width="wrap_content" android:layout_height="wrap_content">
 <Button android:id="@+id/Button01" android:layout_width="wrap_content"
  android:layout_height="wrap_content" android:text="单个独立线程"></Button>
 <Button android:id="@+id/Button02" android:layout_width="wrap_content"
  android:layout_height="wrap_content" android:text="两个独立线程"></Button>
 </LinearLayout>
 <SurfaceView android:id="@+id/SurfaceView01"
 android:layout_width="fill_parent" android:layout_height="fill_parent"></SurfaceView>
</LinearLayout>

Java程序的源码如下:

package com.testSurfaceView;
import java.lang.reflect.Field;
import java.util.ArrayList;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.Button;

public class testSurfaceView extends Activity {
 /** Called when the activity is first created. */
 Button btnSingleThread, btnDoubleThread;
 SurfaceView sfv;
 SurfaceHolder sfh;
 ArrayList<Integer> imgList = new ArrayList<Integer>();
 int imgWidth, imgHeight;
 Bitmap bitmap;//独立线程读取,独立线程绘图

 @Override
 public void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);

 btnSingleThread = (Button) this.findViewById(R.id.Button01);
 btnDoubleThread = (Button) this.findViewById(R.id.Button02);
 btnSingleThread.setOnClickListener(new ClickEvent());
 btnDoubleThread.setOnClickListener(new ClickEvent());
 sfv = (SurfaceView) this.findViewById(R.id.SurfaceView01);
 sfh = sfv.getHolder();
 sfh.addCallback(new MyCallBack());// 自动运行surfaceCreated以及surfaceChanged
 }

 class ClickEvent implements View.OnClickListener {

 @Override
 public void onClick(View v) {

  if (v == btnSingleThread) {
  new Load_DrawImage(0, 0).start();//开一条线程读取并绘图
  } else if (v == btnDoubleThread) {
  new LoadImage().start();//开一条线程读取
  new DrawImage(imgWidth + 10, 0).start();//开一条线程绘图
  }
 }
 }
 class MyCallBack implements SurfaceHolder.Callback {
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
  int height) {
  Log.i("Surface:", "Change");
 }
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
  Log.i("Surface:", "Create");
  // 用反射机制来获取资源中的图片ID和尺寸
  Field[] fields = R.drawable.class.getDeclaredFields();
  for (Field field : fields) {
  if (!"icon".equals(field.getName()))// 除了icon之外的图片
  {
   int index = 0;
   try {
   index = field.getInt(R.drawable.class);
   } catch (IllegalArgumentException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
   } catch (IllegalAccessException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
   }
   // 保存图片ID
   imgList.add(index);
  }
  }
  // 取得图像大小
  Bitmap bmImg = BitmapFactory.decodeResource(getResources(),
   imgList.get(0));
  imgWidth = bmImg.getWidth();
  imgHeight = bmImg.getHeight();
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
  Log.i("Surface:", "Destroy");
 }
 }
 /*
 * 读取并显示图片的线程
 */
 class Load_DrawImage extends Thread {
 int x, y;
 int imgIndex = 0;

 public Load_DrawImage(int x, int y) {
  this.x = x;
  this.y = y;
 }
 public void run() {
  while (true) {
  Canvas c = sfh.lockCanvas(new Rect(this.x, this.y, this.x
   + imgWidth, this.y + imgHeight));
  Bitmap bmImg = BitmapFactory.decodeResource(getResources(),
   imgList.get(imgIndex));
  c.drawBitmap(bmImg, this.x, this.y, new Paint());
  imgIndex++;
  if (imgIndex == imgList.size())
   imgIndex = 0;
  sfh.unlockCanvasAndPost(c);// 更新屏幕显示内容
  }
 }
 };
 /*
 * 只负责绘图的线程
 */
 class DrawImage extends Thread {
 int x, y;
 public DrawImage(int x, int y) {
  this.x = x;
  this.y = y;
 }
 public void run() {
  while (true) {
  if (bitmap != null) {//如果图像有效
   Canvas c = sfh.lockCanvas(new Rect(this.x, this.y, this.x
    + imgWidth, this.y + imgHeight));

   c.drawBitmap(bitmap, this.x, this.y, new Paint());

   sfh.unlockCanvasAndPost(c);// 更新屏幕显示内容
  }
  }
 }
 };
 /*
 * 只负责读取图片的线程
 */
 class LoadImage extends Thread {
 int imgIndex = 0;
 public void run() {
  while (true) {
  bitmap = BitmapFactory.decodeResource(getResources(),
   imgList.get(imgIndex));
  imgIndex++;
  if (imgIndex == imgList.size())//如果到尽头则重新读取
   imgIndex = 0;
  }
 }
 };
}

希望本文所述示例能对大家进行Android的SurfaceView与多线程的混搭编程有所帮助。

 
标签: Android SurfaceView
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 说一说Android Studio和IDEA中一个很有用的内存调试插件
    说一说Android Studio和IDEA中一个很有用的内存
    JetBrains JVM Debugger Memory View plugin 在我最近的研发活动期间寻找新的工具,以提高我的开发经验,使Android Studio的生活更轻松,我发现一个有用的插件,我从来没有听说过。 这就是为什么,我决定写这个强大的工具,它如何帮助我与内存调试我的应用程
  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • SDK热更之如何在SDK代码中自动插桩及如何生成补
    写在前面本文是SDKHotfix相关的SDK热更系列文章中的一篇,以下为项目及系列文章相关链接:SDKHotfix整体介绍:http://blog.bihe0832.com/sdk_hotfix_project.htmlSDKHotfix对应github地址:https://github.com/bihe0832/SDKHoxFix这篇文章主要介绍一下SDK热更
  • 安装量破千万的第一个产品,我总结了3句话
    安装量破千万的第一个产品,我总结了3句话
    在今天的文章中,作者回顾了自己的第一个产品,他说“我做的第一款产品,是我的一块里程碑。”一起来看看~背景老牌大型互联网公司,部门内部创业的一个项目。我作为产品经理,也是第一次做产品经理,主导产品项目。实际上,项目初期包括我和安卓开发2个人。开
  • 移动周刊第 176 期:Android 知识梳理
    移动周刊第 176 期:Android 知识梳理
    写在前面 本期移动周刊第 176 期如约而至,聚焦 Android、iOS、VR/AR/MR、直播等前沿移动开发技术,收录一周最热点,解读开发技巧,每周三移动周刊抢先看,我们希望从中能够让你有一些收获,如果你有好的文章以及优化建议,请发送邮件至mobilehub@csdn.net,
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
点击排行