Android SurfaceView拍照录像实现方法

发布时间 - 2026-01-10 23:01:37    点击率:

Surface的拍照实现也是很简单,一个小demo就可以把流程看懂了。

话不多说,直接上代码

布局文件

<SurfaceView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/sv_main_surface"
    />

  <Button
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:onClick="tackPhoto"
    android:text="拍照"
    />

Activity

public class MainActivity extends AppCompatActivity {

  private SurfaceView sv_main_surface;
  private Camera camera;

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    sv_main_surface = (SurfaceView) findViewById(R.id.sv_main_surface);

    //添加surface回调函数
    sv_main_surface.getHolder().addCallback(new SurfaceHolder.Callback() {


      @Override//控件创建时,打开照相机
      public void surfaceCreated(SurfaceHolder holder) {
        //打开照相机
        camera = Camera.open();
        //设置参数
        Camera.Parameters parameters=camera.getParameters();
        parameters.setPictureFormat(PixelFormat.JPEG);
        parameters.set("jpeg-quality",85);
        camera.setParameters(parameters);
        //将画面展示到SurfaceView
        try {
          camera.setPreviewDisplay(sv_main_surface.getHolder());
        } catch (IOException e) {
          e.printStackTrace();
        }
        //开启预览效果
        camera.startPreview();

      }

      @Override//控件改变
      public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

      }

      @Override//控件销毁
      public void surfaceDestroyed(SurfaceHolder holder) {
        //照相同一时刻只能允许一个软件打开
        if(camera!=null){
          camera.stopPreview();
          camera.release();//释放内存
          camera=null;
        }
      }
    });




  }


  public void takePhoto(View view){
    camera.takePicture(null, null, new Camera.PictureCallback() {
      @Override
      public void onPictureTaken(byte[] bytes, Camera camera) {
        //技术:图片压缩技术(如果图片不压缩,图片大小会过大,会报一个oom内存溢出的错误)
        Bitmap bitmap= BitmapFactory.decodeByteArray(bytes,0,bytes.length);
        try {
          FileOutputStream fos = new FileOutputStream("/mnt/sdcard/qq"+System.currentTimeMillis()+".png");//图片保存路径
          bitmap.compress(Bitmap.CompressFormat.PNG,85,fos);//压缩格式,质量,压缩路径

          camera.stopPreview();
          camera.startPreview();
        } catch (FileNotFoundException e) {
          e.printStackTrace();
        }
      }
    });

  }
}

我们还要添加权限

   <!--打开照相机的权限-->
  <uses-permission android:name="android.permission.CAMERA"></uses-permission>
  <!--创建文件的权限-->
  <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission>
  <!--写内存卡的权限-->
  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>

就这样一个小案例就完成了

既然可以拍照,那肯定也是可以录像的,所以我们再来看看录像是怎么实现的

布局文件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:id="@+id/activity_media_recorder"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:paddingBottom="@dimen/activity_vertical_margin"
  android:paddingLeft="@dimen/activity_horizontal_margin"
  android:paddingRight="@dimen/activity_horizontal_margin"
  android:paddingTop="@dimen/activity_vertical_margin"
  tools:context="com.zking.familyapp.MediaRecorderActivity">

  <SurfaceView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/sv_media_surface"
    />

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    ></LinearLayout>

  <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="开始"
    android:onClick="start"
    />
 <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="停止"
    android:onClick="stop"
    />

</RelativeLayout>


Activity

public class MediaRecorderActivity extends AppCompatActivity {

  private SurfaceView sv_media_surface;
  private MediaRecorder mediaRecorder;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_media_recorder);
    sv_media_surface = (SurfaceView) findViewById(R.id.sv_media_surface);

    //实例化媒体录制器
    mediaRecorder = new MediaRecorder();
  }


  public void start(View view){
    mediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
    mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);

    //设置格式
    mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);

    mediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H263);
    mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);

    //设置保存路径
    mediaRecorder.setOutputFile("/mnt/sdcard/uu"+System.currentTimeMillis()+".mp4");

    mediaRecorder.setPreviewDisplay(sv_media_surface.getHolder().getSurface());

    try {
      mediaRecorder.prepare();
      mediaRecorder.start();
    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  public void stop(View view){
    if(mediaRecorder!=null){
      mediaRecorder.stop();
      mediaRecorder.release();
      mediaRecorder=null;
    }

  }
}


添加权限

<!-- 录像的权限-->
  <uses-permission android:name="android.permission.RECORD_AUDIO"></uses-permission>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # SurfaceView  # 拍照  # 录像  # android surfaceView实现播放视频功能  # android使用surfaceview+MediaPlayer播放视频  # Android SurfaceView预览变形完美解决方法  # Android使用SurfaceView实现飘赞动画  # Android Surfaceview的绘制与应用  # Android截屏SurfaceView黑屏问题的解决办法  # Android系统view与SurfaceView的基本使用及区别分析  # 会报  # 是怎么  # 很简单  # 再来  # 这样一个  # 过大  # 多说  # 回调  # 大家多多  # 看懂  # 就可以  # 完成了  # 压缩技术  # 内存卡  # parameters  # open 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel怎么实现验证码(Captcha)功能  Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置  如何在阿里云ECS服务器部署织梦CMS网站?  如何构建满足综合性能需求的优质建站方案?  如何用花生壳三步快速搭建专属网站?  javascript如何操作浏览器历史记录_怎样实现无刷新导航  专业商城网站制作公司有哪些,pi商城官网是哪个?  Win10如何卸载预装Edge扩展_Win10卸载Edge扩展教程【方法】  bing浏览器学术搜索入口_bing学术文献检索地址  焦点电影公司作品,电影焦点结局是什么?  Laravel如何处理跨站请求伪造(CSRF)保护_Laravel表单安全机制与令牌校验  浅谈redis在项目中的应用  如何快速搭建支持数据库操作的智能建站平台?  使用spring连接及操作mongodb3.0实例  googleplay官方入口在哪里_Google Play官方商店快速入口指南  手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?  如何在云指建站中生成FTP站点?  Laravel如何使用Contracts(契约)进行编程_Laravel契约接口与依赖反转  Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程  网站制作企业,网站的banner和导航栏是指什么?  phpredis提高消息队列的实时性方法(推荐)  如何在服务器上三步完成建站并提升流量?  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  Laravel Telescope怎么调试_使用Laravel Telescope进行应用监控与调试  javascript读取文本节点方法小结  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】  如何安全更换建站之星模板并保留数据?  JavaScript中如何操作剪贴板_ClipboardAPI怎么用  使用Dockerfile构建java web环境  Windows Hello人脸识别突然无法使用  高端智能建站公司优选:品牌定制与SEO优化一站式服务  深圳网站制作平台,深圳市做网站好的公司有哪些?  bootstrap日历插件datetimepicker使用方法  JavaScript如何实现路由_前端路由原理是什么  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑  Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用  桂林网站制作公司有哪些,桂林马拉松怎么报名?  JavaScript模板引擎Template.js使用详解  JS经典正则表达式笔试题汇总  香港服务器网站卡顿?如何解决网络延迟与负载问题?  专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?  如何在VPS电脑上快速搭建网站?  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】  Laravel如何创建自定义Facades?(详细步骤)  微信小程序 canvas开发实例及注意事项  php嵌入式断网后怎么恢复_php检测网络重连并恢复硬件控制【操作】  Laravel如何使用Livewire构建动态组件?(入门代码)  Laravel Vite是做什么的_Laravel前端资源打包工具Vite配置与使用  linux写shell需要注意的问题(必看)