一、多线程概述

一个进程中至少有一个线程,每一个线程都有自己运行的内容,这个内容可以称为线程要执行的任务。

不能没一个问题都使用多线程,能使用单线程解决的问题就不要使用多线程解决。

使用多线程的弊端:

一旦开启了多个程序,电脑就会变卡,原因就是每个程序被CPU处理的几率变小了。最明显的例子就是如果在复制文件的时候如果开启了多个其他程序,则复制文件所需要的时间就会明显变长。

使用多核CPU可以解决一部分问题,它们的瓶颈就是内存。

在学习多线程之前我们写的Java程序也是多线程的,只不过除了主线程之外的其他线程是虚拟机隐式调用的。最明显的例子就是垃圾回收机制。

Object类中有一个方法名为finalize,该方法由垃圾回收器自动调用,不需要程序员维护。

需要注意的是,堆中产生垃圾之后并不会立刻被垃圾回收器回收,而是经过一段时间后才会被回收。这么做的原因就是如果立即回收,就会和主方法争夺CPU的执行权,影响程序执行效率,这在垃圾产生较为频繁的程序中尤其明显。

演示垃圾回收机制:

Demo类:

 public class Demo {
public int data;
public Demo(int data)
{
this.data=data;
}
public void finalize()
{
System.out.println(this.data);
} }

Main类:

 public class Main {
public static void main(String args[])
{
new Demo(1);
new Demo(2);
new Demo(3);
new Demo(4);
System.gc();
System.out.println("Hello,World!");
}
}

使用System.gc()方法可以告知垃圾回收器已经产生垃圾了,但是垃圾回收器可能不会立即启动。

我们使用多线程的目的就是让程序并发执行两段或者两段以上代码。

二、线程创建的方式一:继承Thread类

Thread类位于lang包下,而且不是抽象类,因此使用起来很方便。

创建线程的目的是为了开启一条执行路径,去运行指定的代码并和其他代码实现同时运行。

Thread类用于描述线程,而线程是需要任务的,这个任务由run方法体现。也就是说run方法就是封装自定义线程任务的函数。

使用Thread类创建线程的步骤:

1.定义一个类继承Thread类。

2.覆盖Thread类的run方法

3.创建Thread子类对象。

4.调用start方法启动线程。

Thread类中有两个方法:getName方法用于获取线程名,currentThread用于获取当前线程。前者是非静态类,后者是静态类。

代码演示:

 class Demo extends Thread
{
public void run()
{
for(int i=1;i<=3;i++)
{
System.out.println("Thread_Name="+Thread.currentThread().getName());
}
}
}
public class Main
{
public static void main(String args[])
{
new Demo().start();
new Demo().start();
for(int i=1;i<=3;i++)
{
System.out.println("Thread_Name="+Thread.currentThread().getName());
}
}
}

运行结果:

应当注意,虽然main方法已经结束,但是仍然打印出了Thread-0,是因为每个线程在内存中都会开辟一个单独的栈,栈与栈之间互不影响。

三、线程的几种状态。

线程一共有五种状态:创建、运行、临时堵塞、冻结、消亡。

我们将线程的权限划分为CPU执行资格和CPU执行权。

各个状态之间的转换关系如下:

一个线程如果具备CPU执行资格,它可以在处理队列中排队等待CPU的执行。

一个线程如果具备CPU执行权,说明它正在被CPU执行,并且它一定具备CPU执行资格。

运行状态:具备CPU执行资格和CPU执行权。

冻结状态:既不具备CPU执行资格也不具备CPU执行权。

堵塞状态:具备CPU执行资格,但是不具备CPU执行权。

举例:

老师回答学生问题:

学生排成一队,挨个问老师问题,其中睡着了的同学自动从队伍中剔除出去(这类似于CPU不会关心处于冻结状态的线程),提出的问题正在被老师解答的学生具备着等待老师回答问题的权利和要求老师回答问题的权利(这类似于出于运行状态的线程同时具有CPU执行权和CPU执行资格),正处于队列中的其他同学则等待老师解答问题(这类似于出于堵塞状态的线程正在等待CPU执行权)。

使得线程出于冻结状态的方法有两种:sleep方法,这种方法时间一到,线程自动加入到堵塞队列中,等待CPU执行权;wait方法,这种方法使得线程彻底成为植物人,除非调用notify方法或者notifyAll方法,否则一直处于冻结状态。

四、创建线程的第二种方式:实现Runnable接口

Runnable接口中只有一个方法run,它和Thread类结合可以实现创建新线程的功能。

使用Runnable接口创建新线程的步骤是:

1.覆盖接口的run方法,将线程的任务代码封装到run方法中。
2.通过Thread类创建线程对象,并将Runnable接口的子类对象作为Thread累的构造函数进行传递。
为什么?因为线程的任务都封装在Runnable接口的run方法中。
所以要在线程对象创建时就必须明确要运行的任务。
也就是说,如果你不传递,则线程对象调用自己的run(new Thread())方法,如果你传递,则线程对象调用传递的run(new Thread(target))方法。
3.调用线程对象的start方法开启线程。

使用Runnable接口的方法:

 class Demo implements Runnable
{
public void run()
{
for(int i=1;i<=3;i++)
{
System.out.println("Thread_Name="+Thread.currentThread().getName());
}
}
}
public class Main
{
public static void main(String args[])
{
new Thread(new Demo()).start();
new Thread(new Demo()).start();
for(int i=1;i<=3;i++)
{
System.out.println("Thread_Name="+Thread.currentThread().getName());
}
}
}

运行结果:

对于Thread类的简单模拟:

 class Thread implements Runnable
{
private Runnable target=null;
public Thread(){}
public Thread(Runnable target)
{
this.target=target;
}
public void run()
{
if(this.target)
{
this.target.run();
}
}
public void start()
{
this.run();
}
}

我们可以看到,如果给了实现了Runnable接口的对象,则调用接口中的run方法,否则,不处理,即调用Thread的空的run方法。

使用Runnable方式的好处是什么?

1.将线程的任务从线程的子类中分离出来,进行了单独的封装。
按照面向对象的思想将任务封装成对象。
2.避免了java单继承的局限性。

所以实际开发中大多使用实现Runnable接口的方式创建线程,而不是继承Thread类,这符合少用继承、多用组合原则。

五、综合案例:通过买票系统比较两种方式的异同

需求:将10张票分发给4个站点售出。

1.使用Thread的方式:

 class Ticket extends Thread
{
public static int sum=10;
public void run()
{
while(true)
{
if(sum>0)
{
System.out.println(Thread.currentThread().getName()+":"+sum--);
}
}
} }
public class Demo01
{
public static void main(String args[])
{
Ticket t0=new Ticket();
Ticket t1=new Ticket();
Ticket t2=new Ticket();
Ticket t3=new Ticket();
t0.start();
t1.start();
t2.start();
t3.start(); }
}

运行结果:

2.使用Runnable方式

 class Ticket implements Runnable
{
public int sum=10;
public void run()
{
while(true)
{
if(sum>0)
{
System.out.println(Thread.currentThread().getName()+":"+sum--);
}
}
}
}
public class Demo
{
public static void main(String args[])
{
Ticket t=new Ticket();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
}
}

运行结果:

虽然两种方法都可以达到目的,但是后者并没有采用静态成员变量的方式,具有更大的灵活性。

【JAVA多线程概述】的更多相关文章

  1. Java多线程-Java多线程概述

    第一章 Java多线程概述 线程的启动 线程的暂停 线程的优先级 线程安全相关问题 1.1 进程与线程 进程:可以将运行在内存中的程序(如exe文件)理解为进程,进程是受操作系统管理的基本的运行单元. ...

  2. Java多线程概述

    /*多线程1.首先说进程,进程---就是正在进行的程序    每一个进程都有一个执行程序.该顺序是一个执行路径,或者叫一个控制单元 2.线程:就是进程中的一个独立的进程单元        线程在控制着 ...

  3. Java 多线程概述

    几乎所有的操作系统都支持同时运行多个任务,一 个任务通常就是一个程序,每个运行中的程序就是一个进程.当一个程序运行时,内部可能包含了多个顺序执行流,每个顺序执行流就是一个线程. 线程和进程 几乎所有的 ...

  4. 【程序员翻身计划】Java高性能编程第一章-Java多线程概述

    目标 重点: 线程安全的概念 线程通信的方式与应用 reactor线程模型 线程数量的优化 jdk常用命令 Netty框架的作用 难点 java运行的原理 同步关键字的原理 AQS的抽象 JUC的源码 ...

  5. Java多线程——<一>概述、定义任务

    一.概述 为什么使用线程?从c开始,任何一门高级语言的默认执行顺序是“按照编写的代码的顺序执行”,日常开发过程中写的业务逻辑,但凡不涉及并发的,都是让一个任务顺序执行以确保得到想要的结果.但是,当你的 ...

  6. Java:多线程概述与创建方式

    目录 Java:多线程概述与创建方式 进程和线程 并发与并行 多线程的优势 线程的创建和启动 继承Thread类 start()和run() 实现Runnable接口 实现Callable接口 创建方 ...

  7. Java多线程| 01 | 线程概述

    Java多线程| 01 | 线程概述 线程相关概念 进程与线程 进程:进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是操作系统进行资源分配与调度的基本单位.可以把进程简单的理解 ...

  8. java多线程(一)-概述

    最近这段在看java多线程编程方面的东西.所以特写了几篇文章,来总结和回顾一下自己所学习到的相关知识.因为水平有限,文章中总结不全面甚至理解错误的地方,欢迎读者指点批评. 我们平时所接触到的程序,都是 ...

  9. JAVA 多线程和并发学习笔记(四)

    1. 多进程 实现并发最直接的方式是在操作系统级别使用进程,进程是运行在它自己的地址空间内的自包容的程序.多任务操作系统可以通过周期性地将CPU从一个进程切换到另一个进程,来实现同时运行多个进程. 尽 ...

随机推荐

  1. Caffe学习系列(9):solver优化方法

    介绍了各种优化算法 参考:http://www.cnblogs.com/denny402/p/5074212.html

  2. HTTP协议概念篇

    1.概念 协议是指计算机通信网络中两台计算机之间进行通信所必须共同遵守的规定或规则,超文本传输协议(HTTP)是一种通信协议,它允许将超文本标记语言(HTML)文档从Web服务器传送到客户端的浏览器. ...

  3. phpcms导航中添加内部链接

    phpcms中栏目有3中类型 1.普通栏目 2.单网页 3.外部链接 其中如果想添加本站的内部链接,可以使用3,然后在添加链接的地方填入剩下的地址即可(需要以/开头) 如: /index.php?m= ...

  4. haproxy simple cfg

    global log /dev/log local0 log /dev/log local1 notice chroot /var/lib/haproxy user haproxy group hap ...

  5. rabbitmq使用

    1. 用户管理 用户管理包括增加用户,删除用户,查看用户列表,修改用户密码. 相应的命令 (1) 新增一个用户 rabbitmqctl  add_user  Username  Password (2 ...

  6. DECO 一个REACT NAtive 开发IDE工具

    DECO 一个REACT NAtive 开发IDE工具. 目前只支持 OS,NO WINDOWS https://www.decosoftware.com/ 一个方便的快速 ERXPRESS 教程:h ...

  7. linux mysql查看安装信息

    ps -ef|grep mysql root               ?        :: /bin/sh /usr/bin/mysqld_safe --datadir=/var/lib/mys ...

  8. tortoiseSVN svn+ssh

    2015年6月28日 11:45:10 星期日 今天实验用小海龟svn客户端的svn+ssh协议去链接版本库, 期望会快一点儿 首先在设置里 记着将连接ssh用的用户名和密码一块儿写到输入框中: -l ...

  9. Linux释放内存空间

    Linux服务器运行一段时间后,由于其内存管理机制,会将暂时不用的内存转为buff/cache,这样在程序使用到这一部分数据时,能够很快的取出,从而提高系统的运行效率,所以这也正是linux内存管理中 ...

  10. Nginx反向代理到Tomcat服务器

    在实际生产中,Tomcat服务器一般不单独使用在项目中,对于静态资源的响应Nginx表现的比较好,另外由于nginx是专门用于反向代理的服务器,所以很容易实现将java的请求转发到后端交给tomcat ...