进程和线程

进程:是一个正在执行中的程序。每一个进程执行都有一个执行顺序,该执行顺序是一个执行路径,或者叫一个控制单元。

线程:就是进程中的一个独立的控制单元。

线程在控制着进程的执行。

在计算机中多个线程都获取cpu的执行权,cpu执行到谁,谁就运行,明确一点,在某一个时刻,只能由一个程序运行(多核除外),cpu做着快速切换,以达到看上去是同事运行的效果。我们可以形象的把多线程的运行行为在互相抢夺cpu的执行权,这就是多线程的一个特性:随机性。谁抢到谁执行,至于执行多长时间,cpu说了算。

多线程的优势

l  多线程可以使得软件运行的速度更快,譬如迅雷可以多个线程同时下载一个文件,加快了下载的速度

l  进程之间不能共享内存,而线程之间则可以

如何创建启动线程:

方法一:

1、定义类继承Thread

2、复写Thread类中的run方法,目的是将自定义的代码存储在run方法中让线程运行

3、调用线程的start方法(          该方法有两个作用:1.启动线程;2.调用run方法)

public class ThreadDemo extends Thread {
public static void main(String[] args) {
Thread thread1 = new Thread();
Thread thread2 = new Thread(); thread1.start();
thread2.start();
}
} class ThreadTest extends Thread { @Override
public void run() {
/**
* 这里是要实现多线程的代码
*/
}
}

方法二:

1、定义类实现Runnable接口

2、覆盖Runnable接口中的run方法,将线程要运行的代码存放在该run方法中

3、通过Thread类建立线程对象

4、将Running接口的子类对象作为实际参数传递给Thread类的构造函数。

5、调用Thread类的 start方法开启线程并调用Runnable接口子类的run方法

public class ThreadDemo extends Thread {
public static void main(String[] args) { ThreadTest tt = new ThreadTest(); Thread thread1 = new Thread(tt);
Thread thread2 = new Thread(tt); thread1.start();
thread2.start(); }
} class ThreadTest implements Runnable { @Override
public void run() {
/**
* 这里是要实现多线程的代码
*/
}
}

为什么要将Runnable接口的子类对象传递给Thread的构造函数?

因为自定义的run方法所属的对象是Runnable接口的子类对象,所以要让线程去指定指定对象的run方法

为什么要覆盖run方法?

Thread类用于描述线程,该类就定义了一个功能,用于存储线程要运行的代码。该存储功能就是run方法,也就是说Thread类中的run方法,用于存储线程要运行的代码。

这两种实现方式的区别?

1、实现Runnable接口避免了单继承的局限性,在定义线程时建议使用实现Runnable方式,在实现了多线程之后还可以继承其他类

2、继承Thread类的线程代码存放在Thread子类的run方法中,实现Runnable线程代码存放在接口的子类的run方法中。

线程的状态

         新建和就绪

当使用new关键字新建了一个线程之后,该线程就属于新建状态,这个时候的它和其他Java对象一样,仅仅由JAVA虚拟机为它分配内存,并初始化它的值

调用start()方法之后,该线程属于就绪状态,表示该线程可以运行了,但是何时运行得看JVM的内部调度。

         运行和阻塞

当线程取得执行权限,run方法内的线程体开始执行时,该线程出于运行状态

sleep(time)方法可以让线程冻结time长度的时间,time之后,继续执行线程

wait()方法也可以冻结线程,冻结之后,需要notify()方法唤醒线程,否则会一直冻结

线程死亡

run方法内部代码执行完毕或者遇到异常或error时候,线程结束,线程进入死亡状态

stop()方法可以直接杀掉线程,但容易导致死锁,通常不推荐

         临时状态(阻塞状态)

当同时运行多个线程时候,而cpu只能运行一个线程,那么其他线程就会先进入临时状态(阻塞状态)

典型例子

火车站卖票就是一个典型的多线程的例子,票的总量是固定的,但是会有多个窗口在卖票,假设一共有100张票,一共有6个窗口在售票,当有人来买票的时候,窗口售票员先看是否有余票,如果有,则出售。

//使用实现Runnable接口方式实现
public class SellTicket {
public static void main(String[] args) {
TicketThread tt = new TicketThread(); Thread t1 = new Thread(tt);
Thread t2 = new Thread(tt); t1.start();
t2.start();
}
} class TicketThread implements Runnable { public int ticketCount = 100; @Override
public void run() {
while (true) {
sellTicket();
if (ticketCount == 0) {
break;
}
}
} public void sellTicket() {
if (ticketCount > 0) {
System.out.println(Thread.currentThread().getName() + " 卖了第 "
+ ticketCount-- + " 张票"); try {
Thread.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
} //使用继承Thread类实现
public class SellTicket {
public static void main(String[] args) { TicketThread tt1 = new TicketThread();
TicketThread tt2 = new TicketThread(); tt1.start();
tt2.start();
}
} class TicketThread extends Thread { public static int ticketCount = 100; @Override
public void run() {
while (true) {
sellTicket();
if (ticketCount == 0) {
break;
}
}
} public void sellTicket() {
if (ticketCount > 0) {
System.out.println(Thread.currentThread().getName() + " 卖了第 "
+ ticketCount-- + " 张票"); try {
Thread.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}

多执行几次的话,就会发现,会发生一票多卖的情况,是因为线程获取到CPU的执行权限之后做运算,还没有将票数减少,另外一个线程抢夺到了CPU的执行权限,所以他们卖出的是同一张票,这个时候就出现了问题,为了解决这个问题,JAVA的多线程支持引入了同步监视器来解决这个问题

线程同步

JAVA使用synchronized关键字在线程开始执行同步代码快之前,获取对同步监视器的锁定

synchronized代码块

public void run() {
method();
} public void method () {
synchronized (this){
同步代码段
}
}

  注意,普通方法中的同步对象是this,静态方法中的同步对象是 类名.class

public class SellTicket {
public static void main(String[] args) {
TicketThread tt = new TicketThread(); Thread t1 = new Thread(tt);
Thread t2 = new Thread(tt); t1.start();
t2.start();
}
} class TicketThread implements Runnable { public int ticketCount = 100; @Override
public void run() {
while (true) {
synchronized (this) {
sellTicket();
if (ticketCount == 0) {
break;
}
}
}
} public void sellTicket() {
if (ticketCount > 0) {
System.out.println(Thread.currentThread().getName() + " 卖了第 "
+ ticketCount-- + " 张票"); try {
Thread.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}

synchronized方法

public void run() {
method();
} public synchronized void method () {
同步代码段
}
package cn.lixyz.thread;

public class SellTicket {
public static void main(String[] args) {
TicketThread tt = new TicketThread(); Thread t1 = new Thread(tt);
Thread t2 = new Thread(tt); t1.start();
t2.start();
}
} class TicketThread implements Runnable { public int ticketCount = 100; @Override
public void run() {
while (true) {
sellTicket();
if (ticketCount == 0) {
break;
}
}
} public synchronized void sellTicket() {
if (ticketCount > 0) {
System.out.println(Thread.currentThread().getName() + " 卖了第 "
+ ticketCount-- + " 张票");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}

使用synchronized修饰的代码就好比拥有了一把锁,当线程对象开始执行同步内容代码时,其他线程即时获取了CPU的执行权限也无法执行,直到当前执行对象执行完毕释放权限。

  

JAVA 多线程(一)的更多相关文章

  1. 40个Java多线程问题总结

    前言 Java多线程分类中写了21篇多线程的文章,21篇文章的内容很多,个人认为,学习,内容越多.越杂的知识,越需要进行深刻的总结,这样才能记忆深刻,将知识变成自己的.这篇文章主要是对多线程的问题进行 ...

  2. Java多线程基础知识篇

    这篇是Java多线程基本用法的一个总结. 本篇文章会从一下几个方面来说明Java多线程的基本用法: 如何使用多线程 如何得到多线程的一些信息 如何停止线程 如何暂停线程 线程的一些其他用法 所有的代码 ...

  3. Java多线程系列--“JUC锁”03之 公平锁(一)

    概要 本章对“公平锁”的获取锁机制进行介绍(本文的公平锁指的是互斥锁的公平锁),内容包括:基本概念ReentrantLock数据结构参考代码获取公平锁(基于JDK1.7.0_40)一. tryAcqu ...

  4. Java多线程系列--“JUC锁”04之 公平锁(二)

    概要 前面一章,我们学习了“公平锁”获取锁的详细流程:这里,我们再来看看“公平锁”释放锁的过程.内容包括:参考代码释放公平锁(基于JDK1.7.0_40) “公平锁”的获取过程请参考“Java多线程系 ...

  5. Java多线程--让主线程等待子线程执行完毕

    使用Java多线程编程时经常遇到主线程需要等待子线程执行完成以后才能继续执行,那么接下来介绍一种简单的方式使主线程等待. java.util.concurrent.CountDownLatch 使用c ...

  6. Java多线程 2 线程的生命周期和状态控制

    一.线程的生命周期 线程状态转换图: 1.新建状态 用new关键字和Thread类或其子类建立一个线程对象后,该线程对象就处于新生状态.处于新生状态的线程有自己的内存空间,通过调用start方法进入就 ...

  7. java 多线程 1 线程 进程

    Java多线程(一).多线程的基本概念和使用 2012-09-10 16:06 5108人阅读 评论(0) 收藏 举报  分类: javaSE综合知识点(14)  版权声明:本文为博主原创文章,未经博 ...

  8. 一起阅读《Java多线程编程核心技术》

    目录 第一章 Java多线程技能 (待续...)

  9. 第一章 Java多线程技能

    1.初步了解"进程"."线程"."多线程" 说到多线程,大多都会联系到"进程"和"线程".那么这两者 ...

  10. java从基础知识(十)java多线程(下)

    首先介绍可见性.原子性.有序性.重排序这几个概念 原子性:即一个操作或多个操作要么全部执行并且执行的过程不会被任何因素打断,要么都不执行. 可见性:一个线程对共享变量值的修改,能够及时地被其它线程看到 ...

随机推荐

  1. JS获当前网页元素高度offsetHeight

    本文测试的是offsetHeight,获取网页中某元素的高度,单位是像素,获取的类型是整型,可以进行数字运算.如图,网页中的元素本身的高度包括,自身的内容+padding+border,而margin ...

  2. 看烦了VS2012的黑白调调了吗?换

    VS2012的默认深色主题的确让整个IDE看起来很有气场,而且深色的主题保护眼睛,还是蛮不错的.但是看久了也会烦啊.虽然说重要的不是IDE看起来怎么样,而是写出来的代码质量怎么样,但一个好的环境也是会 ...

  3. 玩转CONSUL(2)–分布式锁

    1. 前言 分布式锁的场景,大家应该都有遇到过.比如对可靠性有较高要求的系统中,我们需要做主备切换.这时我们可以利用分布式锁,来做选主动作,抢到锁作为主,执行对应的任务,剩余的实例作为备份 redis ...

  4. keepalived+lvs tcp check 引起的后端服务报Connection reset by peer

    方法一: 取消LVS方式进行tcp转发,进而改为http方式反向代理,问题即可解决. 当然,这是在业务允许使用http的情况下,如果必须使用tcp协议,那就得使用下面的方法了. 方法二: 修改keep ...

  5. DB2 索引(2)

    最近研究了一点DB2索引相关的东西,做一个总结: (1)在作为主键的列上,强制该列的唯一性和组织表中数据的排列结构: (2)在经常用连接的列(join)上建索引,这些列主要是一些外键,可以加快连接的速 ...

  6. eNSP下配置Trunk接口实现跨交换机传递数据

    当Trunk端口发送数据帧的时候,当所发送帧的VLAN ID与端口的PVID不同是,检查是否允许该VLAN通过,若允许的话,直接透传,不允许就直接丢弃:当该帧的VLAN ID与端口的PVID相同时,则 ...

  7. CSS float属性

    表示向左浮动,比如多个div在一个页面上,默认情况是:一行一个div,但是只要在div的css中使用float:left,可以使一行有多个div,这样可以把网页划分成很多块,但是使用该属性会影响后面的 ...

  8. 1181: 零起点学算法88——偶数求和(C语言)

    一.题目: 题目来源WUSTOJ 二.源代码: #include<stdio.h> int main() { int n, m, num, sum, i, j, k; while (sca ...

  9. C++11<functional>深度剖析:背景、原理、接口与实现

    自C++11以来,C++标准每3年修订一次.C++14/17都可以说是更完整的C++11:即将到来的C++20也已经特性完整了. C++11已经有好几年了,它的年龄比我接触C++的时间要长10倍不止吧 ...

  10. MACD波段选股

    MA12:=MA(C,); {股价连续3天站稳12日均线,且12日均线走平或向上} C1:=EVERY(C>MA12,) AND MA12>=REF(MA12,); {MACD金叉,且DI ...