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该怎么写
  • 湖北22元/月(昔日数据)云服务器,国内湖北十堰云服务器,首月6折

    昔日数据怎么样?昔日数据新上了湖北十堰云服务器,湖北十堰市IDC数据中心 母鸡采用e5 2651v2 SSD MLC企业硬盘 rdid5阵列为数据护航 100G高防 超出防御峰值空路由2小时 不限制流量。目前,国内湖北十堰云服务器,首月6折火热销售限量30台价格低至22元/月。(注意:之前有个xrhost.cn也叫昔日数据,已经打不开了,一看网站LOGO和名称为同一家,有一定风险,所以尽量不要选择...

    物语云-VPS-美国洛杉矶VPS无限流量云windows大带宽100M不限流量 26/月起

    物语云计算怎么样?物语云计算(MonogatariCloud)是一家成立于2016年的老牌国人商家,主营国内游戏高防独服业务,拥有多家机房资源,产品质量过硬,颇有一定口碑。本次带来的是特惠活动为美国洛杉矶Cera机房的不限流量大带宽VPS,去程直连回程4837,支持免费安装Windows系统。值得注意的是,物语云采用的虚拟化技术为Hyper-v,因此并不会超售超开。一、物语云官网点击此处进入物语云...

    提速啦(900元/月),杭州BGP E5-2665/89*2 32核 48G 100G防御

    提速啦的来历提速啦是 网站 本着“良心 便宜 稳定”的初衷 为小白用户避免被坑提速啦的市场定位提速啦主要代理市场稳定速度的云服务器产品,避免新手购买云服务器的时候众多商家不知道如何选择,妮妮云就帮你选择好了产品,无需承担购买风险,不用担心出现被跑路 被诈骗的情况。提速啦的售后保证提速啦退款 通过于合作商的友好协商,云服务器提供3天内全额退款,超过3天不退款 物理机部分支持当天全额退款提速啦提现 充...

    handlerthread为你推荐
    snake模型图像分割与边缘处理有什么区别?请大侠回答,谢谢了。爱短信官网官方飞信,ET飞信,爱短信飞信插件哪个好用??youtube创始人比特币创始人到底是谁微信如何只发文字微信朋友圈如何只发文字,怎么发文字不要图comexception5种常见的Exception!gas是什么意思petrol和gas的区别东兴证券网站东兴证券超强版下载怎样删除聊天记录自己已发出的微信聊天记录怎样删除才不会让对方看见怎样删除聊天记录怎么批量清除微信聊天记录空间导航怎么把空间的导航变成只有留言板跟相册
    长沙域名注册公司 如何查询ip地址 企业域名备案 godaddy域名解析教程 godaddy优惠码 灵动鬼影 ibox官网 架设服务器 idc是什么 hkt 环聊 跟踪路由命令 阿里云官方网站 西安服务器托管 电信网络测速器 谷歌台湾 主机返佣 1美元 带宽测试 网络速度 更多