java多线程编程--基础篇
一、基本概念
a、操作系统中进程与线程的概念
现在的操作系统是多任务操作系统。多线程是实现多任务的一种方式。 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,
一个进程中可以启动多个线程。比如在Windows系统中,一个运行的exe就是一个进程。
线程是指进程中的一个执行流程,一个进程中可以运行多个线程。比如java.exe进程中可以运行很多线程。线程总是属于某个进程,进程中的多个线程共享进程的内存空间。
“同时”执行是人的感觉,在线程之间实际上轮换执行。
b、Java中的线程 在Java中,“线程”指两件不同的事情:
a、java.lang.Thread类的一个实例。
b、线程的执行。
使用java.lang.Thread类或者java.lang.Runnable接口编写代码来定义、实例化和启动新线程。
一个Thread类实例只是一个对象,像Java中的任何其他对象一样,具有变量和方法,在堆上完成对象的创建和销毁。
Java中,每个线程都有一个调用栈,即使不在程序中创建任何新的线程,线程也在后台运行着。
一个Java应用总是从main()方法开始运行,mian()方法运行在一个线程内,它被称为主线程。
一旦创建一个新的线程,就产生一个新的调用栈。线程总体分两类:用户线程和守护线程。
当所有用户线程执行完毕的时候,JVM自动关闭。但是守护线程却独立于JVM,守护线程一般是由操作系统或者用户自己创建的。
二、线程的创建与启动
a、定义线程
1、继承java.lang.Thread类。
此类中有个run()方法,应该注意其用法: public void run() 如果该线程是使用独立的 Runnable 运行对象构造的,则调用该 Runnable 对象的 run 方法;否则,该方法不执行任何操作并返回。 Thread 的子类应该重写该方法。
2、实现java.lang.Runnable接口。
void run() 使用实现接口 Runnable 的对象创建一个线程时,启动该线程将导致在独立执行的线程中调用对象的 run 方法。
方法 run 的常规协定是,它可能执行任何所需的操作。
b、实例化线程
1、如果是继承java.lang.Thread类的线程,则直接new即可。
即使用的时候Thread a=new tThread();
2、如果是实现了java.lang.Runnable接口的类,则用Thread的构造方法:
Thread(Runnable target)
Thread(Runnable target, String name)
Thread(ThreadGroup group, Runnable target)
Thread(ThreadGroup group, Runnable target, String name)
Thread(ThreadGroup group, Runnable target, String name, long stackSize)
rThread a=new rThread();//此处可以使用各个构造函数
Thread b=new Thread(a);
c、启动线程
在线程的Thread对象上调用start()方法,而不是run()或者别的方法。 在调用start()方法之前:线程处于新状态中,新状态指有一个Thread对象,但还没有一个真正的线程。 在调用start()方法之后,发生了一系列复杂的事情:启动新的执行线程(具有新的调用栈); 该线程从新状态转移到可运行状态; 当该线程获得机会执行时,其目标run()方法将运行。
start()方法后,run()方法并没有执行。
注意:对Java来说,run()方法没有任何特别之处。像main()方法一样,它只是新线程知道调用的方法名称(和签名)。因此,在Runnable上或者Thread上调用run方法是合法的,但并不启动新的线程,所以通常不使用
d、基本例子
1、实现Runnable接口的多线程例子
package com.yunhe.thread;
/**
* 实现Runable接口的类
* */
public class DoSomething implements Runnable {
private String name; public DoSomething(String name) {
this.name = name;
} public void run() {
for (int i = 0; i < 5; i++) {
for (long k = 0; k < 100000000; k++);
System.out.println(name + ": " + i);
}
}
}
测试类代码:
package com.yunhe.thread;
/**
* 实现Runnable的线程的测试类
*/ public class TestRunnable { public static void main(String[] args) { DoSomething ds1=new DoSomething("tianti");
DoSomething ds2=new DoSomething("yunhe"); Thread t1=new Thread(ds1);
Thread t2=new Thread(ds2);
t1.start();
t2.start();
} }
运行结果:
tianti:
yunhe:
yunhe:
tianti:
yunhe:
tianti:
yunhe:
tianti:
yunhe:
tianti:
2、继承Thread实现线程的例子
package com.yunhe.thread;
/**
* 继承Thread的线程类
* */
public class DoOtherthing extends Thread {
public DoOtherthing(String name) {
super(name);
} @Override
public void run() {
// TODO Auto-generated method stub
super.run();
for (int i = 0; i < 5; i++) {
for (long k = 0; k < 1000000000; k++);
System.out.println(this.getName() + ": " + i);
}
} }
测试类代码:
package com.yunhe.thread;
/**
* 继承Thread类
* */
public class DoOtherthing extends Thread {
public DoOtherthing(String name) {
super(name);
} @Override
public void run() {
// TODO Auto-generated method stub
super.run();
for (int i = 0; i < 5; i++) {
for (long k = 0; k < 1000000000; k++);
System.out.println(this.getName() + ": " + i);
}
} }
运行结果:
tianti:
yunhe:
tianti:
yunhe:
tianti:
yunhe:
tianti:
yunhe:
tianti:
yunhe:
对于上面的多线程程序代码来说,输出的结果是不确定的。
其中for (long k = 0; k < 100000000; k++);是模拟非常耗时的操作,k值上限较小的时候输出是顺序的,较大的时候是不稳定的交叉输出。
e、一些常见的问题
1、线程的名字,一个运行中的线程总是有名字的,名字有两个来源,一个是JVM虚拟机自己给的名字,一个是你自己的定的名字。在没有指定线程名字的情况下,JVM虚拟机总会为线程指定名字,并且主线程的名字总是main,非主线程的名字不确定。
2、线程都可以设置名字,也可以获取线程的名字,连主线程也不例外。
3、获取当前线程的对象的方法是:Thread.currentThread();
4、在上面的代码中,只能保证:每个线程都将启动,每个线程都将运行直到完成。一系列线程以某种顺序启动并不意味着将按该顺序执行。对于任何一组启动的线程来说,调度程序不能保证其执行次序,持续时间也无法保证。
5、当线程目标run()方法结束时该线程完成。
6、一旦线程启动,它就永远不能再重新启动。只有当线程是新的线程的时候才可以被启动,并且只能一次。一个可运行的线程(就绪状态)或死线程可以被重新启动。
7、线程的调度是JVM的一部分,在一个CPU的机器上上,实际上一次只能运行一个线程。一次只有一个线程栈执行。JVM线程调度程序决定实际运行哪个处于可运行状态的线程(就绪状态)。
众多可运行线程中的某一个会被选中做为当前线程。可运行线程被选择运行的顺序是没有保障的。
8、尽管通常采用队列形式,但这是没有保障的。队列形式是指当一个线程完成“一轮”时,它移到可运行队列的尾部等待,直到它最终排队到该队列的前端为止,它才能被再次选中。事实上,我们把它称为可运行池而不是一个可运行队列,目的是帮助认识线程运行并不都是以某种有保障的顺序排列来调度线程。
9、尽管我们不能控制线程调度的顺序,但可以通过别的方式来影响线程调度的方式。
java多线程编程--基础篇的更多相关文章
- Java多线程系列--“基础篇”11之 生产消费者问题
概要 本章,会对“生产/消费者问题”进行讨论.涉及到的内容包括:1. 生产/消费者模型2. 生产/消费者实现 转载请注明出处:http://www.cnblogs.com/skywang12345/p ...
- Java多线程系列--“基础篇”04之 synchronized关键字
概要 本章,会对synchronized关键字进行介绍.涉及到的内容包括:1. synchronized原理2. synchronized基本规则3. synchronized方法 和 synchro ...
- Java多线程系列--“基础篇”02之 常用的实现多线程的两种方式
概要 本章,我们学习“常用的实现多线程的2种方式”:Thread 和 Runnable.之所以说是常用的,是因为通过还可以通过java.util.concurrent包中的线程池来实现多线程.关于线程 ...
- Java多线程系列--“基础篇”03之 Thread中start()和run()的区别
概要 Thread类包含start()和run()方法,它们的区别是什么?本章将对此作出解答.本章内容包括:start() 和 run()的区别说明start() 和 run()的区别示例start( ...
- Java多线程系列--“基础篇”05之 线程等待与唤醒
概要 本章,会对线程等待/唤醒方法进行介绍.涉及到的内容包括:1. wait(), notify(), notifyAll()等方法介绍2. wait()和notify()3. wait(long t ...
- Java多线程系列--“基础篇”06之 线程让步
概要 本章,会对Thread中的线程让步方法yield()进行介绍.涉及到的内容包括:1. yield()介绍2. yield()示例3. yield() 与 wait()的比较 转载请注明出处:ht ...
- Java多线程系列--“基础篇”07之 线程休眠
概要 本章,会对Thread中sleep()方法进行介绍.涉及到的内容包括:1. sleep()介绍2. sleep()示例3. sleep() 与 wait()的比较 转载请注明出处:http:// ...
- Java多线程系列--“基础篇”08之 join()
概要 本章,会对Thread中join()方法进行介绍.涉及到的内容包括:1. join()介绍2. join()源码分析(基于JDK1.7.0_40)3. join()示例 转载请注明出处:http ...
- Java多线程系列--“基础篇”09之 interrupt()和线程终止方式
概要 本章,会对线程的interrupt()中断和终止方式进行介绍.涉及到的内容包括:1. interrupt()说明2. 终止线程的方式2.1 终止处于“阻塞状态”的线程2.2 终止处于“运行状态” ...
随机推荐
- javascrit2.0完全参考手册(第二版) 第2章第3节 变量
变量存储数据.每个变量都有一个名字,叫做标识符.在js中声明变量使用var关键字,var为新的数据分配存储空间,或者指示一直标识符正在使用.声明变量非常简单: var x; 这个语句告诉解释器一个新的 ...
- Android事件处理
含义:为用户动作提供响应就是事件处理. Android提供了强大的事件处理机制:基于监听的事件处理.基于回调的事件处理. 一.基于监听的事件处理 监听的处理模型主要涉及三类对象 >Event S ...
- vim operation
note: 转自 www.quora.com ,很好的网站. 具体链接如下: https://www.quora.com/What-are-some-impressive-demos-of-Vim- ...
- 第三章 Odoo基本设置
登录 正常访问http://localhost:8069后,登录的界面如下: 这是Odoo默认的认证方式,也是我们最常见最熟悉的认证方式,7.0以前,数据库中的密码都是以明文方式存储,可以很轻松地在r ...
- HTML5 Input 类型
浏览器支持 Input type IE Firefox Opera Chrome Safari email No 4.0 9.0 10.0 No url No 4.0 9.0 10.0 No numb ...
- bootstrap-combined.min.css
/*! * Bootstrap v2.2.2 * * Copyright 2012 Twitter, Inc * Licensed under the Apache License v2.0 * ht ...
- linux一些名词
硬链接: 同一个索引节点的两个文件 ln a b //创建a的连接文件b,删除a,不失效软连接: 不同索引节点,相当于一个快捷方式ln -s a b,删除a,b失效 网络协议: 网络协议就是通信规则的 ...
- cordova-sqlite-plugin常用数据库操作
ionic项目需要做本地缓存,研究了一番,总结出了常用的数据库操作: 1.新建数据库,用angularJS+ionic: //声明一个数据库对象 var db = null; //事件devicere ...
- Range
欢迎转载,转载请注明出处,徽沪一郎. 概要 Scala中Range可以看成是List的特例,Range的包含的元素类型是Int, 本文介绍如何创建Range Range创建 方法一: val r1 = ...
- Amoeba For MySQL读写分离
Amoeba安装,参考http://hi.baidu.com/itfenqing/item/79fe989838e1e8ad83d2959c Amoeba版本为:amoeba-mysql-1.3.1- ...