Android多线程与异步操作实现图片自动轮播

一、实验内容

1.利用Thread和Handler实现异步操作;
2.利用AsyncTast实现异步操作;

1. 利用Thread和Handler实现异步操作

(1)新建一个安卓工程,在默认的layout主界面中添加一个按钮(Button),一个图片显示控件(ImageView)。
(2)将实验四资源中的所有图片导入到工程的drawable目录下。
(3)参考下面的代码构建图片资源列表和计数器变量。
在这里插入图片描述
(4)为按钮添加点击事件,当点击按钮时,启动一个新线程(传入一个handler),并每隔100毫秒发送一条空信息,信息的what标识为0x123。
(5)定义一个handler用于接受线程的信息,并在handleMessage方法中处理信息,参考下列代码循环更新ImageView中的内容,实现动画效果。
在这里插入图片描述
(6)测试上述功能。
在这里插入图片描述在这里插入图片描述

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="100dp"
        android:layout_marginLeft="100dp"
        android:layout_marginTop="200dp"
        android:text="Button"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />


    <ImageView
        android:id="@+id/imageView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:scaleType="fitXY"
        tools:ignore="MissingConstraints" />

</androidx.constraintlayout.widget.ConstraintLayout>
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">


    <ImageView
        android:id="@+id/imageView"
        android:layout_width="300px"
        android:layout_height="300px"
       android:scaleType="fitXY"
        tools:ignore="MissingConstraints" />
</androidx.constraintlayout.widget.ConstraintLayout>
package com.example.myapplication8;

public class images {
    public final static int imgids[]=new int[]{//图片资源列表
            R.drawable.p1,R.drawable.p2,R.drawable.p3,R.drawable.p4,
            R.drawable.p5,R.drawable.p6,R.drawable.p7
    };
}

package com.example.myapplication8;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;

import androidx.annotation.NonNull;
import androidx.viewpager.widget.PagerAdapter;

public class ViewPagerAdapter extends PagerAdapter {
    private Context context;
    private LayoutInflater layoutInflater;
    private  int[] datas;

    public ViewPagerAdapter(Context context,int[] datas)
    {
        this.context=context;
        this.datas=datas;
        layoutInflater=LayoutInflater.from(context);
    }
    @Override
    public int getCount() {
        return datas.length;
    }

    @Override
    public boolean isViewFromObject(@NonNull View view, @NonNull Object object) {
        return view==object;
    }

    @Override
    public void destroyItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
        container.removeView((View) object);
    }

    @NonNull
    @Override
    public Object instantiateItem(@NonNull ViewGroup container, int position) {
        View layout=layoutInflater.inflate(R.layout.viewpager_item,null);
        ImageView iv= layout.findViewById(R.id.imageView);
        iv.setImageResource(datas[position]);
        container.addView(layout);
        return layout;

    }
}

package com.example.myapplication8;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.viewpager.widget.ViewPager;

import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.ImageButton;
import android.widget.ImageView;

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class MainActivity extends AppCompatActivity {
    private int[] imgids= com.example.myapplication8.images.imgids;
    private ImageView imageView;
    public Button btn;
    int imgstart=0;
    private ScheduledExecutorService scheduledExecutorService;
    private Handler handler =new Handler()
    {
        @Override
        public void handleMessage(@NonNull Message msg) {
            switch (msg.what)
            { case 0x123:
                imageView.setImageResource(imgids[imgstart++%7]);
                break;
            }
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        imageView=findViewById(R.id.imageView1);
    }

    @Override
    protected void onStart() {
        super.onStart();
        btn = findViewById(R.id.button);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                scheduledExecutorService = Executors.newSingleThreadScheduledExecutor();
                scheduledExecutorService.scheduleAtFixedRate(new Runnable() {
                    @Override
                    public void run() {
                        handler.sendEmptyMessage(0x123);
                    }
                }, 0, 100, TimeUnit.MILLISECONDS);
            }
        });
    }

    @Override
    protected void onStop() {
        super.onStop();
        if (scheduledExecutorService != null)
        {
            scheduledExecutorService.shutdown();
        }
    }
}

实现过程即通过 scheduledExecutorService的scheduleAtFixedRate()来利用hander的sendEmptyMessage()方法传递what标识符,然后在handleMessage()中switch判断标识符,并修改imageView的图片。

2. 利用AsyncTast实现异步操作

(1)新建另一个安卓工程,在默认的layout主界面中添加一个按钮(Button),一个图片显示控件(ImageView)。
(2)参考课堂教授的方法,使用AsyncTast实现类似的动画效果。
(3)测试上述功能,比较两种方法在代码构造上的优劣。
在这里插入图片描述在这里插入图片描述

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="100dp"
        android:layout_marginLeft="100dp"
        android:layout_marginTop="200dp"
        android:text="Button"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />


    <ImageView
        android:id="@+id/imageView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:scaleType="fitXY"
        tools:ignore="MissingConstraints" />

</androidx.constraintlayout.widget.ConstraintLayout>
package com.example.myapplication;

import androidx.appcompat.app.AppCompatActivity;

import android.os.AsyncTask;
import android.os.Bundle;
import android.text.PrecomputedText;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;

public class MainActivity extends AppCompatActivity {
    private int[] imgids= {//图片资源列表
            R.drawable.p1,R.drawable.p2,R.drawable.p3,R.drawable.p4,
            R.drawable.p5,R.drawable.p6,R.drawable.p7
    };;
    private ImageView imageView;
    public Button btn;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        imageView=findViewById(R.id.imageView1);
        btn=findViewById(R.id.button);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
            Task task=new Task();
            task.execute();
            }
        });
    }
    class Task extends AsyncTask<Void,Integer,Boolean>
   {    int imgstart;

       @Override
       protected void onPreExecute() {
           super.onPreExecute();
       }

       @Override
       protected Boolean doInBackground(Void... params){

           try {
               while(true)
               {
                   Thread.sleep(100);
                   imgstart++;
                   imgstart=imgstart%7;
                   publishProgress(imgstart);
                   if(imgstart==8)
                   {
                        break;
                   }
               }

           } catch (InterruptedException e) {
               e.printStackTrace();
           }
            return true;

       }

       @Override
       protected void onProgressUpdate(Integer... values) {
           super.onProgressUpdate(values);
           imageView.setImageResource(imgids[values[0]]);
       }

       @Override
       protected void onPostExecute(Boolean bollean) {
           super.onPostExecute(bollean);
       }
   }
}

四: 实验总结:

在本实验中,我对多线程与异步操作的实现有了基本了解,并知道了2种线程的实现方法,即利用handle与 scheduledExecutorService和继承AsyncTask的方法,我认为这两种方法都比较好用,但AsyncTask虽然结构比较清晰,但要注意参数配置,还必须在onProgressUpdate()中才能对Android的UI进行更新,容易出错,而handle利用时虽然代码比较复杂,但实验实现时不易出错。

  • 4
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
线程是当前计算机程序开发中非常重要的技术,能够有效提高程序的运行效率和响应速度。Delphi是一种编程语言,支持多线程开发,可以利用多线程来实现一些并发的任务。 在CSDN上有不少关于Delphi多线程开发的范例可以参考。比如,可以利用多线程来实现文件下载功能。在主线程中创建一个子线程来执行文件下载的任务,这样就可以让主线程继续执行其他的任务,提高程序的并发性和用户体验。 另外,还可以利用多线程来实现大数据计算任务。将大数据分成多个子任务,每个子任务由一个线程处理,通过多线程并发执行这些子任务,可以大大提高计算的效率。 此外,还可以利用多线程来实现网络通信功能。创建一个线程用来监听网络连接,当有客户端连接请求时,创建一个新的线程来处理该请求,这样可以实现同时处理多个网络请求的功能。 在Delphi中,可以使用TThread类来创建和控制线程。可以通过重写Execute方法来实现线程的具体功能,通过Synchronize方法来实现线程之间的同步通信,通过阻塞和唤醒机制来实现线程之间的调度和协作。 总而言之,Delphi可以利用多线程来实现各种不同的功能,包括文件下载、大数据计算和网络通信等。对于多线程的开发,需要合理设计和管理线程,处理好线程间的通信和同步,充分利用多核处理器的计算资源,提高程序的效率和性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Pistachiout

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值