Android使用系統(tǒng)相機(jī)進(jìn)行拍照的步驟
前言
我們?cè)谌粘5拈_(kāi)發(fā)中有時(shí)候會(huì)遇到需要用到相機(jī)的需求,而相機(jī)也是很常用的東西,例如掃二維碼啊拍照上傳啊等等。這里我不講像qq那樣自定義很強(qiáng)的拍照功能(事實(shí)上我也不會(huì)),講個(gè)最簡(jiǎn)單的調(diào)用系統(tǒng)相機(jī)拍照并儲(chǔ)存
調(diào)用系統(tǒng)相機(jī)步驟
這里我通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)講這個(gè)內(nèi)容。
我自己寫了一個(gè)demo,布局很簡(jiǎn)單:
<Button android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="4dp" android:text="take phone" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintHorizontal_bias="0.281" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" /> <ImageView android:id="@+id/imageView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="29dp" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toBottomOf="@+id/button" app:srcCompat="@mipmap/ic_launcher_round" />
就是一個(gè)按鈕點(diǎn)擊彈起相機(jī),然后一個(gè)imageView顯示拍到的照片。
接下來(lái)我想一下調(diào)用的整個(gè)過(guò)程我們需要做什么:
首先彈起相機(jī)肯定要跳到相機(jī)這個(gè)應(yīng)用,那么就必須通過(guò)隱性啟動(dòng)相機(jī)的活動(dòng)。
然后當(dāng)我們返回應(yīng)用的時(shí)候,還要將照片顯示,所以這里就要用到startActivityForResult這個(gè)方法。
其次,我們拍照之后肯定要進(jìn)行儲(chǔ)存的,那么就涉及到文件的操作。
涉及到內(nèi)存的操作就肯定要和權(quán)限打交道,所有還有權(quán)限相關(guān)的內(nèi)容。
最后還有一個(gè)問(wèn)題就是,相機(jī)拍完照是要儲(chǔ)存照片的,所以我們要給他一個(gè)地址uri,但是可不可以直接把地址當(dāng)成參數(shù)發(fā)過(guò)去呢?這里就要用到特殊的內(nèi)容提供器FileProvider。
上面就是調(diào)用相機(jī)要用到的內(nèi)容,雖然用的都很淺,但是都會(huì)涉及到。接下來(lái)看看具體怎么實(shí)現(xiàn)??纯碅ctivity中的onCreate的代碼:
private Uri imageUri; private ImageView imageView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); imageView = findViewById(R.id.imageView); Button button = findViewById(R.id.button); button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { //創(chuàng)建一個(gè)File對(duì)象。getExternalCacheDir()獲取應(yīng)用的緩存目錄,outputImage.jpg是照片名稱 File outputImage = new File(getExternalCacheDir(),"outputImage.jpg"); try{ //創(chuàng)建一個(gè)空文件 outputImage.createNewFile(); }catch (IOException e){ e.printStackTrace(); } //不同的安卓版本對(duì)用不同的獲取Uri的方法 if (Build.VERSION.SDK_INT>=24){ imageUri =FileProvider.getUriForFile(MainActivity.this,"huan",outputImage); }else{ imageUri = Uri.fromFile(outputImage); } //啟動(dòng)相機(jī)的對(duì)應(yīng)Activity Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); intent.putExtra(MediaStore.EXTRA_OUTPUT,imageUri); startActivityForResult(intent,1); } });
我們來(lái)看看這里的代碼:前面的代碼很簡(jiǎn)單就是控件的初始化。
我們知道照片是要放文件夾得,所以這里要?jiǎng)?chuàng)建一個(gè)File對(duì)象,指定文件的路徑以及名字。這里路徑為什么要用getExternalCacheDir()呢?因?yàn)槊總€(gè)應(yīng)用都有對(duì)應(yīng)得緩存目錄,訪問(wèn)這些目錄得時(shí)候不用訪問(wèn)內(nèi)存權(quán)限,這樣得話就可以省去需求權(quán)限得步驟啦。這個(gè)目錄在/scare/Android/data//cache。
然后我們?cè)賱?chuàng)建一個(gè)空的文件夾。這里如果已經(jīng)有照片了的話例如我們第二次拍照的時(shí)候,那么就不會(huì)創(chuàng)建新的空文件夾了。直到儲(chǔ)存的時(shí)候才會(huì)被替換掉。
然后我們剛才講到,拍到的圖片要在我們的應(yīng)用中展示,那么就必須用到內(nèi)容提供器。這里用到FileProvider來(lái)獲取uri,關(guān)于provider我在下文有講到可以
如果是低于4.4版本的安卓就用Uri.fromFile(outputImage);方法可以獲取到uri
再通過(guò)隱式啟動(dòng)相機(jī)activity可以打開(kāi)相機(jī)了。這里系統(tǒng)相機(jī)的action是android.media.action.IMAGE_CAPTURE,相機(jī)儲(chǔ)存路徑的參數(shù)名字是MediaStore.EXTRA_OUTPUT,并把uri傳輸進(jìn)去。
好了這樣就完成了拍照并把照片儲(chǔ)存的步驟了。接下來(lái)還差什么?對(duì)了,把照片顯示出來(lái)?,F(xiàn)在在內(nèi)存中已經(jīng)有這個(gè)照片了,而且uri也知道,所以就很容易了,看代碼:
@Override protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { if (requestCode == 1) { if (resultCode == RESULT_OK) try { Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); imageView.setImageBitmap(bitmap); } catch (FileNotFoundException e) { e.printStackTrace(); } } }
我們剛才使用startActivityForResult來(lái)啟動(dòng)活動(dòng)的,所以就要重寫這個(gè)方法來(lái)顯示圖片了。這里首先判斷是哪個(gè)啟動(dòng)命令,然后再判斷是否成功啟動(dòng),再BitmapFactory.decodeStream這個(gè)方法來(lái)獲取bitmap,再把bitmap顯示出來(lái)就行了。BitmapFactory.decodeStream這個(gè)方法需要一個(gè)流,可以通過(guò)getContentResolver().openInputStream這個(gè)方法來(lái)開(kāi)啟一個(gè)流。
到此整個(gè)流程就解決了。
FileProvider
FileProvider是一個(gè)特殊的內(nèi)容提供器,可以把一個(gè)file開(kāi)頭的uri改成content開(kāi)頭的,例如:file://uri -> content://uri。那為什么要這么做呢?這里簡(jiǎn)單講一下:
這個(gè)是因?yàn)樵贏ndroid 7.0之后,官方禁止直接把一個(gè)真實(shí)路徑的uri傳輸?shù)絼e的應(yīng)用,而我們要把地址送給相機(jī),所以就會(huì)出現(xiàn)問(wèn)題了。詳細(xì)可以查閱這篇博客:Android 7.0 行為變更 通過(guò)FileProvider在應(yīng)用間共享文件吧
既然是內(nèi)容提供器那么肯定是要進(jìn)行注冊(cè)的:
<provider android:name="android.support.v4.content.FileProvider" android:authorities="huan" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider>
這里的authorities參數(shù)必須和前面的getUriForFile方法的第二個(gè)參數(shù)保持一致,同個(gè)內(nèi)容提供器的authorities肯定要一樣啦。grantUriPermissions參數(shù)一定要是true,這個(gè)的大概意思就是給他的所有元素授權(quán)可以被訪問(wèn),在FileProvider中這個(gè)參數(shù)必須是true(這也是為什么在4.4一下版本的安卓無(wú)法使用的原因之一,有興趣可以去了解一下)export這個(gè)參數(shù)表示可不可以給其他的應(yīng)用共享,這里要設(shè)置為false。<meta-data這個(gè)是配置我們可以訪問(wèn)的文件路徑,@xml/file_paths這個(gè)就是表示什么文件可以被訪問(wèn),當(dāng)然要建一個(gè)這個(gè)文件??创a:
<?xml version="1.0" encoding="utf-8"?> <paths xmlns:android="http://schemas.android.com/apk/res/android"> <external-path name="my_image" path="/"/> </paths>
<external-path這個(gè)就是表示可以被訪問(wèn)的路徑,name是后面映射用到的,可以自己隨便起,我這里用一橫桿表示整個(gè)目錄可以被訪問(wèn)。
小結(jié)
調(diào)用系統(tǒng)相機(jī)的功能雖然不難,代碼也不多,但是其中的零碎知識(shí)很多,零零散散,還是要注意的。特別是關(guān)于低高配的安卓版本問(wèn)題還是要特別注意一下。
以上就是Android使用系統(tǒng)相機(jī)進(jìn)行拍照的步驟的詳細(xì)內(nèi)容,更多關(guān)于Android 適用系統(tǒng)相機(jī)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
使用Kotlin開(kāi)發(fā)Android應(yīng)用的初體驗(yàn)
本篇文章主要介紹了使用Kotlin開(kāi)發(fā)Android應(yīng)用的初體驗(yàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05Android性能優(yōu)化之利用Rxlifecycle解決RxJava內(nèi)存泄漏詳解
RxJava作為一種響應(yīng)式編程框架,是目前編程界網(wǎng)紅,可謂是家喻戶曉,其簡(jiǎn)潔的編碼風(fēng)格、易用易讀的鏈?zhǔn)椒椒ㄕ{(diào)用、強(qiáng)大的異步支持等使得RxJava被廣泛使用。2017-01-01Android 點(diǎn)擊editview以外位置實(shí)現(xiàn)隱藏輸入法
這篇文章主要介紹了Android 點(diǎn)擊editview以外位置實(shí)現(xiàn)隱藏輸入法的相關(guān)資料,需要的朋友可以參考下2017-06-06Android利用GridView實(shí)現(xiàn)單選功能
這篇文章主要為大家詳細(xì)介紹了Android利用GridView實(shí)現(xiàn)單選功能的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02Android實(shí)現(xiàn)簡(jiǎn)單的下拉刷新控件
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)簡(jiǎn)單的下拉刷新控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-09-09Jetpack Compose按鈕組件使用實(shí)例詳細(xì)講解
這篇文章主要介紹了Jetpack Compose按鈕組件使用實(shí)例,按鈕組件Button是用戶和系統(tǒng)交互的重要組件之一,它按照Material Design風(fēng)格實(shí)現(xiàn),我們先看下Button的參數(shù)列表,通過(guò)參數(shù)列表了解下Button的整體功能2023-04-04Android實(shí)用小技巧之利用Lifecycle寫出更好維護(hù)的代碼
lifecycle是一個(gè)類,用于存儲(chǔ)有關(guān)組件(如Activity或Fragment)的生命周期狀態(tài)的信息,并允許其他對(duì)象觀察此狀態(tài),下面這篇文章主要給大家介紹了關(guān)于Android實(shí)用小技巧之利用Lifecycle寫出更好維護(hù)的代碼的相關(guān)資料,需要的朋友可以參考下2022-05-05Android?Navigation重建Fragment問(wèn)題分析及解決
這篇文章主要介紹了Android?Navigation重建Fragment問(wèn)題分析及解決,文章通過(guò)圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09詳解androidstudio項(xiàng)目上傳到github方法以及步驟
在使用studio開(kāi)發(fā)的項(xiàng)目過(guò)程中有時(shí)候我們想將項(xiàng)目發(fā)布到github上,studio其實(shí)是自帶這種功能的,那么如何使用呢,下面我們就一起來(lái)了解一下2019-01-01