میتوانید از فید دوربینی که ARCore در خط لوله یادگیری ماشین ضبط میکند برای ایجاد یک تجربه واقعیت افزوده هوشمند استفاده کنید. نمونه ARCore ML Kit نحوه استفاده از ML Kit و Google Cloud Vision API را برای شناسایی اشیاء دنیای واقعی نشان می دهد. نمونه از یک مدل یادگیری ماشینی برای طبقه بندی اشیاء در دید دوربین استفاده می کند و یک برچسب به شی در صحنه مجازی می چسباند.
نمونه کیت ARCore ML به زبان Kotlin نوشته شده است. همچنین به عنوان برنامه نمونه ml_kotlin در مخزن ARCore SDK GitHub موجود است.
از تصویر CPU ARCore استفاده کنید
ARCore حداقل دو مجموعه از جریان های تصویر را به طور پیش فرض می گیرد:
- یک جریان تصویر CPU که برای تشخیص ویژگی و پردازش تصویر استفاده می شود. به طور پیش فرض، تصویر CPU دارای وضوح VGA (640x480) است. ARCore را می توان برای استفاده از جریان تصویر با وضوح بالاتر در صورت لزوم پیکربندی کرد.
- یک جریان بافت GPU ، که شامل یک بافت با وضوح بالا، معمولا با وضوح 1080p است. این معمولاً به عنوان پیش نمایش دوربین رو به روی کاربر استفاده می شود. این در بافت OpenGL که توسط
Session.setCameraTextureName()
مشخص شده است ذخیره می شود. - هر جریان اضافی مشخص شده توسط
SharedCamera.setAppSurfaces()
.
ملاحظات اندازه تصویر CPU
در صورت استفاده از جریان پیشفرض CPU با اندازه VGA، هزینه اضافی متحمل نمیشود زیرا ARCore از این جریان برای درک جهان استفاده میکند. درخواست یک جریان با وضوح متفاوت ممکن است گران باشد، زیرا یک جریان اضافی باید ضبط شود. به خاطر داشته باشید که رزولوشن بالاتر ممکن است به سرعت برای مدل شما گران شود: دوبرابر کردن عرض و ارتفاع تصویر، تعداد پیکسل های تصویر را چهار برابر می کند.
اگر مدل شما هنوز هم میتواند روی یک تصویر با وضوح پایینتر عملکرد خوبی داشته باشد، کاهش مقیاس تصویر ممکن است سودمند باشد.
یک جریان تصویر CPU با وضوح بالا اضافی را پیکربندی کنید
عملکرد مدل ML شما ممکن است به وضوح تصویر مورد استفاده به عنوان ورودی بستگی داشته باشد. وضوح این جریان ها را می توان با تغییر CameraConfig
فعلی با استفاده از Session.setCameraConfig()
تنظیم کرد و یک پیکربندی معتبر را از Session.getSupportedCameraConfigs()
انتخاب کرد.
جاوا
CameraConfigFilter cameraConfigFilter = new CameraConfigFilter(session) // World-facing cameras only. .setFacingDirection(CameraConfig.FacingDirection.BACK); List<CameraConfig> supportedCameraConfigs = session.getSupportedCameraConfigs(cameraConfigFilter); // Select an acceptable configuration from supportedCameraConfigs. CameraConfig cameraConfig = selectCameraConfig(supportedCameraConfigs); session.setCameraConfig(cameraConfig);
کاتلین
val cameraConfigFilter = CameraConfigFilter(session) // World-facing cameras only. .setFacingDirection(CameraConfig.FacingDirection.BACK) val supportedCameraConfigs = session.getSupportedCameraConfigs(cameraConfigFilter) // Select an acceptable configuration from supportedCameraConfigs. val cameraConfig = selectCameraConfig(supportedCameraConfigs) session.setCameraConfig(cameraConfig)
تصویر CPU را بازیابی کنید
تصویر CPU را با استفاده از Frame.acquireCameraImage()
بازیابی کنید. این تصاویر باید به محض اینکه دیگر مورد نیاز نیستند دور ریخته شوند.
جاوا
Image cameraImage = null; try { cameraImage = frame.acquireCameraImage(); // Process `cameraImage` using your ML inference model. } catch (NotYetAvailableException e) { // NotYetAvailableException is an exception that can be expected when the camera is not ready // yet. The image may become available on a next frame. } catch (RuntimeException e) { // A different exception occurred, e.g. DeadlineExceededException, ResourceExhaustedException. // Handle this error appropriately. handleAcquireCameraImageFailure(e); } finally { if (cameraImage != null) { cameraImage.close(); } }
کاتلین
// NotYetAvailableException is an exception that can be expected when the camera is not ready yet. // Map it to `null` instead, but continue to propagate other errors. fun Frame.tryAcquireCameraImage() = try { acquireCameraImage() } catch (e: NotYetAvailableException) { null } catch (e: RuntimeException) { // A different exception occurred, e.g. DeadlineExceededException, ResourceExhaustedException. // Handle this error appropriately. handleAcquireCameraImageFailure(e) } // The `use` block ensures the camera image is disposed of after use. frame.tryAcquireCameraImage()?.use { image -> // Process `image` using your ML inference model. }
تصویر CPU را پردازش کنید
برای پردازش تصویر CPU، می توان از کتابخانه های مختلف یادگیری ماشین استفاده کرد.
- ML Kit : ML Kit یک API تشخیص و ردیابی اشیاء روی دستگاه ارائه می دهد. این با یک طبقهبندی درشت درون API عرضه میشود و همچنین میتواند از مدلهای طبقهبندی سفارشی برای پوشش دامنه باریکتری از اشیا استفاده کند . از
InputImage.fromMediaImage
برای تبدیل تصویر CPU خود بهInputImage
استفاده کنید. - Firebase Machine Learning : Firebase API های یادگیری ماشینی را ارائه می دهد که در ابر یا روی دستگاه کار می کنند. مستندات Firebase را در Label Images Securely with Cloud Vision با استفاده از Firebase Auth و Functions در Android ببینید.
نتایج را در صحنه AR خود نمایش دهید
مدل های تشخیص تصویر اغلب اشیاء شناسایی شده را با نشان دادن یک نقطه مرکزی یا یک چندضلعی مرزی که نشان دهنده شی شناسایی شده است، خروجی می دهند.
با استفاده از نقطه مرکزی یا مرکز جعبه مرزی که از مدل خروجی می شود، می توان یک لنگر را به شی شناسایی شده متصل کرد. از Frame.hitTest()
برای تخمین وضعیت یک شی در صحنه مجازی استفاده کنید.
تبدیل مختصات IMAGE_PIXELS
به مختصات VIEW
:
جاوا
// Suppose `mlResult` contains an (x, y) of a given point on the CPU image. float[] cpuCoordinates = new float[] {mlResult.getX(), mlResult.getY()}; float[] viewCoordinates = new float[2]; frame.transformCoordinates2d( Coordinates2d.IMAGE_PIXELS, cpuCoordinates, Coordinates2d.VIEW, viewCoordinates); // `viewCoordinates` now contains coordinates suitable for hit testing.
کاتلین
// Suppose `mlResult` contains an (x, y) of a given point on the CPU image. val cpuCoordinates = floatArrayOf(mlResult.x, mlResult.y) val viewCoordinates = FloatArray(2) frame.transformCoordinates2d( Coordinates2d.IMAGE_PIXELS, cpuCoordinates, Coordinates2d.VIEW, viewCoordinates ) // `viewCoordinates` now contains coordinates suitable for hit testing.
از این مختصات VIEW
برای انجام تست ضربه و ایجاد یک لنگر از نتیجه استفاده کنید:
جاوا
List<HitResult> hits = frame.hitTest(viewCoordinates[0], viewCoordinates[1]); HitResult depthPointResult = null; for (HitResult hit : hits) { if (hit.getTrackable() instanceof DepthPoint) { depthPointResult = hit; break; } } if (depthPointResult != null) { Anchor anchor = depthPointResult.getTrackable().createAnchor(depthPointResult.getHitPose()); // This anchor will be attached to the scene with stable tracking. // It can be used as a position for a virtual object, with a rotation prependicular to the // estimated surface normal. }
کاتلین
val hits = frame.hitTest(viewCoordinates[0], viewCoordinates[1]) val depthPointResult = hits.filter { it.trackable is DepthPoint }.firstOrNull() if (depthPointResult != null) { val anchor = depthPointResult.trackable.createAnchor(depthPointResult.hitPose) // This anchor will be attached to the scene with stable tracking. // It can be used as a position for a virtual object, with a rotation prependicular to the // estimated surface normal. }
ملاحظات عملکرد
برای صرفه جویی در قدرت پردازش و مصرف انرژی کمتر، توصیه های زیر را دنبال کنید:
- مدل ML خود را روی هر فریم ورودی اجرا نکنید. به جای آن، تشخیص شی را با نرخ فریم پایین اجرا کنید.
- برای کاهش پیچیدگی محاسباتی، یک مدل استنتاج ML آنلاین را در نظر بگیرید.
مراحل بعدی
- درباره بهترین روش ها برای مهندسی ML بیاموزید.
- درباره شیوههای هوش مصنوعی مسئول بیاموزید.
- اصول یادگیری ماشین را با دوره TensorFlow دنبال کنید.