android之camera用法實(shí)例詳解
本文實(shí)例講述了android之camera用法。分享給大家供大家參考。具體如下:
1.關(guān)于預(yù)覽橫豎差90度的問題
原因分析
經(jīng)過查證和實(shí)驗(yàn),可以證實(shí):Android提供的SDK(android.hardware.Camera)里大概不能正常的使用豎屏(portrait layout)加載照相機(jī),當(dāng)用豎屏模式加載照相機(jī)時(shí)會(huì)產(chǎn)生以下情況:
①. 照相機(jī)成像左傾90度(傾斜);
②. 照相機(jī)成像長寬比例不對(失比)。
之所以是“大概”,原因是因?yàn)榭赡芸梢酝ㄟ^一些比較復(fù)雜的手段解決。如果以上成立,那為什么豎屏不能正常成像也就很顯然了。為什么會(huì)產(chǎn)生這樣的情況,請看下面的研究分析。
照相機(jī)在一般情況下是必須用landscape layout(橫屏)的,可以證明,先寫一個(gè)照相機(jī)(只要能preview就行),如果Manifest的activity里不加入android:screenOrientation="landscape",即默認(rèn)了 android:screenOrientation="portrait"(豎屏),照相機(jī)preview時(shí)就會(huì)出現(xiàn)左傾90度的現(xiàn)象,并且失比。原因是這樣的(我推測的),攝像頭對照物的映射是Android底層固定了的,以landscape方式為正,并且產(chǎn)生大小為320*480的像,如果換成portrait方式了,攝像頭還是產(chǎn)生320*480的像,然后分別對應(yīng)的放入到一個(gè)480*320的屏內(nèi),顯然會(huì)失比,然后根據(jù)豎、橫屏的規(guī)則,就產(chǎn)生了左傾90度的情況。為了進(jìn)一步證實(shí)我對失比原因的推測,我照相機(jī)內(nèi)加載的SurfaceView調(diào)成了320*213,比例大概是(320:213)*1.5=(480:320),所成像結(jié)果如愿的形成左傾但是沒有失比的狀況,這就證實(shí)了我的想法。
綜上可以看出,左傾是因?yàn)閿z像頭映射產(chǎn)生的,而失比是由于像素比例映射產(chǎn)生的。
解決方案
暫無好的解決方案,只能強(qiáng)制橫屏,記載代碼中加入
暫無好的解決方案,只能強(qiáng)制橫屏,記載代碼中加入
2.關(guān)于拍出來的照片不能正確成像,如綠屏,紅綠相間,重疊等
原因分析
有的手機(jī)不支持parameter.setPictureSize(width,height)、parameters.setPreviewSize(width,height)方法,為了兼容性建議不設(shè)置這兩個(gè)方法。
附:完整示例代碼:
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 xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" android:layout_weight="2"> <SurfaceView android:id="@+id/surfaceView" android:layout_width="fill_parent" android:layout_height="fill_parent" /> </LinearLayout> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:layout_weight="1" android:background="@android:color/white" android:layout_gravity="center"> <ImageButton android:layout_width="fill_parent" android:layout_height="80dip" android:id="@+id/btnTakePicture" android:layout_gravity="center" android:textSize="30dip" android:layout_weight="1" android:src="@drawable/btn_take_pic"/> <ImageButton android:layout_width="fill_parent" android:layout_height="80dip" android:id="@+id/btnAutoFocus" android:layout_gravity="center" android:textSize="30dip" android:layout_weight="1" android:src="@drawable/btn_auto_focus"/> </LinearLayout> </LinearLayout>
二、MainActivity拍照核心代碼:
package cn.itcast.takepicture;
import java.io.File;
import java.io.FileOutputStream;
import android.app.Activity;
import android.content.pm.ActivityInfo;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Matrix;
import android.graphics.PixelFormat;
import android.graphics.Bitmap.CompressFormat;
import android.hardware.Camera;
import android.hardware.Camera.PictureCallback;
import android.os.Bundle;
import android.os.Environment;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import android.view.SurfaceHolder.Callback;
import android.widget.ImageButton;
public class MainActivity extends Activity {
private ImageButton btnTakePicture = null;
private ImageButton btnAutoFocus = null;
private Camera camera = null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 設(shè)置窗口標(biāo)題
requestWindowFeature(Window.FEATURE_NO_TITLE);
// 橫屏
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
// 全屏顯示
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
// 當(dāng)此窗口為用戶可見時(shí),保持設(shè)備常開,并保持亮度不變。
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
setContentView(R.layout.main);
SurfaceView surfaceView = (SurfaceView) this
.findViewById(R.id.surfaceView);
surfaceView.getHolder()
.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
surfaceView.getHolder().setFixedSize(320, 240); // 設(shè)置分辨率
surfaceView.getHolder().addCallback(new SurfaceCallback());
btnTakePicture = (ImageButton) findViewById(R.id.btnTakePicture);
btnAutoFocus = (ImageButton) findViewById(R.id.btnAutoFocus);
btnTakePicture.setOnClickListener(onClickListener);
btnAutoFocus.setOnClickListener(onClickListener);
}
private final View.OnClickListener onClickListener = new View.OnClickListener() {
@Override
public void onClick(View v) {
if (v == btnTakePicture) {
if (camera != null)
camera.takePicture(null, null, new TakePictureCallback()); // 拍照
} else if (v == btnAutoFocus) {
if (camera != null)
camera.autoFocus(null); // 對焦
}
}
};
private final class SurfaceCallback implements Callback {
private boolean preview; // 是否正在預(yù)覽
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width,
int height) {
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
try {
camera = Camera.open();
Camera.Parameters parameters = camera.getParameters();
parameters.setPreviewFrameRate(5); //每秒5幀
parameters.setPictureFormat(PixelFormat.JPEG);//設(shè)置照片的輸出格式
parameters.set("jpeg-quality", 85);//照片質(zhì)量
camera.setParameters(parameters);
camera.setPreviewDisplay(holder);
camera.startPreview();
preview = true;
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
if (camera != null) {
if (preview) {
camera.stopPreview();
preview = false;
}
camera.release();
camera = null; // 記得釋放
}
}
}
private final class TakePictureCallback implements PictureCallback {
public void onPictureTaken(byte[] data, Camera camera) {
Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
Matrix matrix=new Matrix();
//設(shè)置縮放
matrix.postScale(0.5f, 0.5f);
bitmap=Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);
File file = new File(Environment.getExternalStorageDirectory(),
System.currentTimeMillis() + ".jpg");
try {
FileOutputStream outStream = new FileOutputStream(file);
bitmap.compress(CompressFormat.JPEG, 100, outStream);
outStream.close();
camera.startPreview();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
清單文件:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="cn.itcast.takepicture"
android:versionCode="1"
android:versionName="1.0" >
<application
android:icon="@drawable/icon"
android:label="@string/app_name" >
<activity
android:label="@string/app_name"
android:name=".MainActivity" >
<intent-filter >
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
<uses-sdk android:minSdkVersion="7" />
<uses-permission android:name="android.permission.CAMERA" />
<!-- 在SDCard中創(chuàng)建與刪除文件權(quán)限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<!-- 往SDCard寫入數(shù)據(jù)權(quán)限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
</manifest>
希望本文所述對大家的Android程序設(shè)計(jì)有所幫助。
- Android編程中調(diào)用Camera時(shí)預(yù)覽畫面有旋轉(zhuǎn)問題的解決方法
- Android camera實(shí)時(shí)預(yù)覽 實(shí)時(shí)處理,人臉識(shí)別示例
- Android自定義照相機(jī)Camera出現(xiàn)黑屏的解決方法
- Android實(shí)現(xiàn)Camera2預(yù)覽和拍照效果
- Android變形(Transform)之Camera使用介紹
- Android Camera是否支持變焦的判斷方法總結(jié)
- android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
- Android中使用Camera類編寫手機(jī)拍照App的實(shí)例教程
- Android Camera變焦編程步驟
- Android?Camera2開啟自動(dòng)曝光功能
相關(guān)文章
Flutter 首頁必用組件NestedScrollView的示例詳解
今天介紹的組件是NestedScrollView,大部分的App首頁都會(huì)用到這個(gè)組件。對Flutter 首頁必用組件NestedScrollView的相關(guān)知識(shí)感興趣的一起看看吧2020-05-05
Android實(shí)現(xiàn)調(diào)用攝像頭
本文給大家分享的是,在安卓APP開發(fā)的過程中,經(jīng)常會(huì)需要調(diào)用手機(jī)自身攝像頭拍照的代碼,十分的簡單實(shí)用,有需要的小伙伴可以參考下。2015-07-07
Android利用Theme自定義Activity間的切換動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android利用Theme自定義Activity間的切換動(dòng)畫,感興趣的小伙伴們可以參考一下2016-09-09
Android入門之動(dòng)態(tài)BroadCast的使用教程
系統(tǒng)自己在很多時(shí)候都會(huì)發(fā)送廣播,比如電量低或者充足,剛啟動(dòng)完,插入耳機(jī),你有一條新的微信消息,這種都是使用BroadCast機(jī)制去實(shí)現(xiàn)的。BroadCast分為靜態(tài)和動(dòng)態(tài)BroadCast兩種,本文就來聊聊動(dòng)態(tài)BroadCast的使用,需要的可以參考一下2022-12-12
Android ViewPager實(shí)現(xiàn)無限循環(huán)效果
這篇文章主要為大家詳細(xì)介紹了Android ViewPager實(shí)現(xiàn)無限循環(huán)效果的相關(guān)資料,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-03-03
Android 多線程的實(shí)現(xiàn)方法總結(jié)
這篇文章主要介紹了Android 多線程的實(shí)現(xiàn)方法總結(jié)的相關(guān)資料,這里提供三種方法,幫助大家掌握這部分內(nèi)容,需要的朋友可以參考下2017-08-08
Android動(dòng)畫學(xué)習(xí)筆記之補(bǔ)間動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android動(dòng)畫學(xué)習(xí)筆記之補(bǔ)間動(dòng)畫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12

