Android笔记——初用线程(纯笔记,LOW,无太深个人理解)

三更灯火五更鸡,正是男儿读书时。黑发不知勤学早,白首方悔读书迟。——颜真卿

因为内容还没学太深,这篇就是已学到的知识做个总结,还没做太多个人的思考研究,做下记录留给自己看以后好补充。

一、简单方法在WORK线程中更新UI

学习一段时间了,渐渐知道网络连接、IO操作之类应该放在线程中运行,而有时候这类操作过程中我们也许在某一步之后获得数据希望将这个数据显示到界面的时候,就会有些问题,你会发现若是在线程中使用setText这类影响UI的方法时程序就会产生异常,因为我们Android的UI是线程不安全的,所以我们要想更新UI就必须在主线程(UI线程)。但是正如上面举例的情况,我们有时必须要在线程中根据其中运行获得的数据更新UI,这时我们有几种比较好的解决方案:Handler、AsyncTask.....但也许你的线程内的操作并不复杂,或是只想简单的把运行结果显示到UI上,可以用下面的几个简单的方法。

Activity.runOnUiThread(Runnable action)

正如这段英文的意思“运行在UI线程”,不用多说啦,它的使用方法就是:

new Thread(new Runnable() {
    @Override
    public void run() {
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                mTestTextView.setText("啦啦啦");
            }
        });
    }
}).start();

这样就简单的在线程中更新了TextView。当然你可以在runOnUiThread的前后进行其它操作。我们可以看看它的源码:

/**
 * Runs the specified action on the UI thread. If the current thread is the UI
 * thread, then the action is executed immediately. If the current thread is
 * not the UI thread, the action is posted to the event queue of the UI thread.
 *
 * @param action the action to run on the UI thread
 */

public final void runOnUiThread(Runnable action) {
    if (Thread.currentThread() != mUiThread) {
        mHandler.post(action);
    } else {
        action.run();
    }
}

可知该方法会首先判断现在的线程是不是UI线程,是则已,不是则实际使用了Handler的post方法来处理了我们写在runOnUiThread的这段操作。

⑵View.post(Runnable action)

这个方法很尴尬的,我在做类似runOnUiThread那样的操作的时候失败了(因为我是在onCreate方法中写的线程,下面会说为啥不行),这个方法是从网上看到的和runOnUiThread一起被列出的,为啥没效果呢?我稍微研究了一下,记录一下我的错误使用,先来看看它的源码:

/**
 * <p>Causes the Runnable to be added to the message queue.
 * The runnable will be run on the user interface thread.</p>
 *
 * @param action The Runnable that will be executed.
 *
 * @return Returns true if the Runnable was successfully placed in to the
 *         message queue.  Returns false on failure, usually because the
 *         looper processing the message queue is exiting.
 *
 * @see #postDelayed
 * @see #removeCallbacks
 */
public boolean post(Runnable action) {
    final AttachInfo attachInfo = mAttachInfo;
    if (attachInfo != null) {
        return attachInfo.mHandler.post(action);
    }
    // Assume that post will succeed later
    ViewRootImpl.getRunQueue().post(action);
    return true;
}

首先可以看到该方法会先判断AttachInfo对象的内容是否为空,若不为空,则同样是用了Handler方法来处理我们的runnable,若为空,可以看到下面的注释“假设post最后能成功”(什么鬼?)首先ViewRootImpl这个类与视图的绘制有关——扩展阅读ViewRootImpl,然后可以看到其利用了其内部方法得到运行队列,且其post方法是其静态最终内部类RunQueue中自定义的post方法,而其中也有postDelayed方法,参数为(Runnable action, long delayMillis) 第二个参数就是延迟毫秒数。总之看起来View.post()方法看起来就像Handler的post方法,但为什么我像runOnUiThread那样使用却无效呢,我百度了一下查到一句这样的话“This method can be invoked from outside of the UI thread only when this View is attached to a window.”意思大概是该方法要想影响到UI线程必须等到视图(view)附加到窗口(window)。什么鬼?最后参考了一百度了一下这个“attached to a window”了解到Activity中还有个onAttachedToWindow ()方法——扩展阅读他人博客—onAttachedToWindow ()。我们可以知道我们要使用View.post()这个方法就必须等onAttachedToWindow ()这个方法执行过了才行,且启动程序时这个onAttachedToWindow ()运行在onResume()之后,而我在onCreate中开线程异步操作可能在TextView还没attached to a window就post了,所以没有产生丝毫的效果。有兴趣的朋友可以看上面的大佬的博客多了解一下,我理解的可能不太对。(这个方法我最后实验成功是把线程写在一个按钮的点击监听器里,我点击的时候自然已经“attached to a window”了)

LOG

⑶View.postDelayed(Runnable action, long delayMillis)

这个就是上一个方法的延时方法,第二个参数写要延时的毫秒数就行了,1000就是1秒。

二、利用线程进行简单的下载(就是做个笔记)

直接上代码吧:
  DownloadActivity.class

import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.lang.ref.WeakReference;
import java.net.URL;
import java.net.URLConnection;

public class DownloadActivity extends AppCompatActivity {

    public static final String TAG = DownloadActivity.class.getSimpleName() + "-TAG";
    private Button mDownloadButton;
    private ProgressBar mDownloadProgressBar;
    private Handler mHandler = new DownloadHandler(this);
    private TextView mProgressTextView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_download);
        findViews();
        mDownloadButton.setOnClickListener(myClick);
    }

    private void findViews() {
        mDownloadButton = (Button) findViewById(R.id.btn_download);
        mDownloadProgressBar = (ProgressBar) findViewById(R.id.progressBar_download);
        mProgressTextView = (TextView) findViewById(R.id.textView_progress);
    }

    private View.OnClickListener myClick = new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            switch (v.getId()) {
                case R.id.btn_download:
                    new Thread(new Runnable() {
                        @Override
                        public void run() {
                            try {
                                URL url = new URL("http://download.qidian.com/epub/3666197.epub");//下载链接
                                URLConnection connection = url.openConnection();
                                InputStream downloadStream = connection.getInputStream();
                                int contentLength = connection.getContentLength();//获得文件长度
                                Log.i(TAG, "contentLength: " + contentLength);
                                if (contentLength <= 0) {
                                    return;
                                }
                                //文件存放路径
                                String downloadFolderName = Environment.getExternalStorageDirectory() + File.separator + "qdf" + File.separator;
                                Log.i(TAG, "下载路径  " + downloadFolderName);
                                File file = new File(downloadFolderName);
                                if (!file.exists()) {
                                    //如果目录不存在则创建目录
                                    file.mkdir();
                                }
                                //文件目录+文件名
                                String fileName = downloadFolderName + "test.txt";
                                File contentFile = new File(fileName);
                                if (contentFile.exists()) {
                                    contentFile.delete();
                                }

                                int downloadSize = 0;

                                byte[] bytes = new byte[1024];
                                int length;

                                //字节输出流
                                OutputStream outputStream = new FileOutputStream(fileName);

                                while ((length = downloadStream.read(bytes)) != -1) {
                                    outputStream.write(bytes, 0, length);
                                    downloadSize += length;
                                    int progress = downloadSize * 100 / contentLength;
                                    //更新UI
                                    Message message = mHandler.obtainMessage();
                                    message.what = 0;
                                    message.obj = progress;
                                    mHandler.sendMessage(message);
                                    //ProgressBar可以在线程中操作
                                    mDownloadProgressBar.setProgress(progress);
                                }
                                Log.i(TAG, "download success");
                                downloadStream.close();
                                outputStream.close();
                            } catch (IOException e) {
                                e.printStackTrace();
                            }

                        }
                    }).start();
                    break;
            }
        }
    };

    public TextView getProgressTextView() {
        return mProgressTextView;
    }


    public static class DownloadHandler extends Handler {

        public final WeakReference<DownloadActivity> mDownloadActivityWeakReference;

        public DownloadHandler(DownloadActivity downloadActivity) {
            mDownloadActivityWeakReference = new WeakReference<>(downloadActivity);
        }

        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            DownloadActivity activity = mDownloadActivityWeakReference.get();

            switch (msg.what) {
                case 0:
                    int progress = (int) msg.obj;
                    String progress2 = progress + "%";
                    activity.getProgressTextView().setText(progress2);
                    if (progress == 100) {
                        Toast.makeText(activity, "下载成功", Toast.LENGTH_SHORT).show();
                    }
                    break;
            }
        }
    }
}

网络连接、IO操作自然是要在线程中啦,连接网络后通过URLConnection 对象来获得输入流,并可以利用getContentLength()方法来获得文件大小。最后利用字节输出流输出文件,其中根据已下载的大小与总大小之比来更新ProgressBar,因为是在线程中获取数据后立马更新UI所以使用了Handler。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容