Java多线程小结
简述
Java是支持多线程编程的语言,线程相比于进程更加轻量级,线程共享相同的内存空间,但是拥有独立的栈。减少了进程建立、销毁的资源消耗。jdk1.5后对java的多线程编程提供了更完善的支持,使得java的多线程编程更加方便简洁。本文旨在通过对Java多线程知识的梳理,整理出一个大纲,使得读者对多线程的编程有更加完善的认识。
线程生命周期的管理
- 线程的创建
java中线程的创建主要有两种方式:
- 继承Thread类(Thread中方法主要用于针对线程本身的处理);
- 实现Runnable接口;
- 线程的通讯
线程之间的通讯实际上是线程之间传递参数的问题。
- 往线程中传递参数: 通过新建线程的构造函数;
- 线程执行中,往外界传递参数:
- 通过方法的返回值,但可能线程未执行完成,所以返回NULL。所以,用轮询的方式,获取方法返回值。
缺点是浪费CPU周期。 - 通过注册回调方法,在构造线程的时候传入回调类,然后,在线程执行过程中调用回调方法。
- JDK1.5,提供了Future、Callable和Executor,Executor子类ExecutorService创建线程,实现Callable接口作为回调方法(实现call()方法),返回一个Future类。
- 通过方法的返回值,但可能线程未执行完成,所以返回NULL。所以,用轮询的方式,获取方法返回值。
线程同步
当多线程共享资源时,必须考虑同步问题。可以用synchronized关键字标注关键对象或方法。但是,同步不仅影响性能,同步的越多越容易造成死锁问题。(死锁:两个线程想要独占某种资源,但是,两者同时占用这种资源的子集的情况)同步的替代方法:
- 尽可能使用局部变量而不是字段,基本类型传参是值传递,是线程安全的(String也是安全的,一旦创建不能更改);
- 构造函数一般不需要考虑线程安全问题;
- 将非线程安全的类作为线程安全类的一个私有字段;
注:多线程中使用System.out输出,也属于共享资源。
- 线程的调度
JVM的线程调度器,抢占式的(preemptive)和协作式的(cooperative)。由于一个线程长时间占用CPU,会造成其他线程的饥饿状况。可以通过设置线程的优先级来改变这种情况,但是,在相同优先级的线程中,需要使用如下6种类方法,手动控制:
- 对I/O阻塞;
- 放弃,调用Thread.yield(),提供给相同优先级的线程CPU的使用机会;
- 休眠,sleep(),提供给相同优先级及以下优先级CPU的使用机会;
- 连接线程,join(),等待某个指定线程执行结束,或者执行一段时间;
- 等待某个对象,wait(),放弃对一个对象的锁定并暂停(之前的方法并不会放弃资源);
- 结束,方法返回return;
线程池
dk1.5对线程池提供了很好的支持。线程池的建立主要是为了减少内存资源的损耗,减少线程新建和删除的损耗,通过将任务(特定的线程)放置到队列中,然后,使用不同的策略,利用线程池中的线程处理队列中的任务。
ExecutorService调用一个基础API创建线程池,通过不同队列实现不同特性的线程池。队列是独立于线程池的一部分,用于放置线程池中未及时处理的任务。
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* 线程池包装
*/
public class BearThreadPool {
private ExecutorService pool;
public void createCachedThreadPool(){
pool = Executors.newCachedThreadPool();
}
public void addTask(Runnable task){
pool.submit(task);
}
public void termination(){
pool.shutdown();
}
}
/**
* 线程
*/
public class Task implements Runnable {
private String taskId;
private TaskCallBack callback;
public Task(String taskId, TaskCallBack callback){
this.taskId = taskId;
this.callback = callback;
}
public void run() {
synchronized (System.out){
System.out.println(taskId);
callback.before();
callback.middle();
callback.after();
System.out.println("==========");
}
}
}
/**
* 回调函数接口
*/
public interface TaskCallBack {
void before();
void middle();
void after();
}
Java多线程小结的更多相关文章
- 【Java基础】Java多线程小结
在说多线程之前,首先要清楚为啥要提出多线程,这就要明白线程和进程间的区别了. 线程和进程间的区别 进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单 ...
- Java多线程系列--“JUC锁”03之 公平锁(一)
概要 本章对“公平锁”的获取锁机制进行介绍(本文的公平锁指的是互斥锁的公平锁),内容包括:基本概念ReentrantLock数据结构参考代码获取公平锁(基于JDK1.7.0_40)一. tryAcqu ...
- java多线程(精华版)
在 Java 程序中使用多线程要比在 C 或 C++ 中容易得多,这是因为 Java 编程语言提供了语言级的支持.本文通过简单的编程示例来说明 Java 程序中的多线程是多么直观.读完本文以后,用户应 ...
- Java多线程---同步与锁
一,线程的同步是为了防止多个线程访问一个数据对象时,对数据造成的破坏. 二.同步和锁定 1.锁的原理 Java中每个对象都有一个内置锁. 当程序运行到非静态的synchronized同步方法上时,自动 ...
- Java多线程-线程的同步与锁
一.同步问题提出 线程的同步是为了防止多个线程访问一个数据对象时,对数据造成的破坏.例如:两个线程ThreadA.ThreadB都操作同一个对象Foo对象,并修改Foo对象上的数据. package ...
- Java多线程的实现
记得面试的时候,面试官问了Java多线程实现的方式有几种,它们之间的区别是什么?作为一个Java新手,将最近的学习总结如下: 1.Java多线程实现方式 Java多线程实现方式主要有三种:继承Thre ...
- 【Java多线程】两种基本实现框架
Java多线程学习1——两种基本实现框架 一.前言 当一个Java程序启动的时候,一个线程就立刻启动,改程序通常也被我们称作程序的主线程.其他所有的子线程都是由主线程产生的.主线程是程序开始就执行的, ...
- Java多线程详解
Java线程:概念与原理 一.操作系统中线程和进程的概念 现在的操作系统是多任务操作系统.多线程是实现多任务的一种方式. 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程 ...
- Java多线程(二) —— 线程安全、线程同步、线程间通信(含面试题集)
一.线程安全 多个线程在执行同一段代码的时候,每次的执行结果和单线程执行的结果都是一样的,不存在执行结果的二义性,就可以称作是线程安全的. 讲到线程安全问题,其实是指多线程环境下对共享资源的访问可能会 ...
随机推荐
- 自己编写的sublime text 3 插件
一些小功能,比较杂. 具体的功能在这里查看 1.本地环境的php运行结果获取. 2.快捷打开常用的文件,文件夹,url.(ctrl+shift+a) 3.常用的缩进转换. 下边是网络爬虫代码. #py ...
- python【第二十一篇】Django模板继承、分页、cookie验证
1.模板继承 母版master.html {% block title %}{% endblock %}2 {% block table-cont %}{% endblock %} 子板 {% ext ...
- 面试题:实现LRUCache::Least Recently Used的缩写,意思是最近最少使用,它是一种Cache替换算法
Design and implement a data structure for Least Recently Used (LRU) cache. It should support the fol ...
- Eclipse中的add jars和add external jars有什么区别(转载)
转载自:http://blog.csdn.net/yasi_xi/article/details/12772457 add jars和add external jars有什么区别? add ext ...
- java 常用基本数据类型的默认值
在使用基本数据类型作为类成员的时候,有时只初始化了而没有给变量赋值,那么此时,java会给你的变量赋一个默认初始值. boolean false char '/ ...
- PNG在IE6下背景问题
png24.min.js 源代码: var DD_belatedPNG={ns:"DD_belatedPNG",imgSize:{},delay:10,nodesFixed:0,c ...
- Android源码之Matrix
Matrix类在Android中主要用来进行矩阵变换,其可操作的对象包括图像.点阵.Vector(向量).矩形等. Matrix支持的变换类型主要有以下几种: 1.Translate:平移变换 2.R ...
- [LeetCode#277] Find the Celebrity
Problem: Suppose you are at a party with n people (labeled from 0 to n - 1) and among them, there ma ...
- UVA_393_Doors_(计算几何基础+最短路)
描述 https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=5&page ...
- ASP.NET学习路线图
转自:http://www.cnblogs.com/huangmeimujin/archive/2011/08/08/2131242.html 如果你已经有较多的面向对象开发经验,跳过以下这两步: 第 ...