圖片分析用途可為您的應用程式提供 CPU 方便存取的圖片,以供執行圖片處理作業、電腦視覺或機器學習推論。應用程式會導入在每個影格上執行的 analyze()
方法。
如要瞭解如何整合 Google 的 ML Kit 與 CameraX 應用程式,請參閱「ML Kit 分析工具」一文。
操作模式
如果應用程式的分析管道無法達到 CameraX 的畫面更新率要求,您可以透過下列任一種方式將 CameraX 設定為捨棄影格:
非封鎖 (預設):在此模式下,應用程式分析上一張圖片時,執行程式一律會將最新的圖片快取至圖片緩衝區 (類似深度為一的佇列)。如果 CameraX 在應用程式處理完成之前收到新的圖片,系統會將其儲存至同一個緩衝區,並覆寫先前的圖片。請注意,在此情況下,
ImageAnalysis.Builder.setImageQueueDepth()
不會有任何作用,而緩衝區內容一律會被覆寫。您可以利用STRATEGY_KEEP_ONLY_LATEST
呼叫setBackpressureStrategy()
來啟動此非封鎖模式。如要進一步瞭解執行程式的影響,請參閱STRATEGY_KEEP_ONLY_LATEST
的參考說明文件。封鎖:在此模式下,內部執行程式可將多張圖片加入內部圖片佇列,並且僅在佇列已滿時才捨棄影格。封鎖模式適用於整個相機裝置範圍:如果相機裝置有多個已繫結的用途,則在 CameraX 處理這些圖片時,這些用途均會遭到封鎖。舉例來說,如果預覽和圖片分析均繫結至「相機」裝置,那麼當 CameraX 處理圖片時,預覽也會一併遭到封鎖。將
STRATEGY_BLOCK_PRODUCER
傳遞至setBackpressureStrategy()
,即可啟用封鎖模式。您也可以使用 ImageAnalysis.Builder.setImageQueueDepth() 設定圖片佇列深度。
利用低延遲和高效能分析工具,分析圖片的總時間少於 CameraX 影格的時間 (例如 60fps 為 16 毫秒),任何操作模式都能讓整體運作更加順暢。封鎖模式在某些情況下仍是很實用的,例如處理非常短的系統時基誤差。
利用高延遲和高效能分析工具,需要較長佇列的封鎖模式才能彌補延遲時間。但請注意,應用程式仍可以處理所有影格。
搭配使用高延遲和耗時的分析工具時 (分析工具無法處理所有影格),非封鎖模式可能會是更好的選擇,因為分析路徑必須捨棄影格,但其他並行的繫結用途仍能看到所有影格。
導入作業
如要在應用程式中使用圖片分析,請按照下列步驟操作:
- 建構
ImageAnalysis
用途。 - 建立
ImageAnalysis.Analyzer
。 - 將分析工具設定 至
ImageAnalysis
。 - 將生命週期擁有者、相機選取器和
ImageAnalysis
用途繫結至生命週期。
繫結後,CameraX 會立即將圖片傳送至已註冊的分析工具。完成分析後,請呼叫 ImageAnalysis.clearAnalyzer()
或解除 ImageAnalysis
用途,以便停止分析。
建構 ImageAnalysis 用途
ImageAnalysis
將您的分析工具 (圖片使用者) 連結至 CameraX,也就是圖片製作工具。應用程式可以使用 ImageAnalysis.Builder
建構 ImageAnalysis
物件。透過 ImageAnalysis.Builder
,應用程式可以設定以下項目:
- 圖片輸出參數:
- 格式:CameraX 透過
setOutputImageFormat(int)
支援YUV_420_888
和RGBA_8888
。預設格式為YUV_420_888
。 - Resolution 和 AspectRatio:您可以設定其中任一參數,不過請注意,您無法同時設定這兩個值。
- 旋轉。
- 目標名稱:使用此參數進行偵錯。
- 格式:CameraX 透過
- 圖片流程控制項:
應用程式可以設定解析度或長寬比,但不能同時設定兩者。確切的輸出解析度會因應用程式要求的大小 (或長寬比) 和硬體功能而異,且可能會與要求的大小或比例不同。如需解析度比對演算法的相關資訊,請參閱 setTargetResolution()
的說明文件
應用程式可將輸出圖片像素設為 YUV (預設) 或 RGBA 色域。設定 RGBA 輸出格式時,CameraX 內部會將圖片從 YUV 轉換為 RGBA 色域,並依照下列順序將圖片位元封裝為 ImageProxy 第一平面的 ByteBuffer
(未使用其他兩個平面):
ImageProxy.getPlanes()[0].buffer[0]: alpha
ImageProxy.getPlanes()[0].buffer[1]: red
ImageProxy.getPlanes()[0].buffer[2]: green
ImageProxy.getPlanes()[0].buffer[3]: blue
...
執行複雜的圖片分析作業時,如果裝置無法達到畫面更新率,您可以按照本文中操作模式一節所述的策略,將 CameraX 設定為捨棄影格。
建立分析工具
應用程式透過導入 ImageAnalysis.Analyzer
介面並覆寫 analyze(ImageProxy image)
來建立分析工具。在每個分析工具中,應用程式會收到 ImageProxy
,這是 Media.Image 的包裝函式。您可以使用 ImageProxy.getFormat()
查詢圖片格式。格式是應用程式透過 ImageAnalysis.Builder
提供的下列其中一個值:
ImageFormat.RGBA_8888
(如果應用程式要求OUTPUT_IMAGE_FORMAT_RGBA_8888
)。ImageFormat.YUV_420_888
(如果應用程式要求OUTPUT_IMAGE_FORMAT_YUV_420_888
)。
請參閱建構 ImageAnalysis 用途來瞭解色域設定,以及可以擷取像素位元組的位置。
在分析工具中,應用程式應執行以下操作:
- 最好在特定的畫面更新率時間限制內 (例如 30 fps 時不超過 32 毫秒),盡快分析特定影格。如果應用程式無法快速分析影格,請考慮使用其中一種支援的影格捨棄機制。
- 呼叫
ImageProxy.close()
即可將ImageProxy
發布至 CameraX。請注意,您不應呼叫已包裝的 Media.Image 關閉函式 (Media.Image.close()
)。
應用程式可以直接在 ImageProxy 中使用已包裝的 Media.Image
。請勿呼叫已包裝圖片上的 Media.Image.close()
,因為這會導致 CameraX 中的圖片分享機制中斷;請改用 ImageProxy.close()
將基礎 Media.Image
發布至 CameraX。
設定 ImageAnalysis 分析工具
建立分析工具後,請使用 ImageAnalysis.setAnalyzer()
註冊並開始分析。分析完成後,請使用 ImageAnalysis.clearAnalyzer()
移除已註冊的分析工具。
只能設定一個有效的分析工具進行圖片分析。呼叫 ImageAnalysis.setAnalyzer()
會取代註冊的分析工具 (如果已存在的話)。在繫結用途前後,應用程式可以隨時設定新的分析工具。
將 ImageAnalysis 繫結至生命週期
強烈建議您使用 ProcessCameraProvider.bindToLifecycle()
函式將 ImageAnalysis
繫結至現有的 AndroidX 生命週期。請注意,bindToLifecycle()
函式會回傳所選 Camera
裝置,該裝置可用於微調曝光等進階設定。請參閱這份指南,進一步瞭解如何控制相機輸出。
以下範例組合了前述步驟中的所有內容,將 CameraX ImageAnalysis
和 Preview
用途繫結至 lifeCycle
擁有者:
Kotlin
val imageAnalysis = ImageAnalysis.Builder() // enable the following line if RGBA output is needed. // .setOutputImageFormat(ImageAnalysis.OUTPUT_IMAGE_FORMAT_RGBA_8888) .setTargetResolution(Size(1280, 720)) .setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST) .build() imageAnalysis.setAnalyzer(executor, ImageAnalysis.Analyzer { imageProxy -> val rotationDegrees = imageProxy.imageInfo.rotationDegrees // insert your code here. ... // after done, release the ImageProxy object imageProxy.close() }) cameraProvider.bindToLifecycle(this as LifecycleOwner, cameraSelector, imageAnalysis, preview)
Java
ImageAnalysis imageAnalysis = new ImageAnalysis.Builder() // enable the following line if RGBA output is needed. //.setOutputImageFormat(ImageAnalysis.OUTPUT_IMAGE_FORMAT_RGBA_8888) .setTargetResolution(new Size(1280, 720)) .setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST) .build(); imageAnalysis.setAnalyzer(executor, new ImageAnalysis.Analyzer() { @Override public void analyze(@NonNull ImageProxy imageProxy) { int rotationDegrees = imageProxy.getImageInfo().getRotationDegrees(); // insert your code here. ... // after done, release the ImageProxy object imageProxy.close(); } }); cameraProvider.bindToLifecycle((LifecycleOwner) this, cameraSelector, imageAnalysis, preview);
其他資源
如要進一步瞭解 CameraX,請參閱下列其他資源。
程式碼研究室
程式碼範例