Android Studio App开发之利用图片加载框架Glide实现刷新验证码功能(附源码 简单易懂)

简介: Android Studio App开发之利用图片加载框架Glide实现刷新验证码功能(附源码 简单易懂)

运行有问题或需要源码请点赞关注收藏后评论区留言~~~

一、从图片地址获取图像数据

网络上的图片一般都不太大,动用DownloadManager下载图片有点大材小用,如果仅仅是在界面上显示网络图片,不涉及复杂处理的话,其实通过HttpURLConnection就能快速的获取网络图像,因为位图工厂BitmapFactory提供了decodeStream方法,允许从输入流中解码得到位图数据,所以使用GET方式访问图片链接之时,在连接成功后获取HTTP连接的输入流对象,即可解码得到位图

举个图片验证码的例子,为了保证验证码的时效性,验证码图片每隔一段时间就要刷新,因此需要定义一个获取图片验证码的异步任务,以便在界面上按需刷新验证码,考虑到位图数据的回收问题,尽量不要在线程之间直接传递位图,而是先把位图对象保存为图片文件,再将文件路径作为字符串参数间接传递。

效果如下

点击即可刷新验证码

模拟机有一点局限性 建议读者连接真机测试

代码如下

Java类

package com.example.chapter14;
import android.net.Uri;
import android.os.Bundle;
import android.view.View;
import android.widget.ImageView;
import androidx.appcompat.app.AppCompatActivity;
import com.example.chapter14.task.GetImageCodeTask;
import com.example.chapter14.task.GetImageCodeTask.OnImageCodeListener;
public class HttpImageActivity extends AppCompatActivity implements View.OnClickListener, OnImageCodeListener {
    private ImageView iv_image_code;
    private boolean isRunning = false; // 是否正在运行
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_http_image);
        iv_image_code = findViewById(R.id.iv_image_code);
        iv_image_code.setOnClickListener(this);
        getImageCode(); // 获取图片验证码
    }
    // 获取图片验证码
    private void getImageCode() {
        if (!isRunning) {
            isRunning = true;
            GetImageCodeTask task = new GetImageCodeTask(this); // 创建验证码获取的异步任务
            task.setOnImageCodeListener(this); // 设置验证码获取的监听器
            task.execute(); // 把验证码获取任务加入到处理队列
        }
    }
    @Override
    public void onClick(View v) {
        if (v.getId() == R.id.iv_image_code) {
            getImageCode(); // 获取图片验证码
        }
    }
    // 在得到验证码后触发
    @Override
    public void onGetCode(String path) {
        iv_image_code.setImageURI(Uri.parse(path)); // 设置图像视图的图片路径
        isRunning = false;
    }
}

任务类

package com.example.chapter14.task;
import android.content.Context;
import android.graphics.Bitmap;
import android.os.AsyncTask;
import android.os.Environment;
import android.util.Log;
import com.example.chapter14.constant.UrlConstant;
import com.example.chapter14.util.HttpUtil;
import com.example.chapter14.util.FileUtil;
import com.example.chapter14.util.DateUtil;
// 获取图片验证码的异步任务
public class GetImageCodeTask extends AsyncTask<Void, Void, String> {
    private final static String TAG = "GetImageCodeTask";
    private Context mContext; // 声明一个上下文对象
    public GetImageCodeTask(Context ctx) {
        super();
        mContext = ctx;
    }
    // 线程正在后台处理
    protected String doInBackground(Void... params) {
        // 为验证码地址添加一个随机时间串。图片验证码的网址见UrlConstant.java
        String url = UrlConstant.IMAGE_CODE_URL + DateUtil.getNowDateTime();
        Log.d(TAG, "image url=" + url);
        // 获得验证码图片的临时保存路径
        String filePath = String.format("%s/%s.jpg",
                mContext.getExternalFilesDir(Environment.DIRECTORY_PICTURES),
                "verify_"+ DateUtil.getNowDateTime());
        Bitmap bitmap = HttpUtil.getImage(url, null); // 访问网络地址获得位图对象
        FileUtil.saveImage(filePath, bitmap); // 把HTTP获得的位图数据保存为图片
        Log.d(TAG, "image path=" + filePath);
        return filePath; // 返回验证码图片的本地路径
    }
    // 线程已经完成处理
    protected void onPostExecute(String path) {
        mListener.onGetCode(path); // HTTP调用完毕,触发监听器的得到验证码事件
    }
    private OnImageCodeListener mListener; // 声明一个获取图片验证码的监听器对象
    // 设置获取图片验证码的监听器
    public void setOnImageCodeListener(OnImageCodeListener listener) {
        mListener = listener;
    }
    // 定义一个获取图片验证码的监听器接口
    public interface OnImageCodeListener {
        void onGetCode(String path);
    }
}

网络类

package com.example.chapter14.util;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.util.Log;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.security.SecureRandom;
import java.security.cert.X509Certificate;
import java.util.Map;
import java.util.zip.GZIPInputStream;
import javax.net.ssl.HostnameVerifier;
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSession;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
public class HttpUtil {
    private final static String TAG = "HttpUtil";
    private final static int CONNECT_TIMEOUT = 15000;
    private final static int READ_TIMEOUT = 15000;
    // 兼容https开头的调用地址
    private static void compatibleSSL(String callUrl) throws Exception {
        if (callUrl.toLowerCase().startsWith("https")) {
            SSLContext sc = SSLContext.getInstance("TLS");
            sc.init(null, new TrustManager[]{new X509TrustManager() {
                @Override
                public X509Certificate[] getAcceptedIssuers() {
                    return null;
                }
                @Override
                public void checkServerTrusted(X509Certificate[] arg0, String arg1) {
                }
                @Override
                public void checkClientTrusted(X509Certificate[] arg0, String arg1) {
                }
            }}, new SecureRandom());
            HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
            HttpsURLConnection.setDefaultHostnameVerifier(new HostnameVerifier() {
                @Override
                public boolean verify(String hostname, SSLSession session) {
                    return true;
                }
            });
        }
    }
    // 对指定接口地址发起GET调用
    public static String get(String callUrl, Map<String, String> headers) {
        String resp = ""; // 应答内容
        try {
            Log.d(TAG, "请求地址:"+callUrl);
            compatibleSSL(callUrl); // 兼容https开头的调用地址
            URL url = new URL(callUrl); // 根据网址字符串构建URL对象
            // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setRequestMethod("GET"); // 设置请求方式
            setConnHeader(conn, headers);// 设置HTTP连接的头部信息
            conn.connect(); // 开始连接
            // 打印HTTP调用的应答内容长度、内容类型、压缩方式
            Log.d(TAG,  String.format("应答内容长度=%d, 内容类型=%s, 压缩方式=%s",
                    conn.getContentLength(), conn.getContentType(), conn.getContentEncoding()) );
            // 对输入流中的数据解压和字符编码,得到原始的应答字符串
            resp = getUnzipString(conn);
            // 打印HTTP调用的应答状态码和应答报文
            Log.d(TAG,  String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
            conn.disconnect(); // 断开连接
        } catch (Exception e) {
            e.printStackTrace();
        }
        return resp;
    }
    // 从指定url获取图片
    public static Bitmap getImage(String callUrl, Map<String, String> headers) {
        Bitmap bitmap = null; // 位图对象
        try {
            Log.d(TAG, "请求地址:"+callUrl);
            compatibleSSL(callUrl); // 兼容https开头的调用地址
            URL url = new URL(callUrl); // 根据网址字符串构建URL对象
            // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setRequestMethod("GET"); // 设置请求方式
            setConnHeader(conn, headers);// 设置HTTP连接的头部信息
            conn.connect(); // 开始连接
            // 打印图片获取的应答内容长度、内容类型、压缩方式
            Log.d(TAG,  String.format("应答内容长度=%d, 内容类型=%s, 压缩方式=%s",
                    conn.getContentLength(), conn.getContentType(), conn.getContentEncoding()) );
            // 对输入流中的数据解码,得到位图对象
            bitmap = BitmapFactory.decodeStream(conn.getInputStream());
            // 打印图片获取的应答状态码和位图大小
            Log.d(TAG,  String.format("应答状态码=%d, 位图大小=%s", conn.getResponseCode(), bitmap.getByteCount()) );
            conn.disconnect(); // 断开连接
        } catch (Exception e) {
            e.printStackTrace();
        }
        return bitmap;
    }
    // 对指定接口地址发起POST调用
    public static String post(String callUrl, String req, Map<String, String> headers) {
        String resp = ""; // 应答内容
        try {
            Log.d(TAG, "请求地址:"+callUrl+", 请求报文="+req);
            compatibleSSL(callUrl); // 兼容https开头的调用地址
            URL url = new URL(callUrl); // 根据网址字符串构建URL对象
            // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setRequestMethod("POST"); // 设置请求方式
            setConnHeader(conn, headers);// 设置HTTP连接的头部信息
            conn.setRequestProperty("Content-Type", "application/json"); // 请求报文为json格式
            conn.setDoOutput(true); // 准备让连接执行输出操作。默认为false,POST方式需要设置为true
            //conn.setDoInput(true); // 准备让连接执行输入操作。默认为true
            conn.connect(); // 开始连接
            OutputStream os = conn.getOutputStream(); // 从连接对象中获取输出流
            os.write(req.getBytes()); // 往输出流写入请求报文
            // 打印HTTP调用的应答内容长度、内容类型、压缩方式
            Log.d(TAG,  String.format("应答内容长度=%s, 内容类型=%s, 压缩方式=%s",
                    conn.getHeaderField("Content-Length"), conn.getHeaderField("Content-Type"),
                    conn.getHeaderField("Content-Encoding")) );
            // 对输入流中的数据解压和字符编码,得到原始的应答字符串
            resp = getUnzipString(conn);
            // 打印HTTP调用的应答状态码和应答报文
            Log.d(TAG,  String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
            conn.disconnect(); // 断开连接
        } catch (Exception e) {
            e.printStackTrace();
        }
        return resp;
    }
    // 把文件上传给指定的URL
    public static String upload(String uploadUrl, String uploadFile, Map<String, String> headers) {
        String resp = ""; // 应答内容
        // 从本地文件路径获取文件名
        String fileName = uploadFile.substring(uploadFile.lastIndexOf("/"));
        String end = "\r\n"; // 结束字符串
        String hyphens = "--"; // 连接字符串
        String boundary = "WUm4580jbtwfJhNp7zi1djFEO3wNNm"; // 边界字符串
        try (FileInputStream fis = new FileInputStream(uploadFile)) {
            Log.d(TAG, "上传地址:"+uploadUrl+", 上传文件="+uploadFile);
            compatibleSSL(uploadUrl); // 兼容https开头的调用地址
            URL url = new URL(uploadUrl); // 根据网址字符串构建URL对象
            // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setRequestMethod("POST"); // 设置请求方式
            setConnHeader(conn, headers);// 设置HTTP连接的头部信息
            conn.setDoOutput(true); // 准备让连接执行输出操作。默认为false,POST方式需要设置为true
            //conn.setDoInput(true); // 准备让连接执行输入操作。默认为true
            conn.setRequestProperty("Connection", "Keep-Alive"); // 连接过程要保持活跃
            // 请求报文要求分段传输,并且各段之间以边界字符串隔开
            conn.setRequestProperty("Content-Type", "multipart/form-data;boundary=" + boundary);
            // 根据连接对象的输出流构建数据输出流
            DataOutputStream ds = new DataOutputStream(conn.getOutputStream());
            // 以下写入请求报文的头部
            ds.writeBytes(hyphens + boundary + end);
            ds.writeBytes("Content-Disposition: form-data; "
                    + "name=\"file\";filename=\"" + fileName + "\"" + end);
            ds.writeBytes(end);
            // 以下写入请求报文的主体
            byte[] buffer = new byte[1024];
            int length;
            // 先将文件数据写入到缓冲区,再将缓冲数据写入输出流
            while ((length = fis.read(buffer)) != -1) {
                ds.write(buffer, 0, length);
            }
            ds.writeBytes(end);
            // 以下写入请求报文的尾部
            ds.writeBytes(hyphens + boundary + hyphens + end);
            ds.close(); // 关闭数据输出流
            // 打印HTTP调用的应答内容长度、内容类型、压缩方式
            Log.d(TAG,  String.format("应答内容长度=%s, 内容类型=%s, 压缩方式=%s",
                    conn.getHeaderField("Content-Length"), conn.getHeaderField("Content-Type"),
                    conn.getHeaderField("Content-Encoding")) );
            // 对输入流中的数据解压和字符编码,得到原始的应答字符串
            resp = getUnzipString(conn);
            // 打印HTTP上传的应答状态码和应答报文
            Log.d(TAG,  String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
            conn.disconnect(); // 断开连接
        } catch (Exception e) {
            e.printStackTrace();
        }
        return resp;
    }
    // 设置HTTP连接的头部信息
    private static void setConnHeader(HttpURLConnection conn, Map<String, String> headers) {
        conn.setConnectTimeout(CONNECT_TIMEOUT); // 设置连接的超时时间,单位毫秒
        conn.setReadTimeout(READ_TIMEOUT); // 设置读取应答数据的超时时间,单位毫秒
        conn.setRequestProperty("Accept", "*/*"); // 设置数据格式
        conn.setRequestProperty("Accept-Language", "zh-CN"); // 设置文本语言
        conn.setRequestProperty("Accept-Encoding", "gzip, deflate"); // 设置编码格式
        // 设置用户代理,包括操作系统版本、浏览器版本等等
        conn.setRequestProperty("User-Agent", "Mozilla/5.0 (Windows NT 6.1; WOW64; rv:33.0) Gecko/20100101 Firefox/33.0");
        if (headers != null) {
            for (Map.Entry<String, String> item : headers.entrySet()) {
                conn.setRequestProperty(item.getKey(), item.getValue());
            }
        }
    }
    // 把输入流中的数据按照指定字符编码转换为字符串。处理大量数据需要使用本方法
    private static String isToString(InputStream is, String charset) {
        String result = "";
        // 创建一个字节数组的输出流对象
        try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
            int i = -1;
            while ((i = is.read()) != -1) { // 循环读取输入流中的字节数据
                baos.write(i); // 把字节数据写入字节数组输出流
            }
            byte[] data = baos.toByteArray(); // 把字节数组输出流转换为字节数组
            result = new String(data, charset); // 将字节数组按照指定的字符编码生成字符串
        } catch (Exception e) {
            e.printStackTrace();
        }
        return result; // 返回转换后的字符串
    }
    // 从HTTP连接中获取已解压且重新编码后的应答报文
    private static String getUnzipString(HttpURLConnection conn) throws IOException {
        String contentType = conn.getContentType(); // 获取应答报文的内容类型(包括字符编码)
        String charset = "UTF-8"; // 默认的字符编码为UTF-8
        if (contentType != null) {
            if (contentType.toLowerCase().contains("charset=gbk")) { // 应答报文采用gbk编码
                charset = "GBK"; // 字符编码改为GBK
            } else if (contentType.toLowerCase().contains("charset=gb2312")) { // 采用gb2312编码
                charset = "GB2312"; // 字符编码改为GB2312
            }
        }
        String contentEncoding = conn.getContentEncoding(); // 获取应答报文的压缩方式
        InputStream is = conn.getInputStream(); // 获取HTTP连接的输入流对象
        String result = "";
        if (contentEncoding != null && contentEncoding.contains("gzip")) { // 应答报文使用gzip压缩
            // 根据输入流对象构建压缩输入流
            try (GZIPInputStream gis = new GZIPInputStream(is)) {
                // 把压缩输入流中的数据按照指定字符编码转换为字符串
                result = isToString(gis, charset);
            } catch (Exception e) {
                e.printStackTrace();
            }
        } else {
            // 把输入流中的数据按照指定字符编码转换为字符串
            result = isToString(is, charset);
        }
        return result; // 返回处理后的应答报文
    }
}

XML文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="5dp" >
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="点击验证码即可刷新验证码图片"
        android:textColor="@color/black"
        android:textSize="17sp" />
    <ImageView
        android:id="@+id/iv_image_code"
        android:layout_width="match_parent"
        android:layout_margin="5dp"
        android:layout_height="50dp"
        android:scaleType="fitCenter" />
</LinearLayout>

创作不易 觉得有帮助请点赞关注收藏~~~

相关文章
|
9天前
|
Web App开发 前端开发 安全
语音交友app系统源码功能及技术研发流程剖析
语音交友App核心功能包括语音聊天(一对一、群聊、语音消息)、语音房间(直播、主题房、管理)、社交互动(好友、关注、打赏)、内容发现、音效美化、通知提醒及安全隐私等。开发流程涵盖需求分析、技术选型(前端、后端、数据库、实时通信)、UI/UX设计、前后端开发、实时通信集成、音效处理、测试优化、部署上线及运营维护,确保稳定高效运行并持续优化用户体验。
|
2天前
|
JavaScript 搜索推荐 Android开发
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
22 8
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
|
2天前
|
数据采集 JavaScript Android开发
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
24 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
14天前
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
115 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
12天前
|
Dart 前端开发 Android开发
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
35 4
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
7天前
|
开发框架 缓存 搜索推荐
PiliPala:开源项目真香,B站用户狂喜!这个开源APP竟能自定义主题+去广告?PiliPala隐藏功能大揭秘
嗨,大家好,我是小华同学。PiliPala 是一个基于 Flutter 开发的 BiliBili 第三方客户端,提供流畅、个性化的使用体验。核心功能包括视频浏览与推荐、用户互动、丰富的播放设置、多维度搜索和个性化主题等。相比官方客户端,PiliPala 功能更丰富、性能更优、界面更美观。
57 14
|
30天前
|
移动开发 监控 小程序
TP6+Uni-app框架开发,2025年最新圈子系统功能展示,圈子app流量主模式
圈子系统基于TP6+Uni-app框架开发,支持多端账号同步并可快速生成APP。它适用于行业、地方、社交、游戏、兴趣等多种圈子场景,提供广告展示、商品销售、推广结算、交易佣金、入驻费用、会员增值及线上线下活动等多元盈利模式,帮助商户精准定位用户,实现流量变现和业务增长。
|
27天前
|
缓存 前端开发 Android开发
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
76 12
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
|
1月前
|
Dart 前端开发 Android开发
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
36 1
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
2月前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
77 19

热门文章

最新文章

  • 1
    MNN-LLM App:在手机上离线运行大模型,阿里巴巴开源基于 MNN-LLM 框架开发的手机 AI 助手应用
  • 2
    【11】flutter进行了聊天页面的开发-增加了即时通讯聊天的整体页面和组件-切换-朋友-陌生人-vip开通详细页面-即时通讯sdk准备-直播sdk准备-即时通讯有无UI集成的区别介绍-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 3
    微信小程序 app.json 配置文件解析与应用
  • 4
    【Azure App Service】基于Linux创建的App Service是否可以主动升级内置的Nginx版本呢?
  • 5
    【05】flutter完成注册页面完善样式bug-增加自定义可复用组件widgets-严格规划文件和目录结构-规范入口文件-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
  • 6
    【Azure Function】Function App出现System.IO.FileNotFoundException异常
  • 7
    原生鸿蒙版小艺APP接入DeepSeek-R1,为HarmonyOS应用开发注入新活力
  • 8
    【Azure Logic App】使用MySQL 新增行触发器遇见错误 :“Unknown column 'created_at' in 'order clause'”
  • 9
    阿里云APP备案流程图以及备案所需材料整理,跟着教程一步步操作
  • 10
    【07】flutter完成主页-完成底部菜单栏并且做自定义组件-完整短视频仿抖音上下滑动页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈