说起线程 就不得不提进程 他们之间的关系很紧密

进程:内存中运行的应用程序 每个进程都有自己的一块内存空间 而线程是进程中的一个执行单元 一个进程中可以有多个线程 多线程的好处就是可以并发操作程序 将cpu资源利用率最大化 就像我们生活中一样 当我们在一个视屏网站下电影的时候 我们可以去做一些其他的时间 不需要一直等着电影下完再去做不会耗费多余的时间 这就是多线程的好处

如何实现多线程呢

1.继承Thread类 然后重写run方法

2.实现Runnable接口实现重写·

 
特点
继承thread类
只适用于单继承 编写简单 可以直接操作
实现runnable接口
可以实现多继承

线程的状态

1.新建状态

线程在被创建后就进入了新建状态 例如Thread thread=newThread();

2.就绪状态(也称为可执行转态)

线程再被创建后 如果调用了 start();说明该线程随时准备执行
3.运行状态
这时线程已经拿到了cpu执行权限 需要注意的是 线程可能没有被执行完 就被抢了 这时他又会回到就绪状态
4.阻塞状态
因为线程放弃了cpu的使用权 暂时停止执行操作 直到线程进入到就绪状态才有机会转到运行状态
阻塞情况分为三种:
       1.等待阻塞:通过调用线程的wait方法 等待线程的执行 (例如 你去银行取钱的时候 忘记带银行卡啦 所以你打电话给你老婆让他送卡过来 在这段时间里面你就一直在等待他))
       2.其他阻塞:通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态 (例如 去银行取钱的时候感觉有点困 睡着了 别人也进不来)
        3.同步阻塞:

线程在获取 synchronized同步锁失败(因为锁被其他线程锁占用)他会进入到同步阻塞状态 也称只为死锁() (例如 你去银行取钱一直在排队等 但是别人在银行atm里面就一直不出来 但是你不死心 还是一直在等)
5.死亡状态
线程执行完退出exit或者因异常退出
 
线程调度
 join() 插队 让当前线程执行完了之后在执行其他线程
join( long millis); 让出时间让你先执行 时间到了以后再继续抢
yield(); 暂停这次执行机会 再来竞争(相当于让你一次)
sleep() 让此线程休眠多长时间
notify()   唤醒在此对象监视器上等待的单个线程。
notifyAll()  唤醒在此对象监视器上等待的所有线程。
wait()    让当前线程处于“等待(阻塞)状态”,“直到其他线程调用此对象的 notify() 方法或 notifyAll() 方法”,当前线程被唤醒(进入“就绪状态”)。
 
java线程安全
当多个线程去操作同一对象的同一个属性时,执行的结果可能与预期不符。这时 这个对象就称为线程不安全的对象  这个对象所在的类成为线程不安全类
这时我们有两种方法可以解决
1.同步方法 
只需要在方法上添加一个synchronized修饰符即可 
       这种方法非常简单 但是效率很低 因为他把所有的操作都上锁了(例如去银行取钱应该是一个完整的操作 我们只需要将取卡 插卡 取钱 拿卡这几部上锁就够了 因为我们取钱的时候卡已经拿出来了放在手上 如果等到我了 我直接就可以去插卡了 很快 但是如果我取卡的步骤放到atm机里面去 找了很半天才找到卡是不是很浪费时间 如果每个人都是这样时间更长 所以说我在外面将卡找好了就可以节省时间)
2.同步代码块
需要将那些一起执行的代玛写在代码块中
synchronized(这里是一个锁对象 这个锁对象必须是一个不会改变的对象){
                   这里写的是那些需要一起执行的代码块
}
 
接下来举例说明
去银行取钱如果使用单线程会发生什么?
 package com.newroad.thread.test;
//账户类
public class Account {
private double balance;
private Object lock=new Object();
public Account(double balance) {
super();
this.balance = balance;
}
public void withdrawMoney(double money) {
String name=Thread.currentThread().getName();
System.out.println(name+"取款前账户余额为:"+balance);
if (balance>=money) {
balance=balance-money;
System.out.println(name+"取款金额为:"+money +"取款后,余额为:"+balance);
}else {
System.out.println(name+"余额不足,账户余额为:"+balance+"取款金额为;"+money);
} } } //线程类
package com.newroad.thread.test;
public class PersonThread extends Thread {
private Account account;
private double drawbalance;
public PersonThread(Account account, double drawbalance) {
super();
this.account = account;
this.drawbalance = drawbalance;
} @Override
public void run() {
account.withdrawMoney(drawbalance); } } //这是测试类
package com.newroad.thread.test; public class Test {
public static void main(String[] args) {
Account account=new Account(4000);
PersonThread p=new PersonThread(account, 2400);
PersonThread p1=new PersonThread(account, 1000);
PersonThread p2=new PersonThread(account, 600);
p.setName("张三");
p1.setName("张三媳妇");
p2.setName("张三爸");
p.start();
p1.start();
p2.start();
}
}

 这是运行后的结果 明显不对 我们可以明显发现 只有张三的操作是没有问题的 他媳妇和他爸都是有问题的 这肯定是有问题的 这时就可以使用加锁的方法 让一个线程执行完了在让其他线程在执行 而生活中 我们去银行ATM机上取钱都会有一个门 等我所有操作都完了其他人才能进来 这就相当于我们程序中的锁
这时我们只需要修改 账户类就可以啦
 package com.newroad.thread.test;

 public class Account {
private double balance;
private Object lock = new Object(); public Account(double balance) {
super();
this.balance = balance;
}
public void withdrawMoney(double money) {
String name = Thread.currentThread().getName();
synchronized (this) {
System.out.println(name + "取款前账户余额为:" + balance);
if (this.balance >= money) {
balance = balance - money;
System.out.println(name + "取款金额为:" + money + "取款后,余额为:" + balance);
} else {
System.out.println(name + "余额不足,账户余额为:" + balance + "取款金额为;" + money);
}
}
} }

这时我们就可以发现就没问题了

 

java多线程之述解的更多相关文章

  1. Java多线程编程详解

    转自:http://programming.iteye.com/blog/158568 线程的同步 由于同一进程的多个线程共享同一片存储空间,在带来方便的同时,也带来了访问冲突这个严重的问题.Ja ...

  2. Java多线程超级详解(只看这篇就够了)

    多线程能够提升程序性能,也属于高薪必能核心技术栈,本篇会全面详解Java多线程.@mikechen 主要包含如下几点: 基本概念 很多人都对其中的一些概念不够明确,如同步.并发等等,让我们先建立一个数 ...

  3. JAVA多线程synchronized详解

    Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 当两个并发线程访问同一个对象object中的这个synchronized(this)同 ...

  4. Java多线程基础详解

    基础概念进程进程是操作系统结构的基础:是一次程序的执行:是一个程序及其数据在处理机上顺序执行时所发生的活动.操作系统中,几乎所有运行中的任务对应一条进程(Process).一个程序进入内存运行,即变成 ...

  5. 【多线程】java多线程Completablefuture 详解【在spring cloud微服务之间调用,防止接口超时的应用】【未完成】

    参考地址:https://www.jianshu.com/p/6f3ee90ab7d3 示例: public static void main(String[] args) throws Interr ...

  6. Java多线程详解(二)

    评论区留下邮箱可获得<Java多线程设计模式详解> 转载请指明来源 1)后台线程 后台线程是为其他线程服务的一种线程,像JVM的垃圾回收线程就是一种后台线程.后台线程总是等到非后台线程死亡 ...

  7. Java多线程——多线程方法详解

    本系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多线程的深入剖 ...

  8. Java——多线程之方法详解

    Java多线程系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多 ...

  9. Java多线程——对象及变量的并发访问

    Java多线系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多线 ...

随机推荐

  1. Hadoop HA on Yarn——集群配置

    集群搭建 因为服务器数量有限,这里服务器开启的进程有点多: 机器名 安装软件 运行进程 hadoop001 Hadoop,Zookeeper NameNode, DFSZKFailoverContro ...

  2. js数组去重的方法(转)

    JS数组去重的几种常见方法 一.简单的去重方法 // 最简单数组去重法 /* * 新建一新数组,遍历传入数组,值不在新数组就push进该新数组中 * IE8以下不支持数组的indexOf方法 * */ ...

  3. 按要求分解字符串,输入两个数M,N;M代表输入的M串字符串,N代表输出的每串字符串的位数,不够补0。例如:输入2,8, “abc” ,“123456789”,则输出为“abc00000”,“12345678“,”90000000”

    import java.util.ArrayList; import java.util.Scanner; public class Text { @SuppressWarnings("re ...

  4. 如何用代码而非事件触发PBO

    通常我们通过抛出事件触发PBO,但若没有事件发生时,我们其实也可以用代码强制发出命令. 写法如下: CL_GUI_CFW=>SET_NEW_OK_CODE( NEW_CODE = <uco ...

  5. 随手练——Uva-11584 划分成回文串(区间DP)

    思路:dp[i]代表到第i位的最小值,枚举它的前几位,求出最小值. 转移方程:dp[ i ] = min(dp[ i ], dp[ j - 1 ] + 1 ) ; 本来觉得,代码加深部分可以提前bre ...

  6. 学习java前端 两种form表单提交方式

    第一种:原生方式 注意点:button标签的style为submit <form action="/trans/doTrans.do" method="post&q ...

  7. 时间序列分析工具箱—— h2o + timetk

    目录 时间序列分析工具箱-- h2o + timetk h2o 的用途 加载包 安装 h2o 加载包 数据 教程:h2o + timetk,时间序列机器学习 时间序列机器学习 最终的胜利者是... 翻 ...

  8. 20145234黄斐《网络对抗技术》实验八、Web基础

    Apache 先通过apachectl start命令开启Apach,使用netstat -aptn命令查看端口占用: 因为端口号80已经被占用(上次实验设置的),所以先修改/etc/apache2/ ...

  9. HBase数据模型的一些概念

    首先来先理解一个概念:HBase是一种列式存储的分布式数据库. 表              在HBase中数据以表的形式存储.使用表的主要原因是把某些列组织起来一起访问,同一个表中的数据通常是相关的 ...

  10. 洛谷NOIp热身赛题解

    洛谷NOIp热身赛题解 A 最大差值 简单树状数组,维护区间和.区间平方和,方差按照给的公式算就行了 #include<bits/stdc++.h> #define il inline # ...