前言:
每个正在系统上运行的程序都是一个进程。每个进程包含一到多个线程。线程是一组指令的集合,或者是程序的特殊段,它可以在程序里独立执行。也可以把它理解为代码运行的上下文。所以线程基本上是轻量级的进程,它负责在单个程序里执行多任务。通常由操作系统负责多个线程的调度和执行。
 
使用线程可以把占据时间长的程序中的任务放到后台去处理,程序的运行速度可能加快,在一些等待的任务实现上如用户输入、文件读写和网络收发数据等,线程就比较有用了。在这种情况下可以释放一些珍贵的资源如内存占用等等。
如果有大量的线程,会影响性能,因为操作系统需要在它们之间切换,更多的线程需要更多的内存空间,线程的中止需要考虑其对程序运行的影响。通常块模型数据是在多个线程间共享的,需要防止线程死锁情况的发生。
 
正文:

两种实现多线程的方式,继承Thread类或者实现Runnable接口。

继承Thread类 :

package com.zsz.thread;
 
public class MyThread extends Thread {
     
     private int index;
     
     public MyThread( int index){
            this. index = index;
     }
     public void run () { 
           
            forint i=0;i<5;i++){
                System. out.println( index+ ":"+i);
           }
     } 
     public static void main(String[] args){
           MyThread MyThread = new MyThread(1);
           MyThread MyThread2 = new MyThread(2);
           MyThread.start();
           MyThread2.start();
     }
}
可能的运行结果:
1:0
2:0
1:1
2:1
1:2
2:2
1:3
2:3
1:4
2:4
由于执行CPU占用时间被切换,线程状态的切换,导致执行顺序的不同。
 
实现Runnable接口:
package com.zsz.thread;
 
public class MyRunnable implements Runnable{
 
     private int index;
     
     public MyRunnable( int index){
            this. index = index;
     }
     @Override
     public void run() {
            forint i=0;i<5;i++){
                System. out.println( index+ ":"+i);
           }
     }
     
     public static void main(String[] args){
           MyRunnable MyRunnable1 = new MyRunnable(1);
           Thread Thread1 = new Thread(MyRunnable1);
           MyRunnable MyRunnable2 = new MyRunnable(2);
           Thread Thread2 = new Thread(MyRunnable2);
           Thread1.start();
           Thread2.start();
     }
}
可能的结果:
1:0
1:1
1:2
2:0
2:1
2:2
2:3
2:4
1:3
1:4
 
场景应用:
一、车站多个窗口买票,车票总数是一定的,实现Runnable可以共享总票数。
package com.zsz.thread;
 
class SaleTicket  implements Runnable{
     private int ret = 5; //剩下票数
     private int num;            //一次买票张数
     
     public SaleTicket( int num){
            this. num = num;
     }
     @Override
     public void run() {
            synchronizedthis){
                 if( num> ret){
                     System. out.println( "余票不足");
                      return;
                }
                 ret = ret - num;
                System. out.println( "出票"+num+"张成功,剩余票数:" +ret );  //出票成功
           }
     }
}
public class MyRunnable extends Thread{
     public static void main(String[] args){
           SaleTicket saleTicket1 = new SaleTicket(1);
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
            new Thread(saleTicket1).start();
     }
}
可能的执行结果:
出票1张成功,剩余票数:4
出票1张成功,剩余票数:3
出票1张成功,剩余票数:2
出票1张成功,剩余票数:1
出票1张成功,剩余票数:0
余票不足
余票不足
注:考虑数据同步和线程安全,synchronized (this)确保同步,确保一个时刻只有一个线程占用synchronized 程序块,否则会出现线程不安全的情况,。
实现Runnable接口相比继承Thread类:
1):适合多个相同的程序代码的线程去处理同一个资源
2):可以避免java中的单继承的限制
3):增加程序的健壮性,代码可以被多个线程共享,代码和数据独立。
 
main方法其实也是一个线程。
线程其他情况:
线程休眠:Thread.sleep(2000);
线程中断:new Thread(new Runnable()).interrupt();
线程优先级:new Thread(new Runnable()).setPriority(8);
 

线程的状态及说明:
1、新建状态(New):新创建了一个线程对象。 
2、就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中,变得可运行,等待获取CPU的使用权。
3、运行状态(Running):就绪状态的线程获取了CPU,执行程序代码。 
4、阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种: 
  4.1、等待阻塞:运行的线程执行wait()方法,JVM会把该线程放入等待池中。 
  4.2、同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池中。 
  4.3、其他阻塞:运行的线程执行sleep()或join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。 
5、死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。
 

关键字volatile

 

用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值。volatile很容易被误用,用来进行原子性操作。

volatile不用做原子性操作的原因是:线程在运行时会在线程内存块中变量副本,之后,主内存中的变量与线程内的变量不联系,当运行结束时,线程内的变量会将值同步给主内存,因而会有可能出现线程不安全。
 

 有相关问题,可以提出来一起讨论。后续将会张贴一些进阶的多线程、线程池内容。
 
 博客园原文地址:http://www.cnblogs.com/zhongshengzhen
 

多线程基础及实例(java)的更多相关文章

  1. Java基础教程:多线程基础(1)——基础操作

    Java:多线程基础(1) 实现多线程的两种方式 1.继承Thread类 public class myThread extends Thread { /** * 继承Thread类,重写RUN方法. ...

  2. Java基础知识笔记(四:多线程基础及生命周期)

    一.多线程基础 编写线程程序主要是构造线程类.构造线程类的方式主要有两种,一种是通过构造类java.lang.Thread的子类,另一种是通过构造方法实现接口java.lang.Runnable的类. ...

  3. Java多线程干货系列—(一)Java多线程基础

    前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要,下面跟我一起开启本次的学习之旅吧. 正文 线程与进程 1 线程:进程中负责程序执行的 ...

  4. Java多线程总结(一)多线程基础

    多线程是Java学习的非常重要的方面,是每个Java程序员必须掌握的基本技能.本文只是多线程细节.本质的总结,并无代码例子入门,不适合初学者理解.初学者学习多线程,建议一边看书.看博文,以便写代码尝试 ...

  5. Java多线程干货系列(1):Java多线程基础

    原文出处: 嘟嘟MD 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要,下面跟我一起开启本次的学习之旅吧. 正文 线程与进程 1 线程 ...

  6. [转]Java多线程干货系列—(一)Java多线程基础

    Java多线程干货系列—(一)Java多线程基础 字数7618 阅读1875 评论21 喜欢86 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们 ...

  7. Java 多线程——基础知识

    java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...

  8. 【Java学习笔记之三十四】超详解Java多线程基础

    前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要,下面跟我一起开启本次的学习之旅吧. 正文 线程与进程 1 线程:进程中负责程序执行的 ...

  9. Java多线程基础知识总结

    2016-07-18 15:40:51 Java 多线程基础 1. 线程和进程 1.1 进程的概念 进程是表示资源分配的基本单位,又是调度运行的基本单位.例如,用户运行自己的程序,系统就创建一个进程, ...

随机推荐

  1. SQLite数据类型详解

    一.存储种类和数据类型: SQLite将数据值的存储划分为以下几种存储类型: 复制代码代码如下:      NULL: 表示该值为NULL值.      INTEGER: 无符号整型值.      R ...

  2. Android tab_Host页面跳转,传值,刷新等问题汇总

    之前做了一个项目是关于Tab_Host的,现在完成了恰逢闲余写份总结,主要涉及里面遇到问题以及解决方案的. (首先说明这份代码是在eoe 下载的,这里感谢分享的那位朋友,限于我的工程是公司的不能拿出来 ...

  3. MinHash算法-复杂度待整理

    1MinHash简介 传统的hash算法只负责将原始内容尽量均匀随机地映射为一个签名值,原理上相当于伪随机数产生算法.传统hash算法产生的两个签名,如果相等,说明原始内容在一定概率下是相等的:如果不 ...

  4. linux netcat命令

    netcat是网络工具中的“瑞士军刀”,它能通过TCP和UDP在网络中读写数据.通过与其他工具结合和重定向,你可以在脚本中以多种方式使用它.使用netcat命令所能完成的事情令人惊讶. netcat所 ...

  5. 微信支付开发1 微信支付URL配置

    本文介绍微信支付申请时如何设置授权目录及URL. 一.选择支付类型 目前有两种支付类型 JS API网页支付 Native原生支付 如果没有特殊要求,两种都勾选. 二.支付授权目录 目前可以选择htt ...

  6. Android开发之ADT导入Support Library

    在工程中增加(例如 support-v4 Library) 在ADT中需要按照以下步骤:  1.右击当前工程,查找Properties 2.选择Java Build Path 3.选择Librarie ...

  7. 几种java通信(rmi,http,hessian,webservice)协议性能比较

    一.综述 本文比较了RMI,Hessian,Burlap,Httpinvoker,web service等5种通讯协议的在不同的数据结构和不同数据量时的传输性能.RMI是java语言本身提供的通讯协议 ...

  8. freemarker截取字符串

    [#if   a.title?length   lt   23   ]   ${a.title} [#else]${a.title[0..22]}...[/#if]

  9. LinkedBlockingQueue

    LinkedBlockingQueue是一个基于已链接节点的.范围任意的blocking queue的实现.    此队列按 FIFO(先进先出)排序元素.队列的头部 是在队列中时间最长的元素.队列的 ...

  10. bzoj1532

    就题目而言,这道题是裸的二分+最大流 但是这样是TLE的,我们考虑优化 1. 我们可以先贪心,这样二分的上界就可以缩小了 2. 最大流我们可以不急着跑增广路,我们可以先贪心一个流然后再跑增广路 但是我 ...