Android Java开发异步

简介: 【6月更文挑战第15天】

Android Java开发异步

在Android应用程序中,异步编程是至关重要的,它可以确保应用的流畅性和响应性。在本文中,我们将探讨Android开发中的异步编程,以及如何使用Java语言进行异步操作。

为什么需要异步编程?

在Android开发中,主线程(也称为UI线程)负责处理用户界面的更新和响应用户操作。如果在主线程上执行耗时的操作(如网络请求、数据库查询等),会导致应用程序的界面失去响应,用户体验变得不流畅,甚至可能触发应用程序崩溃。 为了避免这种情况,我们需要将耗时的操作转移到其他线程上执行,保持主线程的响应性。这就是异步编程的重要性所在。

AsyncTask类

在Android开发中,最常用的异步编程工具是AsyncTask类。AsyncTask类封装了一些Android提供的异步操作API,使异步任务的编写变得更加简单。 以下是一个使用AsyncTask类的示例代码:

javaCopy code
public class MyAsyncTask extends AsyncTask<Void, Integer, Boolean> {
    @Override
    protected void onPreExecute() {
        // 在执行异步任务之前执行的操作(在主线程中执行)
        // 可以进行一些初始化操作,如显示进度对话框
    }
    @Override
    protected Boolean doInBackground(Void... params) {
        // 在后台线程中执行的耗时操作
        // 可以进行网络请求、数据库查询等操作
        // 返回结果给onPostExecute()方法
    }
    @Override
    protected void onProgressUpdate(Integer... values) {
        // 在后台线程中执行过程中更新UI的操作
        // 例如更新进度条的进度
    }
    @Override
    protected void onPostExecute(Boolean result) {
        // 在执行完异步任务后执行的操作(在主线程中执行)
        // 可以根据任务执行结果更新UI,如隐藏进度对话框
    }
}

在这个示例代码中,MyAsyncTask类继承自AsyncTask类,并重写了其中的几个方法。 onPreExecute()方法在执行异步任务之前在主线程中调用,可以进行一些初始化操作。 doInBackground()方法在后台线程中执行耗时操作,例如进行网络请求或数据库查询等。需要注意的是,这个方法不可以直接操作UI线程,如更新UI元素,否则会抛出异常。 onProgressUpdate()方法在后台线程中执行过程中调用,用于在UI线程中更新进度条等UI元素。 onPostExecute()方法在异步任务执行完成后在主线程中调用,可以根据任务执行结果来更新UI或执行其他操作。 使用MyAsyncTask类的示例代码如下:

javaCopy code
MyAsyncTask myAsyncTask = new MyAsyncTask();
myAsyncTask.execute();

这段代码会创建一个MyAsyncTask对象并调用execute()方法来执行异步任务。

异步编程的注意事项

在进行异步编程时,需要注意以下几点:

  • 不要在异步任务的doInBackground()方法中直接更新UI线程,如修改UI元素的属性或调用UI线程的方法。如果需要更新UI,请使用onProgressUpdate()方法或onPostExecute()方法,或者使用runOnUiThread()方法在UI线程中执行相应操作。
  • 如果异步任务执行过程中需要进行进度更新,可以调用publishProgress()方法来触发onProgressUpdate()方法的执行。
  • 当不需要异步任务时(如Activity销毁),应该及时取消异步任务的执行,以避免内存泄漏。可以调用cancel()方法来取消异步任务的执行。

使用AsyncTask类执行网络请求的代码。

javaCopy code
public class NetworkRequestTask extends AsyncTask<String, Void, String> {
    private TextView resultTextView;
    public NetworkRequestTask(TextView textView) {
        this.resultTextView = textView;
    }
    @Override
    protected void onPreExecute() {
        // 在执行异步任务之前执行的操作(在主线程中执行)
        // 可以进行一些初始化操作,如显示进度对话框
    }
    @Override
    protected String doInBackground(String... urls) {
        // 在后台线程中执行的耗时操作
        // 可以进行网络请求、数据库查询等操作
        // 返回结果给onPostExecute()方法
        String result = "";
        try {
            // 创建URL对象
            URL url = new URL(urls[0]);
            // 打开连接
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
            // 设置请求方法
            connection.setRequestMethod("GET");
            // 获取输入流
            InputStream inputStream = connection.getInputStream();
            // 读取输入流的内容
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
            String line;
            while ((line = bufferedReader.readLine()) != null) {
                result += line;
            }
            // 关闭连接和流
            bufferedReader.close();
            inputStream.close();
            connection.disconnect();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return result;
    }
    @Override
    protected void onPostExecute(String result) {
        // 在执行完异步任务后执行的操作(在主线程中执行)
        // 可以根据任务执行结果更新UI,如隐藏进度对话框
        resultTextView.setText(result);
    }
}

上述代码中,创建了一个名为NetworkRequestTask的子类,继承自AsyncTask类。构造函数接受一个TextView参数,用于显示网络请求的结果。 在doInBackground()方法中执行了网络请求的操作,通过打开连接、设置请求方法、获取输入流等步骤来完成网络请求,并将结果作为字符串返回。 在onPostExecute()方法中更新UI,将网络请求的结果设置到传入的TextView上。 在实际使用时,可以按如下方式调用NetworkRequestTask类:

javaCopy code
TextView resultTextView = findViewById(R.id.result_text_view);
NetworkRequestTask networkRequestTask = new NetworkRequestTask(resultTextView);
networkRequestTask.execute("https://api.example.com/data");

上述代码中,将一个TextView实例传递给NetworkRequestTask的构造函数,并调用execute()方法来执行异步任务。执行过程中,可以在onPreExecute()方法中显示进度对话框,等待网络请求完成后,在onPostExecute()方法中隐藏对话框并更新UI。

使用SQLite数据库进行本地数据存储。为了给出一个具体的示例,我将提供一个使用SQLite数据库的代码示例。 首先,需要创建一个辅助类来管理数据库的创建和版本控制。这个类需要继承自SQLiteOpenHelper类,并实现onCreate()onUpgrade()方法。下面是一个示例:

javaCopy code
public class DatabaseHelper extends SQLiteOpenHelper {
    private static final String DATABASE_NAME = "mydatabase.db";
    private static final int DATABASE_VERSION = 1;
    public DatabaseHelper(Context context) {
        super(context, DATABASE_NAME, null, DATABASE_VERSION);
    }
    @Override
    public void onCreate(SQLiteDatabase db) {
        // 在数据库创建时执行的操作
        // 创建表格或执行其他必要的初始化操作
        String createTableQuery = "CREATE TABLE IF NOT EXISTS users (id INTEGER PRIMARY KEY, name TEXT)";
        db.execSQL(createTableQuery);
    }
    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        // 在数据库升级时执行的操作
        // 可以更新表格结构或执行其他必要的迁移操作
        String dropTableQuery = "DROP TABLE IF EXISTS users";
        db.execSQL(dropTableQuery);
        onCreate(db);
    }
}

接下来,可以使用这个辅助类来进行数据库操作。下面是一个添加用户数据的示例:

javaCopy code
public void addUser(String name) {
    SQLiteDatabase database = getWritableDatabase();
    ContentValues values = new ContentValues();
    values.put("name", name);
    database.insert("users", null, values);
    database.close();
}

在上述示例中,getWritableDatabase()方法获取可写的数据库实例。然后,创建一个ContentValues对象,将要添加的数据放入该对象中。使用database.insert()方法将数据插入到名为users的表格中,再使用database.close()方法关闭数据库连接。 下面是一个查询用户数据的示例:

javaCopy code
public List<String> getUsers() {
    List<String> userList = new ArrayList<>();
    SQLiteDatabase database = getReadableDatabase();
    Cursor cursor = database.rawQuery("SELECT * FROM users", null);
    if (cursor.moveToFirst()) {
        do {
            String name = cursor.getString(1);
            userList.add(name);
        } while (cursor.moveToNext());
    }
    cursor.close();
    database.close();
    return userList;
}

在上述示例中,getReadableDatabase()方法获取可读的数据库实例。然后,使用database.rawQuery()方法执行SQL查询语句,查询所有的用户数据。通过移动Cursor指针,逐行读取查询结果,并将用户名添加到列表中。最后,关闭Cursor和数据库连接,并返回用户列表。 要使用上述代码,需要创建一个DatabaseHelper对象,并调用其方法来执行数据库操作。例如:

javaCopy code
DatabaseHelper dbHelper = new DatabaseHelper(context);
dbHelper.addUser("John");
dbHelper.addUser("Alice");
List<String> userList = dbHelper.getUsers();

上述代码中,先创建一个DatabaseHelper对象,然后使用addUser()方法添加两个用户数据。最后,使用getUsers()方法获取所有用户数据,并将其存储在一个列表中。

总结

在Android开发中,异步编程是确保应用程序流畅性和响应性的重要手段。通过使用AsyncTask类,我们可以方便地执行后台耗时操作,并在主线程中更新UI。同时,我们还需要注意异步编程中的一些注意事项,如避免直接在doInBackground()方法中操作UI线程。希望本文能帮助你更好地理解和使用Android Java开发中的异步编程。

相关文章
|
20天前
|
前端开发 Java 关系型数据库
基于Java+Springboot+Vue开发的鲜花商城管理系统源码+运行
基于Java+Springboot+Vue开发的鲜花商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的鲜花商城管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。技术学习共同进步
97 7
|
1月前
|
人工智能 安全 Java
智慧工地源码,Java语言开发,微服务架构,支持分布式和集群部署,多端覆盖
智慧工地是“互联网+建筑工地”的创新模式,基于物联网、移动互联网、BIM、大数据、人工智能等技术,实现对施工现场人员、设备、材料、安全等环节的智能化管理。其解决方案涵盖数据大屏、移动APP和PC管理端,采用高性能Java微服务架构,支持分布式与集群部署,结合Redis、消息队列等技术确保系统稳定高效。通过大数据驱动决策、物联网实时监测预警及AI智能视频监控,消除数据孤岛,提升项目可控性与安全性。智慧工地提供专家级远程管理服务,助力施工质量和安全管理升级,同时依托可扩展平台、多端应用和丰富设备接口,满足多样化需求,推动建筑行业数字化转型。
67 5
|
26天前
|
人工智能 Java 定位技术
Java 开发玩转 MCP:从 Claude 自动化到 Spring AI Alibaba 生态整合
本文以原理与示例结合的形式讲解 Java 开发者如何基于 Spring AI Alibaba 框架玩转 MCP。
712 90
|
2月前
|
JavaScript Linux 网络安全
Termux安卓终端美化与开发实战:从下载到插件优化,小白也能玩转Linux
Termux是一款安卓平台上的开源终端模拟器,支持apt包管理、SSH连接及Python/Node.js/C++开发环境搭建,被誉为“手机上的Linux系统”。其特点包括零ROOT权限、跨平台开发和强大扩展性。本文详细介绍其安装准备、基础与高级环境配置、必备插件推荐、常见问题解决方法以及延伸学习资源,帮助用户充分利用Termux进行开发与学习。适用于Android 7+设备,原创内容转载请注明来源。
395 76
|
25天前
|
人工智能 Java 定位技术
Java 开发玩转 MCP:从 Claude 自动化到 Spring AI Alibaba 生态整合
本文详细讲解了Java开发者如何基于Spring AI Alibaba框架玩转MCP(Model Context Protocol),涵盖基础概念、快速体验、服务发布与调用等内容。重点包括将Spring应用发布为MCP Server(支持stdio与SSE模式)、开发MCP Client调用服务,以及在Spring AI Alibaba的OpenManus中使用MCP增强工具能力。通过实际示例,如天气查询与百度地图路线规划,展示了MCP在AI应用中的强大作用。最后总结了MCP对AI开发的意义及其在Spring AI中的实现价值。
489 9
|
2月前
|
机器学习/深度学习 人工智能 NoSQL
JAVA接入DeepSeek大模型接口开发---阿里云的百炼模型
随着大模型的越来越盛行,现在很多企业开始接入大模型的接口,今天我从java开发角度来写一个demo的示例,用于接入DeepSeek大模型,国内的大模型有很多的接入渠道,今天主要介绍下阿里云的百炼模型,因为这个模型是免费的,只要注册一个账户,就会免费送百万的token进行学习,今天就从一个简单的可以执行的示例开始进行介绍,希望可以分享给各位正在学习的同学们。
284 3
JAVA接入DeepSeek大模型接口开发---阿里云的百炼模型
|
1月前
|
人工智能 Java 物联网
没有好的学历,Java开发未来的路应该怎么走?
在数字化时代,Java开发者即使没有高学历,也能通过拥抱新兴技术(如大模型应用与鸿蒙系统开发)、积累实战经验、持续学习新技能等途径实现职业突破。从参与开源项目到关注行业动态,再到规划技术专家或管理路线,建立人脉网络并利用教育平台提升能力,开发者可拓宽技术边界,适应日新月异的技术需求,在未来发展中占据一席之地。
|
Java 开发工具 git
Java开发初级6.24.3
5.在Git使用过程中,进行Git配置的操作命令是哪个() A. config B. config -g C. config -a D. git config 相关知识点: 在git中,经常使用git config 命令用来配置git的配置文件,git配置级别主要有:仓库级别 local 【优先级最高】、用户级别 global【优先级次之】、系统级别 system【优先级最低】 正确答案:D 10.RDBMS是什么? A. Rela Database Management Systems B. Relational Database Management Systems C. Relation
167 0
|
SQL 前端开发 JavaScript
Java开发初级6.24.2
3.Java网站src/main/java目录保存的是什么资源? A. Java源代码文件 B. 测试代码 C. JavaScript、CSS等文件 D. 图片资源 正确答案:A 4.什么是索引Index? A. SQL数据库里的表管理工具 B. SQL数据库里的查询工具 C. SQL数据库里的目录工具 D. SQL数据库用来加速数据查询的特殊的数据结构 正确答案:D
216 0
|
Java
Java开发初级6.24.1
1.下面关于泛型的描述中错误的一项是? A. “? extends 类”表示设置泛型上限 B. “? super 类”表示设置泛型下限 C. 利用“?”通配符可以接收全部的泛型类型实例,但却不可修改泛型属性内容 D. 如果类在定义时使用了泛型,则在实例化类对象时需要设置相应的泛型类型,否则程序将无法编译通过 相关知识点: https://edu.aliyun.com/course/35 正确答案:D 2.下列选项中属于SVN中控制鉴权用户访问版本库的权限默认权限的是() A. write B. read C. none D. null 相关知识点: auth-access:取值范围为"writ
274 0