Android中使用Camera類編寫(xiě)手機(jī)拍照App的實(shí)例教程
Camera是Android攝像頭硬件的相機(jī)類,位于硬件包"android.hardware.Camera"下。它主要用于攝像頭捕獲圖片、啟動(dòng)/停止預(yù)覽圖片、拍照、獲取視頻幀等,它是設(shè)備本地的服務(wù),負(fù)責(zé)管理設(shè)備上的攝像頭硬件。
Camera既然用于管理設(shè)備上的攝像頭硬件,那么它也為開(kāi)發(fā)人員提供了相應(yīng)的方法,并且這些方法大部分都是native的,用C++在底層實(shí)現(xiàn),下面簡(jiǎn)單介紹一下Camera的一些方法:
- static Camera open():打開(kāi)Camera,返回一個(gè)Camera實(shí)例。
- static Camera open(int cameraId):根據(jù)cameraId打開(kāi)一個(gè)Camera,返回一個(gè)Camera實(shí)例。
- final void release():釋放掉Camera的資源。
- static int getNumberOfCameras():獲取當(dāng)前設(shè)備支持的Camera硬件個(gè)數(shù)。
- Camera.Parameters getParameters():獲取Camera的各項(xiàng)參數(shù)設(shè)置類。
- void setParameters(Camera.Parameters params):通過(guò)params把Camera的各項(xiàng)參數(shù)寫(xiě)入到Camera中。
- final void setDisplayOrientation(int degrees):攝像預(yù)覽的旋轉(zhuǎn)度。
- final void setPreviewDisplay(SurfaceHolder holder):設(shè)置Camera預(yù)覽的SurfaceHolder。
- final void starPreview():開(kāi)始Camera的預(yù)覽。
- final void stopPreview():停止Camera的預(yù)覽
- final void autoFocus(Camera.AutoFocusCallback cb):自動(dòng)對(duì)焦。
- final takePicture(Camera.ShutterCallback shutter,Camera.PictureCallback raw,Camera.PictureCallback jpeg):拍照。
- final void lock():鎖定Camera硬件,使其他應(yīng)用無(wú)法訪問(wèn)。
- final void unlock():解鎖Camera硬件,使其他應(yīng)用可以訪問(wèn)。
上面已經(jīng)介紹了Camera的常用方法,下面根據(jù)這些方法詳細(xì)講解Android下使用Camera開(kāi)發(fā)拍照應(yīng)用最基本的過(guò)程:
- 使用open()方法獲取一個(gè)Camera對(duì)象,鑒于Android設(shè)備可能配置了多個(gè)攝像頭,open()方法可以通過(guò)攝像頭Id開(kāi)啟指定的攝像頭。
- 為Camera對(duì)象設(shè)置預(yù)覽類,它是一個(gè)SurfaceHolder對(duì)象,通過(guò)setPreviewDisplay(SurfaceHolder)方法設(shè)置。
- 調(diào)用startPreview()方法開(kāi)始Camera對(duì)象的預(yù)覽。
- 調(diào)用takePicture()方法進(jìn)行拍照,其中可以通過(guò)Camera.PictureCallback()回調(diào)獲得拍攝的Image數(shù)據(jù)。
- 當(dāng)拍攝完成后,需要調(diào)用stopPreview()方法停止預(yù)覽,并使用release()釋放Camera占用的資源。
以上介紹的步驟都是最基本的過(guò)程,是必不可少的。Camera沒(méi)有提供公開(kāi)的構(gòu)造函數(shù),只能通過(guò)open()方法獲取,并且必須設(shè)置一個(gè)預(yù)覽類SurfaceHolder,如果不設(shè)置的話,將無(wú)法使用Camera。在使用完成Camera之后,必須使用release()釋放Camera資源。
實(shí)例:
使用Camera控制拍照的幾個(gè)步驟:
1、調(diào)用Camera的open()打開(kāi)相機(jī)
2、調(diào)用Camera的getParameters()獲取拍照參數(shù)。該方法返回一個(gè)Camera.Paremeters對(duì)象
3、調(diào)用Camera.Parameters對(duì)象方法設(shè)置拍照的參數(shù)
4、調(diào)用Camera.startPreview()方法開(kāi)始預(yù)覽取景,在預(yù)覽取景之前需要調(diào)用Camera的setPreviewDisplay(SurfaceHolder holder)方法設(shè)置使用哪個(gè)SurfaceView來(lái)顯示取景圖片。
5、調(diào)用Camera的takePicture()方法進(jìn)行拍照
6、結(jié)束程序時(shí),調(diào)用Camera的stopPreview()結(jié)束取景預(yù)覽,并調(diào)用release()方法釋放資源
代碼:
<uses-permission android:name="android.permission.CAMERA"/> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" > <SurfaceView android:id="@+id/sView" android:layout_width="match_parent" android:layout_height="wrap_content"/> <Button android:id="@+id/take" android:layout_alignParentBottom="true" android:onClick="capture" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/take"/> </RelativeLayout> <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" > <EditText android:id="@+id/photoNmae" android:layout_width="match_parent" android:layout_height="wrap_content"/> <ImageView android:id="@+id/show" android:layout_below="@id/photoNmae" android:layout_width="match_parent" android:layout_height="wrap_content"/> </RelativeLayout>
package com.android.xiong.cameratest; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.graphics.Bitmap; import android.graphics.Bitmap.CompressFormat; import android.graphics.BitmapFactory; import android.graphics.ImageFormat; import android.hardware.Camera; import android.hardware.Camera.AutoFocusCallback; import android.hardware.Camera.PictureCallback; import android.hardware.Camera.ShutterCallback; import android.os.Bundle; import android.os.Environment; import android.util.DisplayMetrics; import android.view.Display; import android.view.Menu; import android.view.SurfaceHolder; import android.view.SurfaceHolder.Callback; import android.view.LayoutInflater; import android.view.SurfaceView; import android.view.View; import android.view.Window; import android.view.WindowManager; import android.widget.EditText; import android.widget.ImageView; public class MainActivity extends Activity { SurfaceView sView; SurfaceHolder surfaceHodler; int screenWidth, screenHeight; // 定義系統(tǒng)所用的照相機(jī) Camera camera; // 是否存在預(yù)覽中 boolean isPreview = false; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 設(shè)置全屏 requestWindowFeature(Window.FEATURE_NO_TITLE); getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); setContentView(R.layout.activity_main); // 獲取窗口管理器 WindowManager wm = getWindowManager(); Display display = wm.getDefaultDisplay(); DisplayMetrics metrics = new DisplayMetrics(); // 獲取屏幕的寬和高 display.getMetrics(metrics); screenWidth = metrics.widthPixels; screenHeight = metrics.heightPixels; sView = (SurfaceView) findViewById(R.id.sView); // 設(shè)置surface不需要自己的維護(hù)緩存區(qū) sView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); // 獲得SurfaceView的SurfaceHolder surfaceHodler = sView.getHolder(); // 為srfaceHolder添加一個(gè)回調(diào)監(jiān)聽(tīng)器 surfaceHodler.addCallback(new Callback() { @Override public void surfaceDestroyed(SurfaceHolder arg0) { // 如果camera不為null,釋放攝像頭 if (camera != null) { if (isPreview) camera.stopPreview(); camera.release(); camera = null; } } @Override public void surfaceCreated(SurfaceHolder arg0) { // 打開(kāi)攝像頭 initCamera(); } @Override public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) { } }); } private void initCamera() { if (!isPreview) { // 此處默認(rèn)打開(kāi)后置攝像頭 // 通過(guò)傳入?yún)?shù)可以打開(kāi)前置攝像頭 camera = Camera.open(); camera.setDisplayOrientation(90); } if (!isPreview && camera != null) { Camera.Parameters parameters = camera.getParameters(); // 設(shè)置預(yù)覽照片的大小 parameters.setPreviewSize(screenWidth, screenHeight); // 設(shè)置預(yù)覽照片時(shí)每秒顯示多少幀的最小值和最大值 parameters.setPreviewFpsRange(4, 10); // 設(shè)置照片的格式 parameters.setPictureFormat(ImageFormat.JPEG); // 設(shè)置JPG照片的質(zhì)量 parameters.set("jpeg-quality", 85); // 設(shè)置照片的大小 parameters.setPictureSize(screenWidth, screenHeight); // 通過(guò)SurfaceView顯示取景畫(huà)面 try { camera.setPreviewDisplay(surfaceHodler); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } // 開(kāi)始預(yù)覽 camera.startPreview(); isPreview = true; } } public void capture(View source) { if (camera != null) { // 控制攝像頭自動(dòng)對(duì)焦后才拍攝 camera.autoFocus(autoFocusCallback); } } AutoFocusCallback autoFocusCallback = new AutoFocusCallback() { @Override public void onAutoFocus(boolean arg0, Camera arg1) { if (arg0) { // takePicture()方法需要傳入三個(gè)監(jiān)聽(tīng)參數(shù) // 第一個(gè)監(jiān)聽(tīng)器;當(dāng)用戶按下快門時(shí)激發(fā)該監(jiān)聽(tīng)器 // 第二個(gè)監(jiān)聽(tīng)器;當(dāng)相機(jī)獲取原始照片時(shí)激發(fā)該監(jiān)聽(tīng)器 // 第三個(gè)監(jiān)聽(tīng)器;當(dāng)相機(jī)獲取JPG照片時(shí)激發(fā)該監(jiān)聽(tīng)器 camera.takePicture(new ShutterCallback() { @Override public void onShutter() { // 按下快門瞬間會(huì)執(zhí)行此處代碼 } }, new PictureCallback() { @Override public void onPictureTaken(byte[] arg0, Camera arg1) { // 此處代碼可以決定是否需要保存原始照片信息 } }, myJpegCallback); } } }; PictureCallback myJpegCallback = new PictureCallback() { @Override public void onPictureTaken(byte[] data, Camera camera) { // 根據(jù)拍照所得的數(shù)據(jù)創(chuàng)建位圖 final Bitmap bm = BitmapFactory.decodeByteArray(data, 0, data.length); // 加載布局文件 View saveDialog = getLayoutInflater().inflate(R.layout.save, null); final EditText potoName = (EditText) saveDialog .findViewById(R.id.photoNmae); // 獲取saveDialog對(duì)話框上的ImageView組件 ImageView show = (ImageView) saveDialog.findViewById(R.id.show); // 顯示剛剛拍得的照片 show.setImageBitmap(bm); // 使用AlertDialog組件 new AlertDialog.Builder(MainActivity.this) .setView(saveDialog) .setNegativeButton("取消", null) .setPositiveButton("保存", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface arg0, int arg1) { // 創(chuàng)建一個(gè)位于SD卡上的文件 File file = new File(Environment .getExternalStorageDirectory() + "/" + potoName.getText().toString() + ".jpg"); FileOutputStream fileOutStream=null; try { fileOutStream=new FileOutputStream(file); //把位圖輸出到指定的文件中 bm.compress(CompressFormat.JPEG, 100, fileOutStream); fileOutStream.close(); } catch (IOException io) { io.printStackTrace(); } } }).show(); //重新瀏覽 camera.stopPreview(); camera.startPreview(); isPreview=true; } }; @Override public boolean onCreateOptionsMenu(Menu menu) { getMenuInflater().inflate(R.menu.main, menu); return true; } }
- android自定義Camera實(shí)現(xiàn)錄像和拍照
- Android自定義Camera實(shí)現(xiàn)拍照功能
- Android實(shí)現(xiàn)Camera2預(yù)覽和拍照效果
- android自定義Camera拍照并查看圖片
- Android Camera實(shí)現(xiàn)毫秒級(jí)拍照實(shí)例
- android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
- Android實(shí)現(xiàn)拍照、選擇圖片并裁剪圖片功能
- Android啟動(dòng)相機(jī)拍照并返回圖片
- Android拍照保存在系統(tǒng)相冊(cè)不顯示的問(wèn)題解決方法
- Android自定義Camera實(shí)現(xiàn)拍照小功能
相關(guān)文章
Android TextView(圓弧)邊框和背景實(shí)例詳解
這篇文章主要介紹了 Android TextView(圓?。┻吙蚝捅尘皩?shí)例詳解的相關(guān)資料,這里提供了實(shí)現(xiàn)代碼和實(shí)現(xiàn)效果圖,需要的朋友可以參考下2016-11-11Android編程之短信列表的時(shí)間顯示實(shí)例分析
這篇文章主要介紹了Android編程之短信列表的時(shí)間顯示,實(shí)例分析了Android中短信列表的時(shí)間顯示及具體注意事項(xiàng),具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11AndroidStuio插件開(kāi)發(fā)適用于jetbrains全家桶
這篇文章主要介紹了AndroidStuio插件開(kāi)發(fā)適用于jetbrains全家桶,本文通過(guò)實(shí)例給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12Android水波紋載入控件CircleWaterWaveView使用詳解
這篇文章主要為大家詳細(xì)介紹了Android水波紋載入控件CircleWaterWaveView使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01Android使用ViewDragHelper實(shí)現(xiàn)QQ聊天氣泡拖動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android使用ViewDragHelper實(shí)現(xiàn)QQ聊天氣泡拖動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01Kotlin整合Vertx開(kāi)發(fā)Web應(yīng)用
這篇文章主要介紹了Kotlin整合Vertx開(kāi)發(fā)Web應(yīng)用,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-02-02Android開(kāi)發(fā)之在xml中設(shè)置自定義屬性的方法
下面小編就為大家分享一篇Android開(kāi)發(fā)之在xml中設(shè)置自定義屬性的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01簡(jiǎn)單實(shí)現(xiàn)Android刮刮卡效果
這篇文章主要教大家如何簡(jiǎn)單的實(shí)現(xiàn)Android刮刮卡效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07Thread、Handler和HandlerThread關(guān)系詳解
這篇文章主要介紹了Thread、Handler和HandlerThread關(guān)系詳解的相關(guān)資料,需要的朋友可以參考下2016-09-09Android布局ConstraintLayout代碼修改約束及輔助功能
這篇文章主要為大家介紹了Android布局ConstraintLayout代碼修改約束及輔助功能示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09