handlerthread线程分出来后,Handler怎么写

handlerthread  时间:2021-06-19  阅读:()

android可以再handler里运行线程么

在handler(二)中点击打开链接,我们说handler是开启了另外一个线程,而且看代码的话确实是这样,实现了runnable接口,这在java中就是开启了一个线程,但是情况中的是这样吗?我们不妨来做个试验,如下 [java] view plaincopy <span style="font-size:14px;">.handlerThread; import android.app.Activity; import android.os.Bundle; import android.os.Handler; public class handlerThread extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); Handler handler = new Handler(); handler.post(r); System.out.println("activity线程ID:"+Thread.currentThread().getId()); System.out.println("activity线程name:"+Thread.currentThread().getName()); } Runnable r = new Runnable(){ @Override public void run() { // TODO Auto-generated method stub System.out.println("handler线程ID:"+Thread.currentThread().getId()); System.out.println("handler线程name:"+Thread.currentThread().getName()); try { Thread.sleep(3000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }; }</span> 运行结果 其实,当我看到这里的时候也不敢相信,但是事实就是这样,handler没有重新开启一个线程,而是跟activity在同一个线程里,但是这种写法也就非常接近java的标准线程的写法了,难怪会误导人,如下是java的标准线程写法。

[java] view plaincopy <span style="font-size:14px;">.handlerThread; import android.app.Activity; import android.os.Bundle; import android.os.Handler; public class handlerThread extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); // Handler handler = new Handler(); // handler.post(r); Thread t = new Thread(r); t.start(); System.out.println("activity线程ID:"+Thread.currentThread().getId()); System.out.println("activity线程name:"+Thread.currentThread().getName()); } Runnable r = new Runnable(){ @Override public void run() { // TODO Auto-generated method stub System.out.println("handler线程ID:"+Thread.currentThread().getId()); System.out.println("handler线程name:"+Thread.currentThread().getName()); try { Thread.sleep(3000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }; }</span> 运行结果如下: 这里就才是我们这种想要的结果,两者比较我们就会发现,handler虽然实现了runnable接口,但是却并没有启动一个线程,而是直接调用run方法。

那andriod为什么要这样设计呢,既然不启动新的线程,为什么还要多此一举来实现runnable接口呢,我们继续探讨,下次再说。

handlerthread会泄漏吗

你好,handlerthread是会泄漏的
  • 什么是java的内存泄漏? Java内存泄漏指的是进程中某些对象已经没有使用价值并且想将其释放回收掉,但是它们却可以直接或间接地被其他对象强引用,导致无法被GC回收。

    无用的对象占据着内存空间,使得实际可使用内存变小,形象地说法就是内存泄漏。

  • 为什么我们上面的代码会产生内存泄漏? 1、在程序启动的时候就在主线程中创建了一个Looper 对象,它内部维护着一个消息队列,并且一条一条的对消息进行处理。

    ? 2、当我们发送消息的时候,在Message的target里面存放着发送该消息的handler对象,即消息里面包含了一个Handler实例的引用,并且就是通过这个handler实例来回调handleMessage方法进行处理。

    ? 如果对上面两点不明白的话,可以看看Looper与Handler解析? 3、在Java中,非静态的内部类和匿名内部类都会隐式地持有其外部类的引用。

    静态的内部类不会持有外部类的引用。

    ? 理解了上面的三点,应该就差不多清楚了为什么上面的代码会出现内存泄漏。

    ?
  • 如何解决这个问题?? 也很简单,上面说到要避免使用非静态的内部类,那我们就使用静态的内部类,或者把内部类单独写个文件,让它成为一个单独的类。

    另外,我们可以在里面增加一个成员变量来弱引用外部类实例,就可以调用外部类的方法。

    ? 第一种改进方法:使用静态内部类 第二种改进方法:单独定义一个类 希望对你有帮助

    android.os.HandlerThread与java.lang.Thread的区别在哪?(最好详细点)

    HandlerThread是继承Thread,主要的作用是建立了一个线程,并且创立了消息队列,有来自己的looper,可以让我们在自己的线程中分发和处理消息。

    android的消息处理通过handler和looper, HandlerThread不但能提供异步处理,Handler处理消息的方法也会在这个线程中执行,他最要的作用就是提供了一个线程。

    Handler Runnable和Thread之间的区别和联系 应用

    在java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口;Thread类是在java.lang包中定义 的。

    一个类只要继承了Thread类同时覆写了本类中的run()方法就可以实现多线程操作了,但是一个类只能继承一个父类,这是此方法的局限。

    下面看例子: .thread.demo; class MyThread extends Thread{ private String name; public MyThread(String name) { super(); this.name = name; } public void run(){ for(int i=0;i<10;i++){ System.out.println("线程开始:"+this.name+",i="+i); } } } .thread.demo; public class ThreadDemo01 { public static void main(String[] args) { MyThread mt1=new MyThread("线程a"); MyThread mt2=new MyThread("线程b"); mt1.run(); mt2.run(); } } 但是,此时结果很有规律,先第一个对象执行,然后第二个对象执行,并没有相互运行。

    在JDK的文档中可以发现,一旦调用start()方法,则会通过JVM找到run()方法。

    下面启动start()方法启动线程: .thread.demo; public class ThreadDemo01 { public static void main(String[] args) { MyThread mt1=new MyThread("线程a"); MyThread mt2=new MyThread("线程b"); mt1.start(); mt2.start(); } }; 这样程序可以正常完成交互式运行。

    那么为啥非要使用start();方法启动多线程呢? 在JDK的安装路径下,src.zip是全部的java源程序,通过此代码找到Thread中的start()方法的定义,可以发现此方法中使用了 private native void start0();其中native关键字表示可以调用操作系统的底层函数,那么这样的技术成为JNI技术(java Native Interface) Runnable接口 在实际开发中一个多线程的操作很少使用Thread类,而是通过Runnable接口完成。

    public interface Runnable{ public void run(); } 例子: .runnable.demo; class MyThread implements Runnable{ private String name; public MyThread(String name) { this.name = name; }public void run(){ for(int i=0;i<100;i++){ System.out.println("线程开始:"+this.name+",i="+i); } } }; 但是在使用Runnable定义的子类中没有start()方法,只有Thread类中才有。

    此时观察Thread类,有一个构造方 法:public Thread(Runnable targer)此构造方法接受Runnable的子类实例,也就是说可以通过Thread类来启动Runnable实现的多线程。

    (start()可以协 调系统的资源): .runnable.demo; .runnable.demo.MyThread; public class ThreadDemo01 { public static void main(String[] args) { MyThread mt1=new MyThread("线程a"); MyThread mt2=new MyThread("线程b"); new Thread(mt1).start(); new Thread(mt2).start(); } } 两种实现方式的区别和联系: 在程序开发中只要是多线程肯定永远以实现Runnable接口为主,因为实现Runnable接口相比继承Thread类有如下好处: 避免点继承的局限,一个类可以继承多个接口。

    适合于资源的共享 以卖票程序为例,通过Thread类完成: .demo.dff; class MyThread extends Thread{ private int ticket=10; public void run(){ for(int i=0;i<20;i++){ if(this.ticket>0){ System.out.println("卖票:ticket"+this.ticket--); } } } }; 下面通过三个线程对象,同时卖票: .demo.dff; public class ThreadTicket { public static void main(String[] args) { MyThread mt1=new MyThread(); MyThread mt2=new MyThread(); MyThread mt3=new MyThread(); mt1.start();//每个线程都各卖了10张,共卖了30张票 mt2.start();//但实际只有10张票,每个线程都卖自己的票 mt3.start();//没有达到资源共享 } } 如果用Runnable就可以实现资源共享,下面看例子: .demo.runnable; class MyThread implements Runnable{ private int ticket=10; public void run(){ for(int i=0;i<20;i++){ if(this.ticket>0){ System.out.println("卖票:ticket"+this.ticket--); } } } } .demo.runnable; public class RunnableTicket { public static void main(String[] args) { MyThread mt=new MyThread(); new Thread(mt).start();//同一个mt,但是在Thread中就不可以,如果用同一 new Thread(mt).start();//个实例化对象mt,就会出现异常 new Thread(mt).start(); } }; 虽然现在程序中有三个线程,但是一共卖了10张票,也就是说使用Runnable实现多线程可以达到资源共享目的。

    Runnable接口和Thread之间的联系: public class Thread extends Object implements Runnable 发现Thread类也是Runnable接口的子类。

    第二: Thread是系统给你的资源,有了Thread你才有从CPU那里得到可执行时间片的权力, Thread并不认识你的程序,不知道有test 这样的类,因为编序员有千千万,每个人命名都不一样,想要做的事都不一样, 所以 Thread只认识一个! 那就是Runnable 。

    Thread认识Runnable 并且知道Runnable 里面有一个run方法. 一旦调用Thread的start方法,Runnable 方法里的run就会被Thread自动运行。

    所以,当我们把我们的类继承(这里应该叫实现接口)自Runnable 的时候,我们的程序就是属于Runnable 一个类型的了。

    虽然是Runnable 的子类,但人家认识你爸爸,当然也知道了你。

    Thread可以不管你内部有什么情况,他只管你有run()方法就行了,他就调start让你去运行run 所以我们在run里面写点东西,这样就可以让系统运行我们想要做的代码了。

    是不是很通俗很易懂呢? 所以要运行线程的步骤是, 1。

    生成我们自己的类对象 2。

    从系统那里得到Thread 3。

    让Threa调我们的类对象,让其start起来 代码: test a=new test(); Thread thread=new Thread(a); //Thread需要一个参数,就是你编的线程类,这样他就认识了你的线程,也有资格向系统申请拿到CPU时间片thread.start(); 你可以简单点写: new Thread(a).start(); 第三: Runnable 并不一定是新开一个线程,比如下面的调用方法就是运行在UI主线程中的: Handler mHandler=new Handler(); mHandler.post(new Runnable(){ @Override public void run() { // TODO Auto-generated method stub } }); 官方对这个方法的解释如下,注意其中的:“The runnable will be run on the user interface thread. ” boolean Android.view.View .post(Runnable action) Causes the Runnable to be added to the message queue. The runnable will be run on the user interface thread. Parameters: action The Runnable that will be executed. Returns: Returns true if the Runnable was essfully placed in to the message queue. Returns false on failure, usually because the looper processing the message queue is exiting. 我们可以通过调用handler的post方法,把Runnable对象(一般是Runnable的子类)传过去;handler会在looper中调用这个Runnable的Run方法执行。

    Runnable是一个接口,不是一个线程,一般线程会实现Runnable。

    所以如果我们使用匿名内部类是运行在UI主线程的,如果我们使用实现这个Runnable接口的线程类,则是运行在对应线程的。

    具体来说,这个函数的工作原理如下: View.post(Runnable)方法。

    在post(Runnable action)方法里,View获得当前线程(即UI线程)的Handler,然后将action对象post到Handler里。

    在Handler里, 它将传递过来的action对象包装成一个Message(Message的callback为action),然后将其投入UI线程的消息循环中。

    在 Handler再次处理该Message时,有一条分支(未解释的那条)就是为它所设,直接调用runnable的run方法。

    而此时,已经路由到UI线 程里,因此,我们可以毫无顾虑的来更新UI。

    如下图,前面看到的代码,我们这里Message的callback为一个Runnable的匿名内部类 这种情况下,由于不是在新的线程中使用,所以千万别做复杂的计算逻辑。

    第四:在多线程编程这块,我们经常要使用Handler,Thread和Runnable这三个类,那么他们之间的关系你是否弄清楚了呢?   首先说明Android的CPU分配的最小单元是线程,Handler一般是在某个线程里创建的,因而Handler和Thread就是相互绑定的,一一对应。

      而Runnable是一个接口,Thread是Runnable的子类。

    所以说,他俩都算一个进程。

      HandlerThread顾名思义就是可以处理消息循环的线程,他是一个拥有Looper的线程,可以处理消息循环。

      与其说Handler和一个线程绑定,不如说Handler是和Looper一一对应的。

      最后需要说明的是,在UI线程(主线程)中:   mHandler=new Handler();   mHandler.post(new Runnable(){   void run(){   //执行代码...   }   });   这个线程其实是在UI线程之内运行的,并没有新建线程。

      常见的新建线程的方法是:   Thread thread = new Thread();   thread.start();   HandlerThread thread = new HandlerThread("string");   thread.start(); 第五:Java Runnable接口在进行相关编写的时候需要我们不断的学习相关代码。

    下面我们就来看炫如何才能使用相关的代码。

    Runnable接口只有一个方法 run(),我们声明自己的类实现Runnable接 口并提供这一方法,将我们的线程代码写入其中,就完成了这一部分的任务。

      但是Runnable接口并没有任何对线程的支持,我们还必须创建Thread类 的实例,这一点通过Thread类的构造函数public Thread(Runnable target);来实现。

    下面是一个例子:   1.public class MyThread implements Runnable   2.{   3.int count= 1, number;   4.public MyThread(int num)   5.{   6.numnumber = num;   7.System.out.println("创建线程 " + number);   8.}   9.public void run()   10.{   11.while(true)   12.{   13.System.out.println   14.("线程 " + number + ":计数 " + count);   15.if(++count== 6) return;   16.}   17.}   18.public static void main(String args[])   19.{   20.for(int i = 0; i 〈 5;   21.i++) new Thread(new MyThread(i+1)).start();   22.}   23.}   严格地说,创建Thread子类的实例也是可行的,但是必须注意的是,该子类必须没有覆盖 Thread 类的 run 方法,否则该线程执行的将是子类的 run 方法,而不是我们用以实现Runnable 接口的类的 run 方法,对此大家不妨试验一下。

      使用 Java Runnable接口来实现多线程使得我们能够在一个类中包容所有的代码,有利于封装,它的缺点在于,我们只能使用一套代码,若想创建多个线程并使各个线 程执行不同的代 码,则仍必须额外创建类,如果这样的话,在大多数情况下也许还不如直接用多个类分别继承 Thread 来得紧凑。

    线程分出来后,Handler怎么写

    是在新的线程类里面定义构造方法吗 public GetImageThread(Handler handler) { super(); this.handler = handler; }那MainActivity该怎么写
  • HostRound:美国达拉斯/洛杉矶/纽约/荷兰大硬盘服务器,1TB NVMe+4TB HDD,$179/月

    hostround怎么样?大硬盘服务器,高防服务器。hostround,美国商家,2017年成立,正规注册公司(Company File #6180543),提供虚拟主机、VPS云主机、美国服务器、荷兰服务器租用等。现在有1款特价大硬盘独服,位于达拉斯,配置还不错,本月订购时包括免费 500Gbps DDoS 保护,有兴趣的可以关注一下。点击直达:hostround官方网站地址美国\荷兰独立服务器...

    Dynadot多种后缀优惠域名优惠码 ,.COM域名注册$6.99

    Dynadot 是一家非常靠谱的域名注册商家,老唐也从来不会掩饰对其的喜爱,目前我个人大部分域名都在 Dynadot,还有一小部分在 NameCheap 和腾讯云。本文分享一下 Dynadot 最新域名优惠码,包括 .COM,.NET 等主流后缀的优惠码,以及一些新顶级后缀的优惠。对于域名优惠,NameCheap 的新后缀促销比较多,而 Dynadot 则是对于主流后缀的促销比较多,所以可以各取所...

    DediPath($1.40),OpenVZ架构 1GB内存

    DediPath 商家成立时间也不过三五年,商家提供的云服务器产品有包括KVM和OPENVZ架构的VPS主机。翻看前面的文章有几次提到这个商家其中机房还是比较多的。其实对于OPENVZ架构的VPS主机以前我们是遇到比较多,只不过这几年很多商家都陆续的全部用KVM和XEN架构替代。这次DediPath商家有基于OPENVZ架构提供低价的VPS主机。这次四折的促销活动不包括512MB内存方案。第一、D...

    handlerthread为你推荐
    建行手机网站建设手机银行首次怎样登录存储区域网络nas、das、san 三种网络环境的区分动态图片格式怎么将GIF动态图片的格式变成JPG 动态效果不变?腾讯合作伙伴大会如何成为腾讯渠道合作伙伴?医院排队系统医院排队机和医院排队机的区别有哪些?泛微协同办公系统泛微软件怎么样?做协同办公的,我要来这做销售前景怎么样?请大家对这个行业或公司了解的给些建议。云家政网腾讯网的网址是多少?aviconverter跪求AVIConverter V1.0下载地址flex是什么Adobe旗下的软件分别是干什么的?网页背景音乐代码网页背景音乐代码,我要哪怕转换网页都不间断的那种
    怎样申请域名 高防dns hostmonster 天猫双十一抢红包 全站静态化 gspeed 申请免费空间和域名 跟踪路由命令 中国电信网络测速 国内域名 永久免费空间 主机返佣 阿里云邮箱怎么注册 hdroad 发证机构 qq部落18-3 厦门电信宽带 北京市机动车牌照申请网站 免费ftp服务器软件 永久免费网络游戏 更多