Android中使用Camera類編寫手機(jī)拍照App的實(shí)例教程
Camera是Android攝像頭硬件的相機(jī)類,位于硬件包"android.hardware.Camera"下。它主要用于攝像頭捕獲圖片、啟動/停止預(yù)覽圖片、拍照、獲取視頻幀等,它是設(shè)備本地的服務(wù),負(fù)責(zé)管理設(shè)備上的攝像頭硬件。
Camera既然用于管理設(shè)備上的攝像頭硬件,那么它也為開發(fā)人員提供了相應(yīng)的方法,并且這些方法大部分都是native的,用C++在底層實(shí)現(xiàn),下面簡單介紹一下Camera的一些方法:
- static Camera open():打開Camera,返回一個Camera實(shí)例。
- static Camera open(int cameraId):根據(jù)cameraId打開一個Camera,返回一個Camera實(shí)例。
- final void release():釋放掉Camera的資源。
- static int getNumberOfCameras():獲取當(dāng)前設(shè)備支持的Camera硬件個數(shù)。
- Camera.Parameters getParameters():獲取Camera的各項參數(shù)設(shè)置類。
- void setParameters(Camera.Parameters params):通過params把Camera的各項參數(shù)寫入到Camera中。
- final void setDisplayOrientation(int degrees):攝像預(yù)覽的旋轉(zhuǎn)度。
- final void setPreviewDisplay(SurfaceHolder holder):設(shè)置Camera預(yù)覽的SurfaceHolder。
- final void starPreview():開始Camera的預(yù)覽。
- final void stopPreview():停止Camera的預(yù)覽
- final void autoFocus(Camera.AutoFocusCallback cb):自動對焦。
- final takePicture(Camera.ShutterCallback shutter,Camera.PictureCallback raw,Camera.PictureCallback jpeg):拍照。
- final void lock():鎖定Camera硬件,使其他應(yīng)用無法訪問。
- final void unlock():解鎖Camera硬件,使其他應(yīng)用可以訪問。
上面已經(jīng)介紹了Camera的常用方法,下面根據(jù)這些方法詳細(xì)講解Android下使用Camera開發(fā)拍照應(yīng)用最基本的過程:
- 使用open()方法獲取一個Camera對象,鑒于Android設(shè)備可能配置了多個攝像頭,open()方法可以通過攝像頭Id開啟指定的攝像頭。
- 為Camera對象設(shè)置預(yù)覽類,它是一個SurfaceHolder對象,通過setPreviewDisplay(SurfaceHolder)方法設(shè)置。
- 調(diào)用startPreview()方法開始Camera對象的預(yù)覽。
- 調(diào)用takePicture()方法進(jìn)行拍照,其中可以通過Camera.PictureCallback()回調(diào)獲得拍攝的Image數(shù)據(jù)。
- 當(dāng)拍攝完成后,需要調(diào)用stopPreview()方法停止預(yù)覽,并使用release()釋放Camera占用的資源。
以上介紹的步驟都是最基本的過程,是必不可少的。Camera沒有提供公開的構(gòu)造函數(shù),只能通過open()方法獲取,并且必須設(shè)置一個預(yù)覽類SurfaceHolder,如果不設(shè)置的話,將無法使用Camera。在使用完成Camera之后,必須使用release()釋放Camera資源。
實(shí)例:
使用Camera控制拍照的幾個步驟:
1、調(diào)用Camera的open()打開相機(jī)
2、調(diào)用Camera的getParameters()獲取拍照參數(shù)。該方法返回一個Camera.Paremeters對象
3、調(diào)用Camera.Parameters對象方法設(shè)置拍照的參數(shù)
4、調(diào)用Camera.startPreview()方法開始預(yù)覽取景,在預(yù)覽取景之前需要調(diào)用Camera的setPreviewDisplay(SurfaceHolder holder)方法設(shè)置使用哪個SurfaceView來顯示取景圖片。
5、調(diào)用Camera的takePicture()方法進(jìn)行拍照
6、結(jié)束程序時,調(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添加一個回調(diào)監(jiān)聽器
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) {
// 打開攝像頭
initCamera();
}
@Override
public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2,
int arg3) {
}
});
}
private void initCamera() {
if (!isPreview) {
// 此處默認(rèn)打開后置攝像頭
// 通過傳入?yún)?shù)可以打開前置攝像頭
camera = Camera.open();
camera.setDisplayOrientation(90);
}
if (!isPreview && camera != null) {
Camera.Parameters parameters = camera.getParameters();
// 設(shè)置預(yù)覽照片的大小
parameters.setPreviewSize(screenWidth, screenHeight);
// 設(shè)置預(yù)覽照片時每秒顯示多少幀的最小值和最大值
parameters.setPreviewFpsRange(4, 10);
// 設(shè)置照片的格式
parameters.setPictureFormat(ImageFormat.JPEG);
// 設(shè)置JPG照片的質(zhì)量
parameters.set("jpeg-quality", 85);
// 設(shè)置照片的大小
parameters.setPictureSize(screenWidth, screenHeight);
// 通過SurfaceView顯示取景畫面
try {
camera.setPreviewDisplay(surfaceHodler);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
// 開始預(yù)覽
camera.startPreview();
isPreview = true;
}
}
public void capture(View source) {
if (camera != null) {
// 控制攝像頭自動對焦后才拍攝
camera.autoFocus(autoFocusCallback);
}
}
AutoFocusCallback autoFocusCallback = new AutoFocusCallback() {
@Override
public void onAutoFocus(boolean arg0, Camera arg1) {
if (arg0) {
// takePicture()方法需要傳入三個監(jiān)聽參數(shù)
// 第一個監(jiān)聽器;當(dāng)用戶按下快門時激發(fā)該監(jiān)聽器
// 第二個監(jiān)聽器;當(dāng)相機(jī)獲取原始照片時激發(fā)該監(jiān)聽器
// 第三個監(jiān)聽器;當(dāng)相機(jī)獲取JPG照片時激發(fā)該監(jiān)聽器
camera.takePicture(new ShutterCallback() {
@Override
public void onShutter() {
// 按下快門瞬間會執(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對話框上的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)建一個位于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)毫秒級拍照實(shí)例
- android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
- Android實(shí)現(xiàn)拍照、選擇圖片并裁剪圖片功能
- Android啟動相機(jī)拍照并返回圖片
- Android拍照保存在系統(tǒng)相冊不顯示的問題解決方法
- Android自定義Camera實(shí)現(xiàn)拍照小功能
相關(guān)文章
Android TextView(圓?。┻吙蚝捅尘皩?shí)例詳解
這篇文章主要介紹了 Android TextView(圓弧)邊框和背景實(shí)例詳解的相關(guān)資料,這里提供了實(shí)現(xiàn)代碼和實(shí)現(xiàn)效果圖,需要的朋友可以參考下2016-11-11
AndroidStuio插件開發(fā)適用于jetbrains全家桶
這篇文章主要介紹了AndroidStuio插件開發(fā)適用于jetbrains全家桶,本文通過實(shí)例給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12
Android水波紋載入控件CircleWaterWaveView使用詳解
這篇文章主要為大家詳細(xì)介紹了Android水波紋載入控件CircleWaterWaveView使用方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-01-01
Android使用ViewDragHelper實(shí)現(xiàn)QQ聊天氣泡拖動效果
這篇文章主要為大家詳細(xì)介紹了Android使用ViewDragHelper實(shí)現(xiàn)QQ聊天氣泡拖動效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-01-01
Kotlin整合Vertx開發(fā)Web應(yīng)用
這篇文章主要介紹了Kotlin整合Vertx開發(fā)Web應(yīng)用,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-02-02
Android開發(fā)之在xml中設(shè)置自定義屬性的方法
下面小編就為大家分享一篇Android開發(fā)之在xml中設(shè)置自定義屬性的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-01-01
Thread、Handler和HandlerThread關(guān)系詳解
這篇文章主要介紹了Thread、Handler和HandlerThread關(guān)系詳解的相關(guān)資料,需要的朋友可以參考下2016-09-09
Android布局ConstraintLayout代碼修改約束及輔助功能
這篇文章主要為大家介紹了Android布局ConstraintLayout代碼修改約束及輔助功能示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09

