Android 天气APP(十二)空气质量、UI优化调整

简介: Android 天气APP(十二)空气质量、UI优化调整

前言


这个空气质量包含的就是一些常规的空气指数,比如PM2.5、PM10等数据,相信任何天气APP都会有这些数据,所以我也加上去吧,并且修改一些UI。


正文


功能分两个,但是开发步骤是①空气质量 ②UI优化调整 ③自定义背景图片,本篇文章内容会比较多,建议一次看不完的朋友收藏文章或者关注博主,保留浏览入口。


空气质量


空气质量的接口数据和前面的接口稍有不同,location的参数值这次不再是区/县,而是国控站点,国控站点是什么鬼,这个我也是问了和风天气的客户人员才知道的,其实就是市,城市代码ID点击打开看到


20200430104108838.png


具体的信息你可以下载找个文档去仔细了解,首先是访问的地址

https://free-api.heweather.net/s6/air/now?location=深圳&key=3086e91d66c04ce588a7f538f917c7f4

访问拿到的返回数据:

20200430104308221.png


把这些数据生成是实体Bean


20200430104751444.png


AirNowCityResponse.java代码如下

package com.llw.goodweather.bean;
import java.util.List;
public class AirNowCityResponse {
    private List<HeWeather6Bean> HeWeather6;
    public List<HeWeather6Bean> getHeWeather6() {
        return HeWeather6;
    }
    public void setHeWeather6(List<HeWeather6Bean> HeWeather6) {
        this.HeWeather6 = HeWeather6;
    }
    public static class HeWeather6Bean {
        /**
         * basic : {"cid":"CN101280601","location":"深圳","parent_city":"深圳","admin_area":"广东","cnty":"中国","lat":"22.54700089","lon":"114.08594513","tz":"+8.00"}
         * update : {"loc":"2020-04-30 10:27","utc":"2020-04-30 02:27"}
         * status : ok
         * air_now_city : {"aqi":"47","qlty":"优","main":"-","pm25":"23","pm10":"47","no2":"28","so2":"7","co":"0.6","o3":"95","pub_time":"2020-04-30 09:00"}
         * air_now_station : [{"air_sta":"通心岭子站","aqi":"43","asid":"CNA1356","co":"0.6","lat":"22.5545","lon":"114.1063","main":"-","no2":"10","o3":"104","pm10":"43","pm25":"23","pub_time":"2020-04-30 09:00","qlty":"优","so2":"7"},{"air_sta":"洪湖","aqi":"49","asid":"CNA1357","co":"0.5","lat":"22.5625","lon":"114.117","main":"-","no2":"17","o3":"113","pm10":"49","pm25":"27","pub_time":"2020-04-30 09:00","qlty":"优","so2":"8"},{"air_sta":"华侨城","aqi":"38","asid":"CNA1358","co":"0.5","lat":"22.5417","lon":"113.987","main":"-","no2":"18","o3":"105","pm10":"38","pm25":"20","pub_time":"2020-04-30 09:00","qlty":"优","so2":"6"},{"air_sta":"南海子站","aqi":"47","asid":"CNA1359","co":"0.5","lat":"22.5171","lon":"113.9181","main":"-","no2":"33","o3":"77","pm10":"47","pm25":"25","pub_time":"2020-04-30 09:00","qlty":"优","so2":"5"},{"air_sta":"盐田","aqi":"42","asid":"CNA1360","co":"0.5","lat":"22.5908","lon":"114.263","main":"-","no2":"30","o3":"83","pm10":"42","pm25":"13","pub_time":"2020-04-30 09:00","qlty":"优","so2":"9"},{"air_sta":"龙岗","aqi":"49","asid":"CNA1361","co":"0.6","lat":"22.7267","lon":"114.24","main":"-","no2":"32","o3":"98","pm10":"49","pm25":"27","pub_time":"2020-04-30 09:00","qlty":"优","so2":"9"},{"air_sta":"西乡","aqi":"54","asid":"CNA1362","co":"0.6","lat":"22.5794","lon":"113.891","main":"PM10","no2":"54","o3":"62","pm10":"58","pm25":"24","pub_time":"2020-04-30 09:00","qlty":"良","so2":"5"},{"air_sta":"南澳","aqi":"36","asid":"CNA1363","co":"0.5","lat":"22.5422","lon":"114.494","main":"-","no2":"19","o3":"113","pm10":"33","pm25":"22","pub_time":"2020-04-30 09:00","qlty":"优","so2":"4"},{"air_sta":"葵涌","aqi":"45","asid":"CNA1364","co":"0.5","lat":"22.6342","lon":"114.41","main":"-","no2":"26","o3":"87","pm10":"45","pm25":"18","pub_time":"2020-04-30 09:00","qlty":"优","so2":"4"},{"air_sta":"梅沙","aqi":"40","asid":"CNA1365","co":"0.5","lat":"22.5978","lon":"114.297","main":"-","no2":"20","o3":"100","pm10":"40","pm25":"21","pub_time":"2020-04-30 09:00","qlty":"优","so2":"8"},{"air_sta":"观澜","aqi":"59","asid":"CNA1366","co":"0.8","lat":"22.75","lon":"114.085","main":"PM10","no2":"40","o3":"93","pm10":"68","pm25":"29","pub_time":"2020-04-30 09:00","qlty":"良","so2":"5"}]
         */
        private BasicBean basic;
        private UpdateBean update;
        private String status;
        private AirNowCityBean air_now_city;
        private List<AirNowStationBean> air_now_station;
        public BasicBean getBasic() {
            return basic;
        }
        public void setBasic(BasicBean basic) {
            this.basic = basic;
        }
        public UpdateBean getUpdate() {
            return update;
        }
        public void setUpdate(UpdateBean update) {
            this.update = update;
        }
        public String getStatus() {
            return status;
        }
        public void setStatus(String status) {
            this.status = status;
        }
        public AirNowCityBean getAir_now_city() {
            return air_now_city;
        }
        public void setAir_now_city(AirNowCityBean air_now_city) {
            this.air_now_city = air_now_city;
        }
        public List<AirNowStationBean> getAir_now_station() {
            return air_now_station;
        }
        public void setAir_now_station(List<AirNowStationBean> air_now_station) {
            this.air_now_station = air_now_station;
        }
        public static class BasicBean {
            /**
             * cid : CN101280601
             * location : 深圳
             * parent_city : 深圳
             * admin_area : 广东
             * cnty : 中国
             * lat : 22.54700089
             * lon : 114.08594513
             * tz : +8.00
             */
            private String cid;
            private String location;
            private String parent_city;
            private String admin_area;
            private String cnty;
            private String lat;
            private String lon;
            private String tz;
            public String getCid() {
                return cid;
            }
            public void setCid(String cid) {
                this.cid = cid;
            }
            public String getLocation() {
                return location;
            }
            public void setLocation(String location) {
                this.location = location;
            }
            public String getParent_city() {
                return parent_city;
            }
            public void setParent_city(String parent_city) {
                this.parent_city = parent_city;
            }
            public String getAdmin_area() {
                return admin_area;
            }
            public void setAdmin_area(String admin_area) {
                this.admin_area = admin_area;
            }
            public String getCnty() {
                return cnty;
            }
            public void setCnty(String cnty) {
                this.cnty = cnty;
            }
            public String getLat() {
                return lat;
            }
            public void setLat(String lat) {
                this.lat = lat;
            }
            public String getLon() {
                return lon;
            }
            public void setLon(String lon) {
                this.lon = lon;
            }
            public String getTz() {
                return tz;
            }
            public void setTz(String tz) {
                this.tz = tz;
            }
        }
        public static class UpdateBean {
            /**
             * loc : 2020-04-30 10:27
             * utc : 2020-04-30 02:27
             */
            private String loc;
            private String utc;
            public String getLoc() {
                return loc;
            }
            public void setLoc(String loc) {
                this.loc = loc;
            }
            public String getUtc() {
                return utc;
            }
            public void setUtc(String utc) {
                this.utc = utc;
            }
        }
        public static class AirNowCityBean {
            /**
             * aqi : 47
             * qlty : 优
             * main : -
             * pm25 : 23
             * pm10 : 47
             * no2 : 28
             * so2 : 7
             * co : 0.6
             * o3 : 95
             * pub_time : 2020-04-30 09:00
             */
            private String aqi;
            private String qlty;
            private String main;
            private String pm25;
            private String pm10;
            private String no2;
            private String so2;
            private String co;
            private String o3;
            private String pub_time;
            public String getAqi() {
                return aqi;
            }
            public void setAqi(String aqi) {
                this.aqi = aqi;
            }
            public String getQlty() {
                return qlty;
            }
            public void setQlty(String qlty) {
                this.qlty = qlty;
            }
            public String getMain() {
                return main;
            }
            public void setMain(String main) {
                this.main = main;
            }
            public String getPm25() {
                return pm25;
            }
            public void setPm25(String pm25) {
                this.pm25 = pm25;
            }
            public String getPm10() {
                return pm10;
            }
            public void setPm10(String pm10) {
                this.pm10 = pm10;
            }
            public String getNo2() {
                return no2;
            }
            public void setNo2(String no2) {
                this.no2 = no2;
            }
            public String getSo2() {
                return so2;
            }
            public void setSo2(String so2) {
                this.so2 = so2;
            }
            public String getCo() {
                return co;
            }
            public void setCo(String co) {
                this.co = co;
            }
            public String getO3() {
                return o3;
            }
            public void setO3(String o3) {
                this.o3 = o3;
            }
            public String getPub_time() {
                return pub_time;
            }
            public void setPub_time(String pub_time) {
                this.pub_time = pub_time;
            }
        }
        public static class AirNowStationBean {
            /**
             * air_sta : 通心岭子站
             * aqi : 43
             * asid : CNA1356
             * co : 0.6
             * lat : 22.5545
             * lon : 114.1063
             * main : -
             * no2 : 10
             * o3 : 104
             * pm10 : 43
             * pm25 : 23
             * pub_time : 2020-04-30 09:00
             * qlty : 优
             * so2 : 7
             */
            private String air_sta;
            private String aqi;
            private String asid;
            private String co;
            private String lat;
            private String lon;
            private String main;
            private String no2;
            private String o3;
            private String pm10;
            private String pm25;
            private String pub_time;
            private String qlty;
            private String so2;
            public String getAir_sta() {
                return air_sta;
            }
            public void setAir_sta(String air_sta) {
                this.air_sta = air_sta;
            }
            public String getAqi() {
                return aqi;
            }
            public void setAqi(String aqi) {
                this.aqi = aqi;
            }
            public String getAsid() {
                return asid;
            }
            public void setAsid(String asid) {
                this.asid = asid;
            }
            public String getCo() {
                return co;
            }
            public void setCo(String co) {
                this.co = co;
            }
            public String getLat() {
                return lat;
            }
            public void setLat(String lat) {
                this.lat = lat;
            }
            public String getLon() {
                return lon;
            }
            public void setLon(String lon) {
                this.lon = lon;
            }
            public String getMain() {
                return main;
            }
            public void setMain(String main) {
                this.main = main;
            }
            public String getNo2() {
                return no2;
            }
            public void setNo2(String no2) {
                this.no2 = no2;
            }
            public String getO3() {
                return o3;
            }
            public void setO3(String o3) {
                this.o3 = o3;
            }
            public String getPm10() {
                return pm10;
            }
            public void setPm10(String pm10) {
                this.pm10 = pm10;
            }
            public String getPm25() {
                return pm25;
            }
            public void setPm25(String pm25) {
                this.pm25 = pm25;
            }
            public String getPub_time() {
                return pub_time;
            }
            public void setPub_time(String pub_time) {
                this.pub_time = pub_time;
            }
            public String getQlty() {
                return qlty;
            }
            public void setQlty(String qlty) {
                this.qlty = qlty;
            }
            public String getSo2() {
                return so2;
            }
            public void setSo2(String so2) {
                this.so2 = so2;
            }
        }
    }
}


这个实体Bean的有很多返回值,关键数据在AirNowCityBean,这个里面对应的参数名和描述在下面这张表里,写的时候注意一下就可以了


1665403275540.png


接下来就是写API接口了,打开ApiService.java,在里面增加


20200430105401348.png


这一步相信你已经很熟悉了吧,我就不过多的解释了,继续往下走。


接下来打开WeatherContract.java,里面增加空气质量数据的订阅

    /**
         * 空气质量数据
         * @param context
         * @param location
         */
        public void airNowCity(final Context context,String location){
            ApiService service = ServiceGenerator.createService(ApiService.class,0);
            service.getAirNowCity(location).enqueue(new NetCallBack<AirNowCityResponse>() {
                @Override
                public void onSuccess(Call<AirNowCityResponse> call, Response<AirNowCityResponse> response) {
                    if(getView() != null){
                        getView().getAirNowCityResult(response);
                    }
                }
                @Override
                public void onFailed() {
                    if(getView() != null){
                        getView().getDataFailed();
                    }
                }
            });
        }
    //查询空气质量的数据返回
        void getAirNowCityResult(Response<AirNowCityResponse> response);


接下来是MainActivity.java

20200430112610473.png


在定位返回的数据里面新增一个city的变量

20200430144726735.png


然后有两个地方要进行赋值

①定位之后

20200430144850984.png


②切换城市,点击市级列表的时候赋值

20200430145008215.png


请求接口,修改三处地方,定位之后、下拉的时候、点击区/县的时候


20200430145202790.png

20200430145256602.png


修改activity_main.xml布局文件,放在七天天气预报的后面与风车的前面。

在修改布局之前,先自定义一个VIew,这个VIew并不是我自己写的,也是从网络上找的,只不过忘记出处了,因为这种东西网络上一大把,注释都在里面,其实就是一个圆环进度条,增加动画效果了,这种自定义View的代码和样式我都是放在mvplibrary下面的。


首先在style文件中创建样式

20200505232624467.png


样式代码如下

  <!--圆环进度条-->
    <declare-styleable name="RoundProgressBar">
        <attr name="round_max_progress" format="float"/>
        <attr name="round_bg_color" format="color"/>
        <attr name="round_stroke_width" format="dimension"/>
        <attr name="round_progress" format="float"/>
        <attr name="round_progress_color" format="color"/>
        <attr name="round_first_text" format="string"/>
        <attr name="round_first_text_color" format="color"/>
        <attr name="round_first_text_size" format="dimension"/>
        <attr name="round_second_text" format="string"/>
        <attr name="round_second_text_color" format="color"/>
        <attr name="round_second_text_size" format="dimension"/>
        <attr name="round_min_text" format="string"/>
        <attr name="round_min_text_color" format="color"/>
        <attr name="round_min_text_size" format="dimension"/>
        <attr name="round_max_text" format="string"/>
        <attr name="round_max_text_color" format="color"/>
        <attr name="round_max_text_size" format="dimension"/>
        <attr name="round_angle_size" format="float"/>
        <attr name="round_start_angle" format="float"/>
    </declare-styleable>


然后创建这个View


20200430154205869.png


代码如下:


package com.llw.mvplibrary.view;
import android.animation.ValueAnimator;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.view.View;
import androidx.annotation.Nullable;
import com.llw.mvplibrary.R;
public class RoundProgressBar extends View {
    private int mStrokeWidth = dp2px(8);//圆弧的宽度
    private float mStartAngle = 135;//圆弧开始的角度
    private float mAngleSize = 270;//起点角度和终点角度对应的夹角大小
    private int mArcBgColor;//圆弧背景颜色
    private float mMaxProgress;//最大的进度,用于计算进度与夹角的比例
    private float mCurrentAngleSize;//当前进度对应的起点角度到当前进度角度夹角的大小
    private float mCurrentProgress = 0;//当前进度
    private long mDuration = 2000;//动画的执行时长
    private int mProgressColor;//进度圆弧的颜色
    private String mFirstText = "0";//第一行文本
    private int mFirstTextColor = Color.WHITE;//第一行文本的颜色
    private float mFirstTextSize = 56f;//第一行文本的字体大小
    private String mSecondText = " ";//第二行文本
    private int mSecondTextColor = Color.WHITE;//第二行文本的颜色
    private float mSecondTextSize = 56f;//第二行文本的字体大小
    private String mMinText = "0";//进度最小值
    private int mMinTextColor = Color.WHITE;//最小值文本的颜色
    private float mMinTextSize = 32f;//最小值字体大小
    private String mMaxText = "0";//进度最大值
    private int mMaxTextColor = Color.WHITE;//最大值文本的颜色
    private float mMaxTextSize = 32f;//最大值字体大小
    public RoundProgressBar(Context context) {
        super(context, null);
    }
    public RoundProgressBar(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs, 0);
    }
    public RoundProgressBar(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initAttr(context, attrs);
    }
    /**
     * 设置初始化的参数
     *
     * @param context
     * @param attrs
     */
    private void initAttr(Context context, AttributeSet attrs) {
        TypedArray array = context.obtainStyledAttributes(attrs, R.styleable.RoundProgressBar);
        mMaxProgress = array.getFloat(R.styleable.RoundProgressBar_round_max_progress, 500f);
        mArcBgColor = array.getColor(R.styleable.RoundProgressBar_round_bg_color, Color.YELLOW);
        mStrokeWidth = dp2px(array.getDimension(R.styleable.RoundProgressBar_round_stroke_width, 12f));
        mCurrentProgress = array.getFloat(R.styleable.RoundProgressBar_round_progress, 300f);
        mProgressColor = array.getColor(R.styleable.RoundProgressBar_round_progress_color, Color.RED);
        mFirstText = array.getString(R.styleable.RoundProgressBar_round_first_text);
        mFirstTextSize = dp2px(array.getDimension(R.styleable.RoundProgressBar_round_first_text_size, 20f));
        mFirstTextColor = array.getColor(R.styleable.RoundProgressBar_round_first_text_color, Color.RED);
        mSecondText = array.getString(R.styleable.RoundProgressBar_round_second_text);
        mSecondTextSize = dp2px(array.getDimension(R.styleable.RoundProgressBar_round_second_text_size, 20f));
        mSecondTextColor = array.getColor(R.styleable.RoundProgressBar_round_second_text_color, Color.RED);
        mMinText = array.getString(R.styleable.RoundProgressBar_round_min_text);
        mMinTextSize = dp2px(array.getDimension(R.styleable.RoundProgressBar_round_min_text_size, 20f));
        mMinTextColor = array.getColor(R.styleable.RoundProgressBar_round_min_text_color, Color.RED);
        mMaxText = array.getString(R.styleable.RoundProgressBar_round_max_text);
        mMaxTextSize = dp2px(array.getDimension(R.styleable.RoundProgressBar_round_max_text_size, 20f));
        mMaxTextColor = array.getColor(R.styleable.RoundProgressBar_round_max_text_color, Color.RED);
        mAngleSize = array.getFloat(R.styleable.RoundProgressBar_round_angle_size, 270f);
        mStartAngle = array.getFloat(R.styleable.RoundProgressBar_round_start_angle, 135f);
        array.recycle();
    }
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        int centerX = getWidth() / 2;
        RectF rectF = new RectF();
        rectF.left = mStrokeWidth;
        rectF.top = mStrokeWidth;
        rectF.right = centerX * 2 - mStrokeWidth;
        rectF.bottom = centerX * 2 - mStrokeWidth;
        //画最外层的圆弧
        drawArcBg(canvas, rectF);
        //画进度
        drawArcProgress(canvas, rectF);
        //绘制第一级文本
        drawFirstText(canvas, centerX);
        //绘制第二级文本
        drawSecondText(canvas, centerX);
        //绘制最小值文本
        drawMinText(canvas, rectF.left, rectF.bottom);
        //绘制最大值文本
        drawMaxText(canvas, rectF.right, rectF.bottom);
    }
    /**
     * 画最开始的圆弧
     *
     * @param canvas
     * @param rectF
     */
    private void drawArcBg(Canvas canvas, RectF rectF) {
        Paint mPaint = new Paint();
        //画笔的填充样式,Paint.Style.FILL 填充内部;Paint.Style.FILL_AND_STROKE 填充内部和描边;Paint.Style.STROKE 描边
        mPaint.setStyle(Paint.Style.STROKE);
        //圆弧的宽度
        mPaint.setStrokeWidth(mStrokeWidth);
        //抗锯齿
        mPaint.setAntiAlias(true);
        //画笔的颜色
        mPaint.setColor(mArcBgColor);
        //画笔的样式 Paint.Cap.Round 圆形,Cap.SQUARE 方形
        mPaint.setStrokeCap(Paint.Cap.ROUND);
        //开始画圆弧
        canvas.drawArc(rectF, mStartAngle, mAngleSize, false, mPaint);
    }
    /**
     * 画进度的圆弧
     *
     * @param canvas
     * @param rectF
     */
    private void drawArcProgress(Canvas canvas, RectF rectF) {
        Paint paint = new Paint();
        paint.setStyle(Paint.Style.STROKE);
        paint.setStrokeWidth(mStrokeWidth);
        paint.setColor(mProgressColor);
        paint.setAntiAlias(true);
        paint.setStrokeCap(Paint.Cap.ROUND);
        canvas.drawArc(rectF, mStartAngle, mCurrentAngleSize, false, paint);
    }
    /**
     * 绘制第一级文字
     *
     * @param canvas  画笔
     * @param centerX 位置
     */
    private void drawFirstText(Canvas canvas, float centerX) {
        Paint paint = new Paint();
        paint.setAntiAlias(true);
        paint.setColor(mFirstTextColor);
        paint.setTextAlign(Paint.Align.CENTER);
        paint.setTextSize(mFirstTextSize);
        Rect firstTextBounds = new Rect();
        paint.getTextBounds(mFirstText, 0, mFirstText.length(), firstTextBounds);
        canvas.drawText(mFirstText, centerX, firstTextBounds.height() / 2 + getHeight() * 2 / 5, paint);
    }
    /**
     * 绘制第二级文本
     *
     * @param canvas  画笔
     * @param centerX 文本
     */
    private void drawSecondText(Canvas canvas, float centerX) {
        Paint paint = new Paint();
        paint.setAntiAlias(true);
        paint.setColor(mSecondTextColor);
        paint.setTextAlign(Paint.Align.CENTER);
        paint.setTextSize(mSecondTextSize);
        Rect bounds = new Rect();
        paint.getTextBounds(mSecondText, 0, mSecondText.length(), bounds);
        canvas.drawText(mSecondText, centerX, getHeight() / 2 + bounds.height() / 2 +
                getFontHeight(mSecondText, mSecondTextSize), paint);
    }
    /**
     * 绘制最小值文本
     *
     * @param canvas  画笔
     * @param leftX   X轴位置
     * @param bottomY Y轴位置
     */
    private void drawMinText(Canvas canvas, float leftX, float bottomY) {
        Paint paint = new Paint();
        paint.setAntiAlias(true);
        paint.setColor(mMinTextColor);
        paint.setTextAlign(Paint.Align.CENTER);//设置绘制方式 中心对齐
        paint.setTextSize(mMinTextSize);
        Rect bounds = new Rect();
        paint.getTextBounds(mMinText, 0, mMinText.length(), bounds);//TextView的高度和宽度
        canvas.drawText(mMinText, leftX + bounds.width() * 4, bottomY+16, paint);
    }
    /**
     * 绘制最大值文本
     *
     * @param canvas  画笔
     * @param rightX  X轴位置
     * @param bottomY Y轴位置
     */
    private void drawMaxText(Canvas canvas, float rightX, float bottomY) {
        Paint paint = new Paint();
        paint.setAntiAlias(true);
        paint.setColor(mMaxTextColor);
        paint.setTextAlign(Paint.Align.CENTER);//设置绘制方式 中心对齐
        paint.setTextSize(mMaxTextSize);
        Rect bounds = new Rect();
        paint.getTextBounds(mMaxText, 0, mMaxText.length(), bounds);//TextView的高度和宽度
        canvas.drawText(mMaxText, rightX - bounds.width(), bottomY+16, paint);
    }
    /**
     * 设置最大的进度
     *
     * @param progress
     */
    public void setMaxProgress(int progress) {
        if (progress < 0) {
            throw new IllegalArgumentException("Progress value can not be less than 0 ");
        }
        mMaxProgress = progress;
    }
    /**
     * 设置当前进度
     *
     * @param progress
     */
    public void setProgress(float progress) {
        if (progress < 0) {
            throw new IllegalArgumentException("Progress value can not be less than 0");
        }
        if (progress > mMaxProgress) {
            progress = mMaxProgress;
        }
        mCurrentProgress = progress;
        float size = mCurrentProgress / mMaxProgress;
        mCurrentAngleSize = (int) (mAngleSize * size);
        setAnimator(0, mCurrentAngleSize);
    }
    /**
     * 设置进度圆弧的颜色
     *
     * @param color
     */
    public void setProgressColor(int color) {
        if (color == 0) {
            throw new IllegalArgumentException("Color can no be 0");
        }
        mProgressColor = color;
    }
    /**
     * 设置圆弧的颜色
     *
     * @param color
     */
    public void setArcBgColor(int color) {
        if (color == 0) {
            throw new IllegalArgumentException("Color can no be 0");
        }
        mArcBgColor = color;
    }
    /**
     * 设置圆弧的宽度
     *
     * @param strokeWidth
     */
    public void setStrokeWidth(int strokeWidth) {
        if (strokeWidth < 0) {
            throw new IllegalArgumentException("strokeWidth value can not be less than 0");
        }
        mStrokeWidth = dp2px(strokeWidth);
    }
    /**
     * 设置动画的执行时长
     *
     * @param duration
     */
    public void setAnimatorDuration(long duration) {
        if (duration < 0) {
            throw new IllegalArgumentException("Duration value can not be less than 0");
        }
        mDuration = duration;
    }
    /**
     * 设置第一行文本
     *
     * @param text
     */
    public void setFirstText(String text) {
        mFirstText = text;
    }
    /**
     * 设置第一行文本的颜色
     *
     * @param color
     */
    public void setFirstTextColor(int color) {
        if (color <= 0) {
            throw new IllegalArgumentException("Color value can not be less than 0");
        }
        mFirstTextColor = color;
    }
    /**
     * 设置第一行文本的大小
     *
     * @param textSize
     */
    public void setFirstTextSize(float textSize) {
        if (textSize <= 0) {
            throw new IllegalArgumentException("textSize can not be less than 0");
        }
        mFirstTextSize = textSize;
    }
    /**
     * 设置第二行文本
     *
     * @param text
     */
    public void setSecondText(String text) {
        mSecondText = text;
    }
    /**
     * 设置第二行文本的颜色
     *
     * @param color
     */
    public void setSecondTextColor(int color) {
        if (color == 0) {
            throw new IllegalArgumentException("Color value can not be less than 0");
        }
        mSecondTextColor = color;
    }
    /**
     * 设置第二行文本的大小
     *
     * @param textSize
     */
    public void setSecondTextSize(float textSize) {
        if (textSize <= 0) {
            throw new IllegalArgumentException("textSize can not be less than 0");
        }
        mSecondTextSize = textSize;
    }
    /**
     * 设置最小值文本
     *
     * @param text
     */
    public void setMinText(String text) {
        mMinText = text;
    }
    /**
     * 设置最小值文本的颜色
     *
     * @param color
     */
    public void setMinTextColor(int color) {
        if (color == 0) {
            throw new IllegalArgumentException("Color value can not be less than 0");
        }
        mMinTextColor = color;
    }
    /**
     * 设置最小值文本的大小
     *
     * @param textSize
     */
    public void setMinTextSize(float textSize) {
        if (textSize <= 0) {
            throw new IllegalArgumentException("textSize can not be less than 0");
        }
        mMinTextSize = textSize;
    }
    /**
     * 设置最大值文本
     *
     * @param text
     */
    public void setMaxText(String text) {
        mMaxText = text;
    }
    /**
     * 设置最大值文本的颜色
     *
     * @param color
     */
    public void setMaxTextColor(int color) {
        if (color == 0) {
            throw new IllegalArgumentException("Color value can not be less than 0");
        }
        mMaxTextColor = color;
    }
    /**
     * 设置最大值文本的大小
     *
     * @param textSize
     */
    public void setMaxTextSize(float textSize) {
        if (textSize <= 0) {
            throw new IllegalArgumentException("textSize can not be less than 0");
        }
        mMaxTextSize = textSize;
    }
    /**
     * 设置圆弧开始的角度
     *
     * @param startAngle
     */
    public void setStartAngle(int startAngle) {
        mStartAngle = startAngle;
    }
    /**
     * 设置圆弧的起始角度到终点角度的大小
     *
     * @param angleSize
     */
    public void setAngleSize(int angleSize) {
        mAngleSize = angleSize;
    }
    /**
     * dp转成px
     *
     * @param dp
     * @return
     */
    private int dp2px(float dp) {
        float density = getResources().getDisplayMetrics().density;
        return (int) (dp * density + 0.5f * (dp >= 0 ? 1 : -1));
    }
    /**
     * 设置动画
     *
     * @param start  开始位置
     * @param target 结束位置
     */
    private void setAnimator(float start, float target) {
        ValueAnimator valueAnimator = ValueAnimator.ofFloat(start, target);
        valueAnimator.setDuration(mDuration);
        valueAnimator.setTarget(mCurrentAngleSize);
        valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator valueAnimator) {
                mCurrentAngleSize = (float) valueAnimator.getAnimatedValue();
                invalidate();
            }
        });
        valueAnimator.start();
    }
    /**
     * 测量字体的高度
     *
     * @param textStr
     * @param fontSize
     * @return
     */
    private float getFontHeight(String textStr, float fontSize) {
        Paint paint = new Paint();
        paint.setTextSize(fontSize);
        Rect bounds = new Rect();
        paint.getTextBounds(textStr, 0, textStr.length(), bounds);
        return bounds.height();
    }
}


空气污染指数划分为0-50、51-100、101-150、151-200、201-300和大于300六档。

最小值为0,最大值为500。

接下来修改main_activity.xml布局文件


20200505233112898.png

20200505233654488.png


然后是这一部分的代码,只要修改原来的代码即可


              <!--分隔线 增加UI效果-->
                            <View
                                android:layout_marginTop="8dp"
                                android:layout_marginLeft="20dp"
                                android:layout_marginRight="20dp"
                                android:layout_width="match_parent"
                                android:layout_height="1dp"
                                android:background="@color/white"
                                android:alpha="0.1"/>
                            <!--用于显示逐小时天气-->
                            <androidx.recyclerview.widget.RecyclerView
                                android:padding="12dp"
                                android:id="@+id/rv_hourly"
                                android:layout_width="match_parent"
                                android:layout_height="wrap_content"/>
                            <!--用于显示天气预报数据-->
                            <androidx.recyclerview.widget.RecyclerView
                                android:paddingLeft="8dp"
                                android:paddingRight="8dp"
                                android:id="@+id/rv"
                                android:layout_width="match_parent"
                                android:layout_height="wrap_content"/>
                            <!--空气质量-->
                            <LinearLayout
                                android:padding="20dp"
                                android:orientation="vertical"
                                android:layout_width="match_parent"
                                android:layout_height="wrap_content">
                                <!--标题-->
                                <TextView
                                    android:textSize="18sp"
                                    android:textColor="@color/white"
                                    android:text="空气质量"
                                    android:layout_width="wrap_content"
                                    android:layout_height="wrap_content"/>
                                <LinearLayout
                                    android:padding="8dp"
                                    android:orientation="horizontal"
                                    android:layout_width="match_parent"
                                    android:layout_height="wrap_content">
                                    <!--污染指数 动画展示-->
                                    <LinearLayout
                                        android:gravity="center"
                                        android:orientation="vertical"
                                        android:layout_width="0dp"
                                        android:layout_height="match_parent"
                                        android:layout_weight="1">
                                        <TextView
                                            android:layout_marginBottom="8dp"
                                            android:textSize="14sp"
                                            android:textColor="#DAEBEE"
                                            android:text="污染指数"
                                            android:layout_width="wrap_content"
                                            android:layout_height="wrap_content"/>
                                        <!--显示污染指数进度值-->
                                        <com.llw.mvplibrary.view.RoundProgressBar
                                            android:id="@+id/rpb_aqi"
                                            app:round_bg_color="#C6D7F4"
                                            app:round_progress_color="#FBFEF7"
                                            android:layout_gravity="center"
                                            android:layout_width="120dp"
                                            android:layout_height="120dp"/>
                                    </LinearLayout>
                                    <!--其他指数-->
                                    <LinearLayout
                                        android:orientation="vertical"
                                        android:layout_width="0dp"
                                        android:layout_height="wrap_content"
                                        android:layout_weight="1">
                                        <!--PM10-->
                                        <LinearLayout
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <TextView
                                                android:gravity="center"
                                                android:textColor="@color/blue_one"
                                                android:text="PM10"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_pm10"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                        <!--PM2.5-->
                                        <LinearLayout
                                            android:layout_marginTop="12dp"
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <TextView
                                                android:gravity="center"
                                                android:textColor="@color/blue_one"
                                                android:text="PM2.5"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_pm25"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                        <!--NO2 二氧化氮-->
                                        <LinearLayout
                                            android:layout_marginTop="12dp"
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <LinearLayout
                                                android:gravity="center"
                                                android:layout_width="0dp"
                                                android:layout_height="wrap_content"
                                                android:layout_weight="1">
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="NO"
                                                    android:textSize="12sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="2"
                                                    android:textSize="8sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                            </LinearLayout>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_no2"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                        <!--SO2 二氧化硫-->
                                        <LinearLayout
                                            android:layout_marginTop="12dp"
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <LinearLayout
                                                android:gravity="center"
                                                android:layout_width="0dp"
                                                android:layout_height="wrap_content"
                                                android:layout_weight="1">
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="SO"
                                                    android:textSize="12sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="2"
                                                    android:textSize="8sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                            </LinearLayout>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_so2"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                        <!--O3 臭氧-->
                                        <LinearLayout
                                            android:layout_marginTop="12dp"
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <LinearLayout
                                                android:gravity="center"
                                                android:layout_width="0dp"
                                                android:layout_height="wrap_content"
                                                android:layout_weight="1">
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="O"
                                                    android:textSize="12sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                                <TextView
                                                    android:textColor="@color/blue_one"
                                                    android:text="3"
                                                    android:textSize="8sp"
                                                    android:layout_width="wrap_content"
                                                    android:layout_height="wrap_content"/>
                                            </LinearLayout>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_o3"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                        <!--CO 一氧化碳-->
                                        <LinearLayout
                                            android:layout_marginTop="12dp"
                                            android:layout_width="match_parent"
                                            android:layout_height="wrap_content">
                                            <TextView
                                                android:gravity="center"
                                                android:textColor="@color/blue_one"
                                                android:text="CO"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                            <TextView
                                                android:gravity="center"
                                                android:id="@+id/tv_co"
                                                android:textColor="@color/white"
                                                android:textSize="12sp"
                                                android:layout_width="0dp"
                                                android:layout_weight="1"
                                                android:layout_height="wrap_content"/>
                                        </LinearLayout>
                                    </LinearLayout>
                                </LinearLayout>
                            </LinearLayout>
                            <!--风力展示-->
                            <LinearLayout
                                android:orientation="vertical"
                                android:padding="20dp"
                                android:layout_width="match_parent"
                                android:layout_height="wrap_content">
                                <!--标题-->
                                <TextView
                                    android:textSize="18sp"
                                    android:textColor="@color/white"
                                    android:text="风向风力"
                                    android:layout_width="wrap_content"
                                    android:layout_height="wrap_content"/>
                                <LinearLayout
                                    android:layout_marginTop="8dp"
                                    android:layout_width="match_parent"
                                    android:layout_height="wrap_content">
                                    <RelativeLayout
                                        android:id="@+id/rl_wind"
                                        android:layout_width="130dp"
                                        android:layout_height="120dp">
                                        <!--大风车-->
                                        <com.llw.mvplibrary.view.WhiteWindmills
                                            android:id="@+id/ww_big"
                                            android:layout_width="100dp"
                                            android:layout_height="120dp" />
                                        <!--小风车-->
                                        <com.llw.mvplibrary.view.WhiteWindmills
                                            android:id="@+id/ww_small"
                                            android:layout_width="50dp"
                                            android:layout_height="60dp"
                                            android:layout_alignParentBottom="true"
                                            android:layout_alignParentRight="true"
                                            />
                                    </RelativeLayout>
                                    <LinearLayout
                                        android:gravity="center"
                                        android:orientation="vertical"
                                        android:layout_weight="1"
                                        android:layout_width="0dp"
                                        android:layout_height="match_parent">
                                        <!--风向-->
                                        <TextView
                                            android:id="@+id/tv_wind_direction"
                                            android:textColor="@color/white"
                                            android:textSize="@dimen/sp_14"
                                            android:layout_width="wrap_content"
                                            android:layout_height="wrap_content"/>
                                        <!--风力-->
                                        <TextView
                                            android:layout_marginTop="20dp"
                                            android:id="@+id/tv_wind_power"
                                            android:textColor="@color/white"
                                            android:textSize="@dimen/sp_14"
                                            android:layout_width="wrap_content"
                                            android:layout_height="wrap_content"/>
                                    </LinearLayout>
                                </LinearLayout>
                            </LinearLayout>


接下来在mvplibrary中的res文件下新建一个colors.xml

20200505234214460.png


<?xml version="1.0" encoding="utf-8"?>
<resources>
    <color name="arc_bg_color">#C6D7F4</color>
    <color name="arc_progress_color">#FBFEF7</color>
    <color name="white">#ffffff</color>
    <color name="black">#000000</color>
    <color name="blue_one">#9FC8E9</color>
    <color name="transparent">#00000000</color>
    <color name="transparent_bg">#22000000</color>
</resources>


然后在MainActivity中

20200505234335776.png


20200505234508594.png


  //空气质量数据返回
    @Override
    public void getAirNowCityResult(Response<AirNowCityResponse> response) {
        dismissLoadingDialog();//关闭弹窗
        if (("ok").equals(response.body().getHeWeather6().get(0).getStatus())) {
            //UI显示
            AirNowCityResponse.HeWeather6Bean.AirNowCityBean data = response.body().getHeWeather6().get(0).getAir_now_city();
            if (!ObjectUtils.isEmpty(data) && data != null) {
                //污染指数
                rpbAqi.setMaxProgress(500);//最大进度,用于计算
                rpbAqi.setMinText("0");//设置显示最小值
                rpbAqi.setMinTextSize(32f);
                rpbAqi.setMaxText("500");//设置显示最大值
                rpbAqi.setMaxTextSize(32f);
                rpbAqi.setProgress(Float.valueOf(data.getAqi()));//当前进度
                rpbAqi.setArcBgColor(getResources().getColor(R.color.arc_bg_color));//圆弧的颜色
                rpbAqi.setProgressColor(getResources().getColor(R.color.arc_progress_color));//进度圆弧的颜色
                rpbAqi.setFirstText(data.getQlty());//空气质量描述  取值范围:优,良,轻度污染,中度污染,重度污染,严重污染
                rpbAqi.setFirstTextSize(44f);
                rpbAqi.setSecondText(data.getAqi());//空气质量值
                rpbAqi.setSecondTextSize(64f);
                rpbAqi.setMinText("0");
                rpbAqi.setMinTextColor(getResources().getColor(R.color.arc_progress_color));
                tvPm10.setText(data.getPm10());//PM10
                tvPm25.setText(data.getPm2p5());//PM2.5
                tvNo2.setText(data.getNo2());//二氧化氮
                tvSo2.setText(data.getSo2());//二氧化硫
                tvO3.setText(data.getO3());//臭氧
                tvCo.setText(data.getCo());//一氧化碳
            }
        } else {
            ToastUtils.showShortToast(context, response.body().getHeWeather6().get(0).getStatus());
        }
    }

然后运行一下:


20200505234633370.png


这样,空气质量就完成了,UI也只是小改动而已,主要是空气质量数据的请求的UI展示。如果你在写作过程中遇到什么问题,及时提出来,我会最快回复你的。


相关实践学习
部署Stable Diffusion玩转AI绘画(GPU云服务器)
本实验通过在ECS上从零开始部署Stable Diffusion来进行AI绘画创作,开启AIGC盲盒。
相关文章
|
3月前
|
JavaScript 前端开发 Android开发
【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
126 13
【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
2月前
|
JavaScript Linux 网络安全
Termux安卓终端美化与开发实战:从下载到插件优化,小白也能玩转Linux
Termux是一款安卓平台上的开源终端模拟器,支持apt包管理、SSH连接及Python/Node.js/C++开发环境搭建,被誉为“手机上的Linux系统”。其特点包括零ROOT权限、跨平台开发和强大扩展性。本文详细介绍其安装准备、基础与高级环境配置、必备插件推荐、常见问题解决方法以及延伸学习资源,帮助用户充分利用Termux进行开发与学习。适用于Android 7+设备,原创内容转载请注明来源。
422 76
|
6月前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
3月前
|
JavaScript 搜索推荐 Android开发
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
106 8
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
|
3月前
|
数据采集 JavaScript Android开发
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
121 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
5月前
|
存储 监控 API
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
389 11
|
5月前
|
人工智能 搜索推荐 算法
婚恋交友系统UI/UX设计优化 婚恋交友系统用户界面友好性提升 婚恋交友系统用户行为分析与优化 婚恋交友系统用户反馈收集与处理
针对婚恋交友系统的UI/UX设计优化,本文提出多项策略:简化用户界面、提升交互体验、个性化推荐算法;增强用户界面友好性,包括适应性、无障碍及情感化设计;通过数据收集与分析优化用户行为路径;建立多渠道反馈机制,分类处理并及时告知结果。这些措施旨在提高用户体验和满意度,促进平台健康发展。[点击查看完整演示和免费源码](https://gitee.com/duoke-official-open-source/hunlianjiaoyou)
314 5
|
5月前
|
机器学习/深度学习 前端开发 算法
婚恋交友系统平台 相亲交友平台系统 婚恋交友系统APP 婚恋系统源码 婚恋交友平台开发流程 婚恋交友系统架构设计 婚恋交友系统前端/后端开发 婚恋交友系统匹配推荐算法优化
婚恋交友系统平台通过线上互动帮助单身男女找到合适伴侣,提供用户注册、个人资料填写、匹配推荐、实时聊天、社区互动等功能。开发流程包括需求分析、技术选型、系统架构设计、功能实现、测试优化和上线运维。匹配推荐算法优化是核心,通过用户行为数据分析和机器学习提高匹配准确性。
446 4
|
5月前
|
XML 搜索推荐 前端开发
安卓开发中的自定义视图:打造个性化UI组件
在安卓应用开发中,自定义视图是一种强大的工具,它允许开发者创造独一无二的用户界面元素,从而提升应用的外观和用户体验。本文将通过一个简单的自定义视图示例,引导你了解如何在安卓项目中实现自定义组件,并探讨其背后的技术原理。我们将从基础的View类讲起,逐步深入到绘图、事件处理以及性能优化等方面。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
6月前
|
传感器 iOS开发 UED
探索iOS生态系统:从App Store优化到用户体验提升
本文旨在深入探讨iOS生态系统的多个方面,特别是如何通过App Store优化(ASO)和改进用户体验来提升应用的市场表现。不同于常规摘要仅概述文章内容的方式,我们将直接进入主题,首先介绍ASO的重要性及其对开发者的意义;接着分析当前iOS平台上用户行为的变化趋势以及这些变化如何影响应用程序的设计思路;最后提出几点实用建议帮助开发者更好地适应市场环境,增强自身竞争力。

热门文章

最新文章