Java多线程高并发学习笔记(一)——Thread&Runnable
进程与线程
首先来看百度百科关于进程的介绍:
进程是一个具有独立功能的程序关于某个数据集合的一次运行活动。它可以申请和拥有系统资源,是一个动态的概念,是一个活动的实体。它不只是程序的代码,还包括当前的活动,通过程序计数器的值和处理寄存器的内容来表示。
直观一点:

windows的任务管理里面,我们看到的eclipse和360等等,都是进程。(想深入了解的可以看看计算机操作系统)
什么是线程?
线程是在一个进程独立运行的子任务。线程是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点儿在运行中必不可少的资源,但它可与同属一个进程的其它线程共享进程所拥有的全部资源。
打个比如说,我们都用的QQ软件,它是一个进程,我们可以和人一边聊天一边传输文件,这就是多线程。
为什么要使用多线程?
1.多线程可以充分的利用系统的cpu,达到更快的处理效果
2.提高系统的运行效率
想入更深入了解的可以重新学习计算机操作系统,这里就不多介绍了,接下来我们怎么用!(毕竟这才是关键)
Thread and Runnable
Java中实现线程有两个方式,一个是继承Thread,另一个是实现Runnable接口,首先来看继承Thread的第一个实例:
package com.chapter1;
public class FirstThread {
public static void main(String[] args) {
InnerThread thread = new InnerThread();
// 线程启动
thread.start();
}
}
class InnerThread extends Thread {
// Override run方法,写入自己想要实现的业务
@Override
public void run() {
super.run();
System.out.println("Hello World!");
}
}
继承run方法的时候,我们通常会重写run方法,来实现自己的业务,接下来看,如何使用Runnable实现线程:
package com.chapter1;
public class FirstRunnable {
public static void main(String[] args) {
MyRunnable myRunnable;
myRunnable = new MyRunnable();
new Thread(myRunnable).start();
}
}
class MyRunnable implements Runnable {
@Override
public void run() {
System.out.println("Hello World!");
}
}
Runnable是一个接口,接口意味着更加灵活一些,也是推荐使用实现Runable接口来写线程的。
现在我们看到的都是一个单线程的例子,接下来写一个多线程的:
package com.chapter1;
/**
* 多个线程实例
* @author tangj
*
*/
public class ManyThread { public static void main(String args[]) {
Runnable runnable = new MyRunnable2();
new Thread(runnable, "a").start();
new Thread(runnable, "b").start();
new Thread(runnable, "c").start();
new Thread(runnable, "d").start();
}
} class MyRunnable2 implements Runnable { int count = 0;
@Override
public void run() {
autoIncrement();
System.out.println(Thread.currentThread().getName()+"计算了"+"count:" + count);
}
// 执行自增操作
private void autoIncrement(){
count++;
}
}
我们来看下运行结果:

再运行一次

可以看到,多线程的执行是无序的,而且这个结果有点奇怪,难道不是从1增加到4吗,怎么会出现重复的?
在JVM中,执行自增操作的分为三个步骤:
1.取得现有值count
2.执行count+1操作
3.将count+1赋值给count.
所以就会遇到这样的情况,一个线程在取得count值的时候,count操作正处于第二个步骤,上一个线程执行的还未进行赋值操作,这就涉及到线程安全问题。
下面给出上一个例子的解决方案
package com.chapter1; /**
* 多个线程实例
*
* @author tangj
*
*/
public class ManyThread { public static void main(String args[]) {
Runnable runnable = new MyRunnable2();
new Thread(runnable, "a").start();
new Thread(runnable, "b").start();
new Thread(runnable, "c").start();
new Thread(runnable, "d").start();
}
} class MyRunnable2 implements Runnable { int count = 0; // 注意在这里增加了关键字 synchronized
@Override
synchronized public void run() {
autoIncrement();
System.out.println(Thread.currentThread().getName() + "计算了" + "count:" + count);
} // 执行自增操作
private void autoIncrement() {
count++;
}
}
来看输出结果(多执行几次):


我们可以看到:不管哪个线程先执行,最后的打印顺序肯定是递增的顺序。
接下来我们看另外一中实现方式:使用重入锁(ReentrantLock)
package com.chapter1; import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class ManyThread2 { public static void main(String[] args) {
Runnable runnable = new MyRunnable3();
new Thread(runnable, "a").start();
new Thread(runnable, "b").start();
new Thread(runnable, "c").start();
new Thread(runnable, "d").start();
}
} class MyRunnable3 implements Runnable{ int count = 0;
// 使用重入锁ReentrantLock
Lock lock = new ReentrantLock(); @Override
public void run() {
// 锁住
lock.lock();
count++;
System.out.println(Thread.currentThread().getName() + "计算了" + "count:" + count);
// 解锁
lock.unlock();
} }

得到的结果也是整齐的递增顺序,说明它也是线程安全的。
总结
所以以后开发多线程的业务的时候,首先应该考虑的问题应该是这样的一个流程:
1.能否用单线程解决?(单线程本身就是线程安全的)
2.我开发的多线程业务是否线程安全?
线程安全是保证程序正常运行的关键,所以应该把线程安全作为开发多线程对于考虑的首要问题。
最后说两句:
本文所以代码都更新到我的github中,大家可以去clone或者Fork,我会持续更新的。
点击这里进入我的Github
喜欢的朋友可以点击下方的推荐,或者写个评论我们共同探讨Java高并发!!!
Java多线程高并发学习笔记(一)——Thread&Runnable的更多相关文章
- Java多线程高并发学习笔记——阻塞队列
在探讨可重入锁之后,接下来学习阻塞队列,这边篇文章也是断断续续的写了很久,因为最近开始学ssm框架,准备做一个自己的小网站,后续可能更新自己写网站的技术分享. 请尊重作者劳动成果,转载请标明原文链接: ...
- Java多线程高并发学习笔记(二)——深入理解ReentrantLock与Condition
锁的概念 从jdk发行1.5版本之后,在原来synchronize的基础上,增加了重入锁ReentrantLock. 本文就不介绍synchronize了,有兴趣的同学可以去了解一下,本文重点介绍Re ...
- JAVA多线程高并发学习笔记(三)——Callable、Future和FutureTask
为什么要是用Callable和Future Runnable的局限性 Executor采用Runnable作为基本的表达形式,虽然Runnable的run方法能够写入日志,写入文件,写入数据库等操作, ...
- Java多线程高并发学习笔记(三)——深入理解线程池
线程池最核心的一个类:ThreadPoolExecutor. 看一下该类的构造器: public ThreadPoolExecutor(int paramInt1, int paramInt2, lo ...
- Java 多线程高并发编程 笔记(一)
本篇文章主要是总结Java多线程/高并发编程的知识点,由浅入深,仅作自己的学习笔记,部分侵删. 一 . 基础知识点 1. 进程于线程的概念 2.线程创建的两种方式 注:public void run( ...
- JAVA 多线程和并发学习笔记(三)
Java并发编程中使用Executors类创建和管理线程的用法 1.类 Executors Executors类可以看做一个“工具类”.援引JDK1.6 API中的介绍: 此包中所定义的 Execut ...
- Java 多线程高并发编程 笔记(二)
1. 单例模式(在内存之中永远只有一个对象) 1.1 多线程安全单例模式——不使用同步锁 public class Singleton { private static Singleton sin=n ...
- JAVA 多线程和并发学习笔记(二)
一.Java中创建线程方法 1. 继承Thread类创建线程类 定义Thread类的子类,重写该类的run()方法.该方法为线程执行体. 创建Thread子类的实例.即线程对象. 调用线程对象的sta ...
- JAVA 多线程和并发学习笔记(四)
1. 多进程 实现并发最直接的方式是在操作系统级别使用进程,进程是运行在它自己的地址空间内的自包容的程序.多任务操作系统可以通过周期性地将CPU从一个进程切换到另一个进程,来实现同时运行多个进程. 尽 ...
随机推荐
- Linux下进程描述(1)—进程控制块
进程概念介绍 进程是操作系统对运行程序的一种抽象. • 一个正在执行的程序: • 一个正在计算机上执行的程序实例: • 能分配给处理器并由处理器执行的实体: • 一个具有普以下特征的活动单元:一组指令 ...
- Entity Framework入门教程:Oracle数据源访问
由于System.Data.OracleClient.dll从.NET Framework4.0之后已被弃用,所以我们无法在.NET Framework高版本中使用.一番搜索之后,发现好多文章提到.N ...
- 一步步学习操作系统(1)——参照ucos,在STM32上实现一个简单的多任务(“啰里啰嗦版”)
该篇为“啰里啰嗦版”,另有相应的“精简版”供参考 “不到长城非好汉:不做OS,枉为程序员” OS之于程序员,如同梵蒂冈之于天主教徒,那永远都是块神圣的领土.若今生不能亲历之,实乃憾事! 但是,圣域不是 ...
- spring boot 登录注册 demo (一)
Welcome to Spring Boot 代码结构 src/main/java 下 controller层,路由功能dao层,数据库的访问domain,bean的存放service,业务层appl ...
- [js高手之路]从原型链开始图解继承到组合继承的产生
基于javascript原型链的层层递进查找规则,以及原型对象(prototype)的共享特性,实现继承是非常简单的事情 一.把父类的实例对象赋给子类的原型对象(prototype),可以实现继承 f ...
- 【Maven】添加ueditor到maven本地仓库
问题出现:ueditor不存在远程和本地仓库,项目的pom.xml中无法添加依赖,导致无法使用mvn打包发布 解决办法:将ueditor-1.1.2.jar添加到本地仓库 办法1.使用命令行,这个没就 ...
- linux下mysql忘记密码的解决方案
1.首先确认服务器出于安全的状态,也就是没有人能够任意地连接MySQL数据库. 因为在重新设置MySQL的root密码的期间,MySQL数据库完全出于没有密码保护的 状态下,其他的用户也可以任意地 ...
- 细说 Azure Storage 的冗余策略
当我们想要把应用搬到云端的时候,首先要关注的便是数据的安全性.当然所有的云服务厂商都会对用户数据承诺一个非常高的安全性,但万一出现意外呢?我们是不是还要有适当的应对方案?比如今年的3月8日晚间,Azu ...
- jQuery DOM对象区别与联系
对两种对象类型的定义,只要能理解并转换成自己的说法就可以,不用死板按照资料所写 jQuery对象(jq对象)其实就是通过jquery类库选择器获得的对象(或者说是通过$获取的对象或者说是通过jquer ...
- 数据结构-单向链表 C和C++的实现
数据结构,一堆数据的存放方式. 今天我们学习数据结构中的 链表: 链表的结构: 链表是一种特殊的数组,它的每个元素称为节点,每个节点包括两个部分: 数据域:存放数据,此部分与数组相同 指针域:存放了下 ...