当前位置:首页 > 实用技巧 >

安卓手机上的涂鸦怎么用(安卓涂鸦怎么看原图)

来源:原点资讯(www.yd166.com)时间:2023-11-08 01:01:18作者:YD166手机阅读>>

本文适合有 Java 基础的人群

安卓手机上的涂鸦怎么用,安卓涂鸦怎么看原图(1)

作者:DJL-Lanking

HelloGitHub 推出的《讲解开源项目》系列。有幸邀请到了亚马逊 Apache 的工程师:Lanking( https://github.com/lanking520 ),为我们讲解 DJL —— 完全由 Java 构建的深度学习平台,本文为系列的第三篇。

一、前言

在 2018 年时,Google 推出了《猜画小歌》应用:玩家可以直接与 AI 进行你画我猜的游戏。通过画出一个房子或者一个猫,AI 会推断出各种物品被画出的概率。它的实现得益于深度学习模型在其中的应用,通过深度神经网络的归纳,曾经令人头疼的绘画识别也变得易如反掌。现如今,只要使用一个简单的图片分类模型,我们便可以轻松的实现绘画识别。试试看这个在线涂鸦小游戏吧:

在线涂鸦小游戏:https://djl.ai/website/demo.html#doodle

在当时,大部分机器学习计算任务仍旧需要依托网络在云端进行。随着算力的不断增进,机器学习任务已经可以直接在边缘设备部署,包括各类运行安卓系统的智能手机。但是,由于安卓本身主要是用 Java ,部署基于 Python 的各类深度学习模型变成了一个难题。为了解决这个问题,AWS 开发并开源了 DeepJavaLibrary (DJL),一个为 Java 量身定制的深度学习框架。

在这个文章中,我们将尝试通过 PyTorch 预训练模型在安卓平台构建一个涂鸦绘画的应用。由于总代码量会比较多,我们这次会挑重点把最关键的代码完成。你可以后续参考我们完整的项目进行构建。

涂鸦应用完整代码:https://github.com/aws-samples/djl-demo/tree/master/android

二、环境配置

为了兼容 DJL 需求的 Java 功能,这个项目需要 Android API 26 及以上的版本。你可以参考我们案例配置来节约一些时间,下面是这个项目需要的依赖项:

案例 gradle: https://github.com/aws-samples/djl-demo/blob/master/android/quickdraw_recognition/build.gradle

dependencies{ implementation'androidx.appcompat:appcompat:1.2.0' implementation'ai.djl:api:0.7.0' implementation'ai.djl.android:core:0.7.0' runtimeOnly'ai.djl.pytorch:pytorch-engine:0.7.0' runtimeOnly'ai.djl.android:pytorch-native:0.7.0' }

我们将使用 DJL 提供的 API 以及 PyTorch 包。

三、构建应用3.1 第一步:创建 Layout

我们可以先创建一个 View class 以及 layout(如下图)来构建安卓的前端显示界面。

安卓手机上的涂鸦怎么用,安卓涂鸦怎么看原图(2)

如上图所示,你可以在主界面创建两个 View 目标。PaintView 是用来让用户画画的,在右下角 ImageView 是用来展示用于深度学习推理的图像。同时我们预留一个按钮来进行画板的清空操作。

3.2 第二步: 应对绘画动作

在安卓设备上,你可以自定义安卓的触摸事件响应来应对用户的各种触控操作。在我们的情况下,我们需要定义下面三种时间响应:

  • touchStart:感应触碰时触发
  • touchMove:当用户在屏幕上移动手指时触发
  • touchUp:当用户抬起手指时触发

与此同时,我们用 paths 来存储用户在画板所绘制的路径。现在我们看一下实现代码。

3.2.1 重写 OnTouchEvent 和 OnDraw 方法

现在我们重写 onTouchEvent 来应对各种响应:

@Override publicbooleanonTouchEvent(MotionEventevent){ floatx=event.getX(); floaty=event.getY(); switch(event.getAction()){ caseMotionEvent.ACTION_DOWN: touchStart(x,y); invalidate(); break; caseMotionEvent.ACTION_MOVE: touchMove(x,y); invalidate(); break; caseMotionEvent.ACTION_UP: touchUp(); runInference(); invalidate(); break; } returntrue; }

如上面代码所示,你可以添加一个 runInference 方法在 MotionEvent.ACTION_UP 事件响应上。这个方法是用来在用户绘制完后对结果进行推理。在之后的几步中,我们会讲解它的具体实现。

我们同样需要重写 onDraw 方法来展示用户绘制的图像:

@Override protectedvoidonDraw(canvascanvas){ canvas.save(); this.canvas.drawColor(DEFAULT_BG_COLOR); for(Pathpath:paths){ paint.setColor(DEFAULT_PAINT_COLOR); paint.setStrokeWidth(BRUSH_SIZE); this.canvas.drawPath(path,paint); } canvas.drawBitmap(bitmap,0,0,bitmapPaint); canvas.restore(); }

真正的图像会保存在一个 Bitmap 上。

3.2.2 操作开始(touchStart)

当用户触碰行为开始时,下面的代码会建立一个新的路径同时记录路径中每一个点在屏幕上的坐标。

privatevoidtouchStart(floatx,floaty){ path=newPath(); paths.add(path); path.reset(); path.moveTo(x,y); this.x=x; this.y=y; } 3.2.3 手指移动(touchMove)

在手指移动中,我们会持续记录坐标点然后将它们构成一个 quadratic bezier. 通过一定的误差阀值来动态优化用户的绘画动作。只有差别超出误差范围内的动作才会被记录下来。

quadratic bezier 文档: https://developer.android.com/reference/android/graphics/Path

privatevoidtouchMove(floatx,floaty){ if(x<0||x>getWidth()||y<0||y>getHeight()){ return; } floatdx=Math.abs(x-this.x); floatdy=Math.abs(y-this.y); if(dx>=TOUCH_TOLERANCE||dy>=TOUCH_TOLERANCE){ path.quadTo(this.x,this.y,(x this.x)/2,(y this.y)/2); this.x=x; this.y=y; } } 3.2.4 操作结束(touchUp)

当触控操作结束后,下面的代码会绘制一个路径同时计算最小长方形目标框。

privatevoidtouchUp(){ path.lineTo(this.x,this.y); maxBound.add(newPath(path)); } 3.3 第三步:开始推理

为了在安卓设备上进行推理任务,我们需要完成下面几个任务:

  • 从 URL 读取模型
  • 构建前处理和后处理过程
  • 从 PaintView 进行推理任务

为了完成以下目标,我们尝试构建一个 DoodleModel class。在这一步,我们将介绍一些完成这些任务的关键步骤。

3.3.1 读取模型

DJL 内建了一套模型管理系统。开发者可以自定义储存模型的文件夹。

Filedir=getFilesDir(); System.setProperty("DJL_CACHE_DIR",dir.getAbsolutePath());

通过更改 DJL_CACHE_DIR 属性,模型会被存入相应路径下。

下一步可以通过定义 Criteria 从指定 URL 处下载模型。下载的 zip 文件内包含:

  • doodle_mobilenet.pt:PyTorch 模型
  • synset.txt:包含分类任务中所有类别的名称

Criteria<Image,Classifications>criteria= Criteria.builder() .setTypes(Image.class,Classifications.class) .optModelUrls("https://djl-ai.s3.amazonaws.com/resources/demo/pytorch/doodle_mobilenet.zip") .optTranslator(translator) .build(); returnModelZoo.loadModel(criteria);

上述代码同时定义了 translator,它会被用来做图片的前处理和后处理。

最后,如下述代码创建一个 Model 并用它创建一个 Predictor:

@Override protectedBooleandoInBackground(Void...params){ try{ model=DoodleModel.loadModel(); predictor=model.newPredictor(); returntrue; }catch(IOException|ModelExceptione){ Log.e("DoodleDraw",null,e); } returnfalse; }

更多关于模型加载的信息,请参阅如何加载模型。

DJL 模型加载文档:http://docs.djl.ai/docs/load_model.html

3.3.2 用 Translator 定义前处理和后处理

在 DJL 中,我们定义了 Translator 接口进行前处理和后处理。在 DoodleModel 中我们定义了 ImageClassificationTranslator 来实现 Translator:

ImageClassificationTranslator.builder() .addTransform(newToTensor()) .optFlag(Image.Flag.GRAYSCALE) .optApplySoftmax(true).build());

下面我们详细阐述 translator 所定义的前处理和后处理如何被用在模型的推理步骤中。当你创建 translator 时,内部程序会自动加载 synset.txt 文件得到做分类任务时所有类别的名称。当模型的 predict() 方法被调用时,内部程序会先执行所对应的 translator 的前处理步骤,而后执行实际推理步骤,最后执行 translator 的后处理步骤。对于前处理,我们会将 Image 转化 NDArray,用于作为模型推理过程的输入。对于后处理,我们对推理输出的结果(NDArray)进行 softmax 操作。最终返回结果为 Classifications 的一个实例。

自定义 Translator 案例:http://docs.djl.ai/jupyter/pytorch/load_your_own_pytorch_bert.html

3.3.3 用 PaintView 进行推理任务

最后,我们来实现之前定义好的 runInference 方法。

publicvoidrunInference(){ //拷贝图像 Bitmapbmp=Bitmap.createBitmap(bitmap); //缩放图像 bmp=Bitmap.createScaledBitmap(bmp,64,64,true); //执行推理任务 Classificationsclassifications=model.predict(bmp); //展示输入的图像 Bitmappresent=Bitmap.createScaledBitmap(bmp,imageView.getWidth(),imageView.getHeight(),true); imageView.setImageBitmap(present); //展示输出的图像 if(messageToast!=null){ messageToast.cancel(); } messageToast=Toast.makeText(getContext(),classifications.toString(),Toast.LENGTH_SHORT); messageToast.show(); }

这将会创建一个 Toast 弹出页面用于展示结果,示例如下:

安卓手机上的涂鸦怎么用,安卓涂鸦怎么看原图(3)

恭喜你!我们完成了一个涂鸦识别小程序!

3.4 可选优化:输入裁剪

为了得到更高的模型推理准确度,你可以通过截取图像来去除无意义的边框部分。

安卓手机上的涂鸦怎么用,安卓涂鸦怎么看原图(4)

上面右侧的图片会比左边的图片有更好的推理结果,因为它所包含的空白边框更少。你可以通过 Bound 类来寻找图片的有效边界,即能把图中所有白色像素点覆盖的最小矩形。在得到 x 轴最左坐标,y 轴最上坐标,以及矩形高度和宽度后,就可以用这些信息截取出我们想要的图形(如右图所示)实现代码如下:

RectFbound=maxBound.getBound(); intx=(int)bound.left; inty=(int)bound.top; intwidth=(int)Math.ceil(bound.width()); intheight=(int)Math.ceil(bound.height()); //截取部分图像 Bitmapbmp=Bitmap.createBitmap(bitmap,x,y,width,height);

恭喜你!现在你就掌握了全部教程内容!期待看到你创建的第一个 DoodleDraw 安卓游戏!

最后,可以在GitHub找到本教程的完整案例代码。

涂鸦应用完整代码:https://github.com/aws-samples/djl-demo/tree/master/android

关于 DJL

安卓手机上的涂鸦怎么用,安卓涂鸦怎么看原图(5)

Deep Java Library (DJL) 是一个基于 Java 的深度学习框架,同时支持训练以及推理。DJL 博取众长,构建在多个深度学习框架之上 (TenserFlow、PyTorch、MXNet 等) 也同时具备多个框架的优良特性。你可以轻松使用 DJL 来进行训练然后部署你的模型。

它同时拥有着强大的模型库支持:只需一行便可以轻松读取各种预训练的模型。现在 DJL 的模型库同时支持高达 70 个来自 GluonCV、 HuggingFace、TorchHub 以及 Keras 的模型。

项目地址:https://github.com/awslabs/djl/

栏目热文

安卓手机怎么找涂鸦(安卓涂鸦怎么看原图)

安卓手机怎么找涂鸦(安卓涂鸦怎么看原图)

现在打开B612咔叽相机选择《第五人格》游戏专属贴图“庄园魅影”草帽 麻花辫 纽扣眼 手套不管你是男是女只要对准摄像头园...

2023-11-08 01:15:04查看全文 >>

手机涂鸦从哪里打开(手机上的涂鸦怎么弄)

手机涂鸦从哪里打开(手机上的涂鸦怎么弄)

相信大家在给视频进行处理时,都会运用一些视频剪辑工具来给视频增添一些涂鸦等效果,比如把云朵涂鸦成独角兽或者与其相似的图案...

2023-11-08 01:26:42查看全文 >>

丰田花冠按键说明什么(丰田花冠车内功能按键介绍)

丰田花冠按键说明什么(丰田花冠车内功能按键介绍)

引言:很多人对日系车不屑一顾,在他们眼中日系车一无是处,质感差没驾驶感,车皮薄安全性差,总之日系车就不应该被生产制造出来...

2023-11-08 01:27:39查看全文 >>

丰田花冠挡位介绍(丰田花冠手动挡位图解)

丰田花冠挡位介绍(丰田花冠手动挡位图解)

01 通关前一转眼,已经自驾西藏行摄阿里第九天了,这几天的行程一直在318国道上。今天的行程是从波密到林芝地区八一镇。波...

2023-11-08 01:21:28查看全文 >>

2006款丰田花冠有几把钥匙(04年丰田花冠的钥匙都有什么功能)

2006款丰田花冠有几把钥匙(04年丰田花冠的钥匙都有什么功能)

哈喽大家好,检车家老司机又和大家见面了,我是检车家铁头。今天给大家分享一台丰田花冠的二手车检测案例。可能不少年轻的朋友对...

2023-11-08 00:58:03查看全文 >>

手机上的涂鸦怎么用(手机涂鸦有几种操作方法)

手机上的涂鸦怎么用(手机涂鸦有几种操作方法)

随着时代的发展,大众对于“艺术”这个词的诠释也会变得不同。我们在制作视频的时候,通常会在上面添加一些涂鸦来增强视频的趣味...

2023-11-08 01:28:34查看全文 >>

手机涂鸦怎么查看(手机自带的涂鸦在哪里)

手机涂鸦怎么查看(手机自带的涂鸦在哪里)

*本文来自花粉俱乐部论坛-在死海游泳的咸鱼今天给大家安利两招屏幕涂鸦的操作,不要怀疑,对!屏幕也可以涂鸦!不仅趣味慢慢而...

2023-11-08 01:33:14查看全文 >>

手机涂鸦怎么消失(手机怎么涂鸦)

手机涂鸦怎么消失(手机怎么涂鸦)

本文来自游研社APP社区用户投稿,如果你对自己的游戏经历或是见闻有撰写成文章的想法,也可以前往游研社APP投稿中心与我们...

2023-11-08 01:22:30查看全文 >>

车辆起步吱吱摩擦异响(车辆起步刹车会吱吱响怎么回事)

车辆起步吱吱摩擦异响(车辆起步刹车会吱吱响怎么回事)

车辆概况:雪佛兰科鲁泽 2019款,提车不到1年行驶1.2万公里,未出过重大交通事故。故障描述:车辆起步时,松开刹车会有...

2023-11-08 01:31:48查看全文 >>

文档排行