Android自定义View工具:Paint&Canvas(一)

本文涉及的产品
模型在线服务 PAI-EAS,A10/V100等 500元 1个月
交互式建模 PAI-DSW,每月250计算时 3个月
模型训练 PAI-DLC,100CU*H 3个月
简介: 本文主要讲的是自定义View时我们经常用到的Canvas和Paint,像平时画画一样,我们需要画布和画笔,而Canvas就是画布,Paint就是画笔

安卓的graphics提供了2D图形各种绘制工具,如Canvas(画布), color filters(颜色过滤器), points(点), rectangles(矩形)等,利用这些工具可以直接在界面上进行绘制。
本文主要讲的是自定义View时我们经常用到的Canvas和Paint,像平时画画一样,我们需要画布和画笔,而Canvas就是画布,Paint就是画笔.

Canvas官网地址:
https://developer.android.com/reference/android/graphics/Canvas.html
Paint官网地址:
https://developer.android.com/reference/android/graphics/Paint.html

先来看Paint,Paint常用方法一览:

Paint.setAntiAlias(boolean flag);//设置抗锯齿效果 设置true的话边缘会将锯齿模糊化
Paint.setDither(boolean flag);//设置防抖动,设置true的话图片看上去会更柔和点
Paint.setColor(int color);//设置画笔颜色
###TODO
Paint.setARGB(int a, int r, int g, int b); //设置画笔的ARGB值
Paint.setAlpha(int alpha);//设置画笔的Alpha值
Paint.setStyle(); //设置画笔的style (三种:FILL填充 FILL_AND_STROKE填充加描边 STROKE描边 )
Paint.setStrokeWidth(float width);//设置描边宽度

Paint.setXfermode(Xfermode xfermode);//设置图形重叠时的处理方式,如合并,取交集或并集,经常用来制作橡皮的擦除效果
Paint.setShader(Shader shader);//设置图像效果,使用Shader可以绘制出各种渐变效果   
Paint.setShadowLayer(float radius ,float dx,float dy,int color);//在图形下面设置阴影层,产生阴影效果,radius为阴影的半径,dx和dy为阴影在x轴和y轴上的距离,color为阴影的颜色  
//下面写文本的时候经常用到的
Paint.setTextSize(float textSize);//设置画笔文字大小
Paint.measureText(String text);//测试文本的长度
Paint.setTextAlign(Paint.Align align);// CENTER(文本居中) LEFT(文本左对齐) RIGHT(文本右对齐)

先来看上面Paint的几个主要方法,结合代码和效果图:

  • Paint.setStyle(); //设置画笔的style

     Paint.Style.FILL //填充 
     Paint.Style.FILL_AND_STROKE //填充加描边 
     Paint.Style.STROKE //描边 

    测试伪代码:

Paint mPaint= new Paint();
mPaint.setColor(Color.RED);//画笔颜色为红色
mPaint.setStrokeWidth(80); //描边宽度为80(为了区分效果,特意设置特别大)

float radius = 100f;
// 填充
mPaint.setStyle(Paint.Style.FILL);
canvas.drawCircle(400, 500, radius, mPaint);
// 描边
mPaint.setStyle(Paint.Style.STROKE);
canvas.drawCircle(400, 200, radius, mPaint);
// 描边加填充
mPaint.setStyle(Paint.Style.FILL_AND_STROKE);
canvas.drawCircle(400, 900, radius, mPaint);

  来看效果图:  

![Paint.Style.png](http://upload-images.jianshu.io/upload_images/587163-1c7e24d128c6e2bf.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
如图,根据测试,**设置FILL_AND_STROKE模式时在其圆的外围的描边宽度并不是StrokeWidth的宽度,而是StrokeWidth/2的宽度.**

* ####Paint.setShader(Shader shader)//设置图像效果
[Shader](https://developer.android.com/reference/android/graphics/Shader.html)是着色器,用来给图像着色,Shader 是基类基类,它有5个已知的子类:
[BitmapShader](https://developer.android.com/reference/android/graphics/BitmapShader.html),
 [ComposeShader](https://developer.android.com/reference/android/graphics/ComposeShader.html),
 [LinearGradient](https://developer.android.com/reference/android/graphics/LinearGradient.html), 
[RadialGradient](https://developer.android.com/reference/android/graphics/RadialGradient.html),
 [SweepGradient](https://developer.android.com/reference/android/graphics/SweepGradient.html)

在讲这5个子类之前,先了解一个枚举Shader.TileMode,它里面有三个值:{CLAMP,REPEAT,MIRROR}:
######Shader.TileMode.CLAMP:
如果shader绘制范围大于原有的范围时,会用原有图像四边的颜色填充剩余空间。
######Shader.TileMode.REPEAT:
在水平和竖直方向重复shader图像。
######Shader.TileMode.MIRROR:
在水平和竖直方向重复shader图像,这一点和REPEAT相似,不同的是MIRROR模式下相邻的两个图像互为镜像。
接下来结合例子分别来看一下Shader的5个子类和Shader.TileMode的使用姿势。

先来看下原图(**用我家两只猫咪镇楼!**)

![cat.png](http://upload-images.jianshu.io/upload_images/587163-efb85e77723ed2b4.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
#####BitmapShader:
BitmapShader本质上就是绘制一个bitmap,并用这个bitmap对需要绘制的图形进行填充。

BitmapShader(Bitmap bitmap, Shader.TileMode tileX, Shader.TileMode tileY)

BitmapShader初始化时的三个参数:

BitmapShader参数| 备注
:----:|:----:
bitmap| 用来填充图形的Bitmap 
tileX| X轴Bitmap用Shader.TileMode模式填充
tileY| Y轴Bitmap用Shader.TileMode模式填充

示例:

BitmapShader shader = new BitmapShader(bitmap, BitmapShader.TileMode.CLAMP, BitmapShader.TileMode.MIRROR);
mPaint.setShader(shader);
canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

效果图:

![catb.png](http://upload-images.jianshu.io/upload_images/587163-f26bbac2ac0642fb.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
X轴用TileMode.CLAMP模式,即用bitmap的右边缘去填充X轴其余空间,
Y轴用TileMode.MIRROR模式,即在用相邻两张图像互为镜像的方式填充整个Y轴其余空间。
X轴和Y轴分别换一下参数模式:

BitmapShader shader = new BitmapShader(bitmap, BitmapShader.TileMode.MIRROR, BitmapShader.TileMode.REPEAT);
mPaint.setShader(shader);
canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

效果图:
![cat2.png](http://upload-images.jianshu.io/upload_images/587163-17349b18c6f3e163.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

X轴用TileMode.MIRROR模式,即用相邻两张图像互为镜像的方式填充整个X轴其余空间,
Y轴用TileMode.REPEAT模式,即用相同的图像重复填充整个Y轴其余空间。
#####LinearGradient:

LinearGradient(float x0, float y0, float x1, float y1, int color0, int color1,TileMode tile)

LinearGradient是沿一条直线用来创建线性渐变效果,(x0,y0),(x1,y1)分别是起始坐标和终止坐标,color0,color1分别是起始颜色和终止颜色,tile为
Shader.TileMode(CLAMP,REPEAT,MIRROR)模式中的一个。

LinearGradient参数| 备注
:----:|:----:
x0| 渐变线起始坐标的X坐标 
y0| 渐变线起始坐标的Y坐标 
x1| 渐变线终止坐标的X坐标 
y1| 渐变线终止坐标的Y坐标
color0| 渐变线起始颜色
color1| 渐变线终止颜色
tile| 渐变线用Shader.TileMode模式填充
示例:

LinearGradient linearGradient = new LinearGradient(200, 200, 600, 600, Color.GREEN, Color.YELLOW, Shader.TileMode.MIRROR);
mPaint.setShader(linearGradient);
canvas.drawRect(200, 200, 600, 600, mPaint);

效果图:
![LinearGradient.png](http://upload-images.jianshu.io/upload_images/587163-cdf2b6a2fca6e528.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
下面修改一下代码,扩大一下绘制范围:

canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

并且分别在LinearGradient构造函数中设置Shader.TileMode为CLAMP,REPEAT,MIRROR:
效果图:

![threeMode.png](http://upload-images.jianshu.io/upload_images/587163-d31e20aaa68bf22d.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

LinearGradient还有另外一个构造函数:

LinearGradient(float x0, float y0, float x1, float y1, int colors[], float positions[],TileMode tile)

和第一个构造函数不同的是colors和positions,可以传多个color及对应的position进行线性渐变。

LinearGradient参数| 备注
:----:|:----:
colors| 用colors数组线性填充
positions|每个position取值范围[0,1],并且和colors数组中对应位置的color一一对应

int[] colors = {Color.GREEN, Color.GRAY, Color.RED, Color.BLUE};
float[] positions = {0f, 0.5f, 0.75f, 1f};
LinearGradient linearGradient = new LinearGradient(200, 200, 600, 600, colors, positions, Shader.TileMode.CLAMP);
mPaint.setShader(linearGradient);
canvas.drawRect(200, 200, 600, 600, mPaint);

效果图:
![muticolor.png](http://upload-images.jianshu.io/upload_images/587163-97a5f3893d9d3849.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
同样修改代码扩大一下范围并且修改Shader.TileMode模式:

canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

效果图:

![muticolor (2).png](http://upload-images.jianshu.io/upload_images/587163-31529483ebb8ee2d.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
#####RadialGradient:
RadialGradient也用来创建渐变效果,和LinearGradient 不同的是,LinearGradient 是线性渐变,而RadialGradient是径向渐变,也就是从中心向四周发散渐变,RadialGradient也有两个构造函数,先看第一个:

RadialGradient(float centerX, float centerY, float radius, int centerColor, int edgeColor, TileMode tileMode)

(centerX,centerY)是圆心坐标,radius是圆半径,centerColor是圆中心颜色,edgeColor是圆边缘颜色。

RadialGradient参数| 备注
:----:|:----:
centerX| 圆中心的X轴坐标
centerY|圆中心的Y轴坐标
radius| 圆半径
centerColor|圆中心颜色
edgeColor| 圆边缘颜色
tileMode|径向渐变Shader.TileMode模式填充

示例:

RadialGradient gradient = new RadialGradient(getMeasuredWidth() / 2, getMeasuredHeight() / 2, radius, Color.GREEN, Color.BLACK, Shader.TileMode.CLAMP);
mPaint.setShader(gradient);
canvas.drawCircle(getMeasuredWidth() / 2, getMeasuredHeight() / 2, radius, mPaint);

效果图:
![radial.png](http://upload-images.jianshu.io/upload_images/587163-ae8e6331c75f1498.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
因为RadialGradient范围和canvas范围是一样大小,所以RadialGradient构造函数最后一个参数Shader.TileMode不起作用,同样的,我们来扩大canvas范围:

canvas.drawCircle(getMeasuredWidth() / 2, getMeasuredHeight() / 2, getMeasuredWidth() / 2, mPaint);

我们将圆的半径扩大至屏幕宽度的一半,然后看效果图:
![radialMode.png](http://upload-images.jianshu.io/upload_images/587163-492925d9aadd2e0e.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
再来看RadialGradient的另一个构造函数:

RadialGradient(float centerX, float centerY, float radius,int colors[], float stops[],TileMode tileMode)

和上一个不同的是colors和stops,单独列一下:

RadialGradient参数| 备注
:----:|:----:
colors| color数组分布在圆的中心和边缘之间
stops|取值范围在[0.0f,1.0f],并且和colors数组中对应位置的color一一对应,如果为null,颜色均匀的分布在中心和边缘之间

#####SweepGradient:
SweepGradient用来创建围绕一个中心点360度沿顺时针旋转渐变效果:

SweepGradient(float cx, float cy, int color0, int color1)

SweepGradient参数| 备注
:----:|:----:
cx| 圆中心的X轴坐标
cy|圆中心的Y轴坐标
color0| 开始旋转起始颜色,起始点在3点钟方向,顺时针
color1| 结束旋转终止颜色,终止点也在3点钟方向
示例:

SweepGradient gradient = new SweepGradient(getMeasuredWidth() / 2, getMeasuredHeight() / 2, Color.GREEN, Color.RED);
mPaint.setShader(gradient);
canvas.drawCircle(getMeasuredWidth() / 2, getMeasuredHeight() / 2, getMeasuredWidth() / 2, mPaint);

效果图:
![sweepCircle.png](http://upload-images.jianshu.io/upload_images/587163-ec56fa4e8dd1c308.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

修改一下canvas形状

canvas.drawRect(0, (getMeasuredHeight() - getMeasuredWidth()) / 2, getMeasuredWidth(), (getMeasuredHeight() + getMeasuredWidth()) / 2, mPaint);

效果图:
![sweepSquare.png](http://upload-images.jianshu.io/upload_images/587163-72952753319ffa62.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

SweepGradient另一个构造函数:

SweepGradient(float cx, float cy, int colors[], float positions[])

和前面不同的是colors和positions:

SweepGradient参数| 备注
:----:|:----:
colors| color数组顺时针分布
positions|每个position取值范围[0,1],并且和colors数组中对应位置的color一一对应
示例:

int[] colors = {Color.GREEN, Color.YELLOW, Color.BLACK, Color.BLUE, Color.RED};
float[] positions = {0.0f, 0.25f, 0.5f, 0.75f, 1.0f};
SweepGradient gradient = new SweepGradient(getMeasuredWidth() / 2, getMeasuredHeight() / 2, colors, positions);
mPaint.setShader(gradient);
canvas.drawCircle(getMeasuredWidth() / 2, getMeasuredHeight() / 2, getMeasuredWidth() / 2, mPaint);

效果图:
![sweepMultiColor.png](http://upload-images.jianshu.io/upload_images/587163-58646075f04b495e.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

#####ComposeShader:
ComposeShader结合Xfermode模式,是两个Shader的组合模式,ComposeShader有两个构造函数:

ComposeShader(Shader shaderA, Shader shaderB, PorterDuff.Mode mode)
ComposeShader(Shader shaderA, Shader shaderB, Xfermode mode)

Xfermode可以用于实现新绘制的像素与Canvas上对应位置已有的像素按照混合规则进行颜色混合,Xfermode 有三个子类:AvoidXfermode, PixelXorXfermode和PorterDuffXfermode,前两个已废弃,PorterDuffXfermode初始化时需要传入PorterDuff.Mode即:PorterDuffXfermode(PorterDuff.Mode mode),所以上面第一个构造函数是第二个构造函数的一种情况,我们只看第一个构造函数就可以了:

ComposeShader参数| 备注
:----:|:----:
shaderA| 目标像素DST
shaderB|源像素SRC
mode|新绘制的像素与Canvas上对应位置已有的像素按照混合规则进行颜色混合

* ####Paint.setShadowLayer(float radius ,float dx,float dy,int color);
//在图形下面设置阴影层,产生阴影效果:

setShadowLayer参数| 备注
:----:|:----:
radius | radius为阴影半径,半径越大,阴影面积越大,越模糊;反之,半径越小,阴影面积越小,也越清晰,radius=0时,阴影消失
dx|dx为阴影在x轴上的偏移值
dy|dy为阴影在y轴上的偏移值
color|color为阴影的颜色 

示例:

Paint paint = new Paint();
paint.setColor(Color.RED);
paint.setShadowLayer(20, 0, 0, Color.YELLOW);
paint.setTextSize(200);
canvas.drawText("Hello World", 200, 300, paint);

效果图:

![B78480E8AC4E55D310BB0469B05D49E1.jpg](http://upload-images.jianshu.io/upload_images/587163-433815d1b96a5c81.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

修改一下:

paint.setShadowLayer(20,50, 50, Color.YELLOW);

其他代码不变,效果图:
![3978CA920B5314397C82FE8FC931C421.jpg](http://upload-images.jianshu.io/upload_images/587163-156be8d59c3e49fe.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
可以看到阴影偏移量起始坐标(x,y)从(0,0)变成了(50,50),即阴影位置从(0,0)移动到了(50,50)的位置,再改一下:

paint.setShadowLayer(1,50, 50, Color.YELLOW);

其他代码不变,效果图:
![08352B8C321173BEB880E2E763265A00.jpg](http://upload-images.jianshu.io/upload_images/587163-5496184009d39cfa.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
 
阴影半径radius从20变成1,可以看到阴影清晰了很多,再来改一下:

paint.setShadowLayer(0,50, 50, Color.YELLOW);

其他代码不变,效果图:
![C9B902E437521FA902C514527C1694F2.jpg](http://upload-images.jianshu.io/upload_images/587163-1eb64fe0f06ce1f6.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
阴影半径radius变成0时,阴影消失,接着看下面代码:

Paint paint = new Paint();
paint.setColor(Color.GREEN);
paint.setShadowLayer(30, 0, 0, Color.BLACK);
canvas.drawCircle(400, 800, 100, paint);

效果图:
![4675614E0EC4D773515187BD45E6D9DD.jpg](http://upload-images.jianshu.io/upload_images/587163-1397b74c5d5c2cc7.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
**纳尼?我们预期的黑边阴影肿么没有出现?What a fucking day!别急,只要给paint加一句:**

setLayerType(LAYER_TYPE_SOFTWARE, paint);

然后来看下效果图:
![72CC47BD0C72C1BD7A6F89A251EAF3BC.jpg](http://upload-images.jianshu.io/upload_images/587163-07b8a5d2a377b1fc.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
终于看到黑色阴影了,为毛要加setLayerType呢,google工程师给出解释,链接:
http://stackoverflow.com/questions/17410195/setshadowlayer-android-api-differences

Paint先介绍到这里,接下篇:
[Android自定义View工具:Paint&Canvas(二)](http://www.jianshu.com/p/adbe33e887be)
相关实践学习
使用PAI-EAS一键部署ChatGLM及LangChain应用
本场景中主要介绍如何使用模型在线服务(PAI-EAS)部署ChatGLM的AI-Web应用以及启动WebUI进行模型推理,并通过LangChain集成自己的业务数据。
机器学习概览及常见算法
机器学习(Machine Learning, ML)是人工智能的核心,专门研究计算机怎样模拟或实现人类的学习行为,以获取新的知识或技能,重新组织已有的知识结构使之不断改善自身的性能,它是使计算机具有智能的根本途径,其应用遍及人工智能的各个领域。 本课程将带你入门机器学习,掌握机器学习的概念和常用的算法。
相关文章
|
2月前
|
缓存 前端开发 Android开发
安卓开发中的自定义视图:从零到英雄
【10月更文挑战第42天】 在安卓的世界里,自定义视图是一块画布,让开发者能够绘制出独一无二的界面体验。本文将带你走进自定义视图的大门,通过深入浅出的方式,让你从零基础到能够独立设计并实现复杂的自定义组件。我们将探索自定义视图的核心概念、实现步骤,以及如何优化你的视图以提高性能和兼容性。准备好了吗?让我们开始这段创造性的旅程吧!
31 1
|
2月前
|
XML 前端开发 Android开发
Android:UI:Drawable:View/ImageView与Drawable
通过本文的介绍,我们详细探讨了Android中Drawable、View和ImageView的使用方法及其相互关系。Drawable作为图像和图形的抽象表示,提供了丰富的子类和自定义能力,使得开发者能够灵活地实现各种UI效果。View和ImageView则通过使用Drawable实现了各种图像和图形的显示需求。希望本文能为您在Android开发中使用Drawable提供有价值的参考和指导。
46 2
|
2月前
|
搜索推荐 前端开发 Android开发
安卓应用开发中的自定义视图实现
【10月更文挑战第30天】在安卓开发的海洋中,自定义视图是那抹不可或缺的亮色,它为应用界面的个性化和交互体验的提升提供了无限可能。本文将深入探讨如何在安卓平台创建自定义视图,并展示如何通过代码实现这一过程。我们将从基础出发,逐步引导你理解自定义视图的核心概念,然后通过一个实际的代码示例,详细讲解如何将理论应用于实践,最终实现一个美观且具有良好用户体验的自定义控件。无论你是想提高自己的开发技能,还是仅仅出于对安卓开发的兴趣,这篇文章都将为你提供价值。
|
2月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
44 5
|
3月前
|
缓存 数据处理 Android开发
在 Android 中使用 RxJava 更新 View
【10月更文挑战第20天】使用 RxJava 来更新 View 可以提供更优雅、更高效的解决方案。通过合理地运用操作符和订阅机制,我们能够轻松地处理异步数据并在主线程中进行 View 的更新。在实际应用中,需要根据具体情况进行灵活运用,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在 Android 中使用 RxJava 更新 View 的技巧和方法,为开发高质量的 Android 应用提供有力支持。
|
Android开发
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(二)
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(二)
172 0
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(二)
|
移动开发 Android开发 C++
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(一)
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(一)
159 0
【Android 逆向】Android 逆向通用工具开发 ( Android 平台运行的 cmd 程序类型 | Android 平台运行的 cmd 程序编译选项 | 编译 cmd 可执行程序 )(一)
|
2月前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
1月前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
54 19
|
1月前
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
60 14