JAVA自学笔记23

1、多线程
1)引入:

2)进程
是正在运行的程序。是系统进行资源分配和调用的独立单位。每一个进程都有它自己的内存空间和系统资源。
多进程:
单进程的计算机只能做一件事情,而现在的计算机都可以做多件事情。CPU在某个时间点上只能做一件事。每一个进程都有它自己的内存空间和系统资源。
3)多线程
-是进程中的单个顺序控制流,是一条执行路径
-一个进程如果只有一条执行路径,则称为单线程程序。一个进程如果有多条执行路径,称为多线程程序。
4)并行与并发
前者是逻辑上同时发生,指在某一个时间段内同时运行多个程序;后者是物理上同时发生,指在,某一个时间点内同时运行多个程序
5)Java程序运行原理
java命令会启动java虚拟机,启动jvm,等于启动了一个应用程序,也就是启动了一个进程,该进程会自动地启动一个“主线程”,然后主线程去调用某个类的main方法,所以main方法运行在主线程中。在此之前的所有程序都是单线程的。

JVM的启动时单线程还是多线程的呢?
多线程的。垃圾回收线程也要先启动,否则很容易出现内容溢出。最少都启动了两个线程。


创建新执行线程有两种方法,一种方法时将类声明为Thread的子类。该子类应重写Thread类的run方法。接下来可以分配并启动该子类的实例。
即-自定义一个继承自Thread的类
-重写run()方法(不是类中的所有代码都需要被线程执行。此时,为了区分哪些代码能够被执行。java提供了Thread类中的run()方法用于包含那些被执行的代码。)
-创建对象
-启动线程
另一种方法是声明实现Runnable接口的类,该类然后实现run方法,然后就可以分配该类的实例。start()方法:使该线程开始执行,Java虚拟机调用该线程的run方法。它和run()的区别是run()仅仅是封装被线程执行的代码,直接调用的是普通方法start()首先启动了线程,然后再由jvm去调用该线程的run()方法。

//多线程的实现
//方式1
public class MyThread extends Thread{public void run(){
//一般来说,被线程执行的代码肯定是比较耗时的
for(int x=0;x<10000;x++){
System.out.println("cc"+x);
}
}
}//测试类
public class MyThreadDemo{public static void main (String[] args){
//创建线程对象
MyThread my1=new MyThread();
MyThread my2=new MyThread();
//启动线程
my1.start();
my2.start();
}
}

5)获取和设置线程对象的名称
获取:
public final String getName(); //程序将输出Thread-?,?按顺序从0开始

设置
public final void setName(String name);

public class MyThread extends Thread{//无参构造
public MyThread(){}
//带参构造
public MyThread(String name){
super(name);
}
public void run(){
//一般来说,被线程执行的代码肯定是比较耗时的
for(int x=0;x<10000;x++){
System.out.println("getName()+cc"+x);
}
}
}
//创建线程对象
MyThread my1=new MyThread();
MyThread my2=new MyThread();
//设置名称
my1.setName("cc");
my2.setName("dd");
//启动线程
my1.start();//Thread-0
my2.start();//Thread-1
}
}//带参构造
MyThread my1=new Mythread("许先生");
MyThread my1=new Mythread("刘先生");//获取main方法对应线程的名称
//public static Thread currentThread()
返回当前正在执行的线程对象
System.out.println(Thread currrentThread().getName());

6)线程调度
①分时调度模型:
所有线程轮流使用CPU的使用权,平均分配每个线程占用CPU的时间片
②抢占式调度模型 优先让优先级高的线程使用CPU。如果线程的优先级相同,那么会随机选择一个,优先级高的线程获取的CPU时间片相对多一些。Java使用此类模型

设置优先级的方法:
public final void setPriority (newPriority)
MAX_PRIORITY=10
MIN_PRIORITY=1
NORM_PRIORITY=5

获取线程对象的优先级,默认优先级是5:
public final int getPriority()

public class ThreadPriority extends Thread{public void run(){
//一般来说,被线程执行的代码肯定是比较耗时的
for(int x=0;x<10000;x++){
System.out.println("getName()+cc"+x);
}
}
}public class ThreadPriorityDemo{public static void main(String args[]){
ThreadPriority tp1=new ThreadPriority();
ThreadPriority tp2=new ThreadPriority();
ThreadPriority tp3=new ThreadPriority();
tp1.setName("aa");
tp2.setName("bb");
tp3.setName("cc");tp1.getPriority();
tp2.getPriority();
tp3.getPriority()tp1.setPriorty(8);//存在随机性tp1.start();
tp2.start();
tp3.start();
}
}

7)线程控制
线程休眠
public static void sleep(long mills)
在指定毫秒内让当前正在执行的线程休眠
线程加入
public final void join()
等待该线程终止

public class ThreadJoin extends Thread{public void run(){
for(int x=0;x<10000;x++){
System.out.println("getName()+cc"+x);
}
}
}
public class ThreadJoinDemo{public static void main(String args[]){
ThreadJoin tp1=new ThreadJoin();
ThreadJoin tp2=new ThreadJoin();
ThreadJoin tp3=new ThreadJoin();
tp1.setName("aa");
tp2.setName("bb");
tp3.setName("cc");tp1.start();
tp1.join();
tp2.start();
tp3.start();
}
}

线程礼让
public static void yield()

public class ThreadJoin extends Thread{public void run(){
for(int x=0;x<10000;x++){
System.out.println("getName()+cc"+x);
Thread.yield();
}
}
}
public class ThreadYieldDemo{public static void main(String args[]){
ThreadYield tp1=new ThreadYield();
ThreadYield tp2=new ThreadYield();tp1.setName("aa");
tp2.setName("bb");tp1.start();
tp2.start();}
}

后台线程
public final void setDaemon(boolean on)
将该线程标记为守护线程或用户线程,当正在运行的线程都是守护线程时,jvm退出。必须在启动线程前调用。

中断线程
public final void stop()//已过时但仍可使用,具有不安全性
public void interrupt()//把线程状态终止,并抛出InterruptedException 异常

public class ThreadStop extends Thread{public void run(){
System.out.println("开始执行:"+new Date());
}
}
public class ThreadStopDemo{public static void main(String args[]){
ThreadStop ts=new ThreadStop();
ts.start();
Thread.sleep(3000);
//ts.stop();
ts.intereupt();
}
}

8)线程生命周期图解:

9)多线程的实现方案2
①好处:可以避免由于java单继承而带来的局限性。适合多个相同程序的代码去处理同一个资源的情况,把线程同程序的代码,数据有效分离,较好地体现; 面向对象的设计思想。
-自定义类MyRunnable实现Runnable接口
-重写run()方法
-创建MyRunnable类的对象
-创建Thread类的对象,作为上一步骤的参数传递

public class MyRunnable implements Runnable{public void run(){
for(int x=0;x<100){
System.out.println(Thread.currentThread.geiName()+":"+x);
}
}
}
public class MyRunnableDemo{public static void main(String args[]){
MyRunnable my=new MyRunnable();//Thread t1=new Thread(my,"cc");
//Thread t2=new Thread(my,"dd");
Thread t1=new Thread(my);
Thread t2=new Thread(my);
t1.setName("cc");
t1.setName("dd");
t1.start();
t2.start();
}
}

②两种方式的比较图解

@例题1:售卖电影票

public class SellTicketDemo{public static void main(String args[]){
SellTicket st1=new SelTicket();
SellTicket st2=new SelTicket();
SellTicket st3=new SelTicket();st1.setName("窗口1");
st2.setName("窗口2");
st3.setName("窗口3");st1.start();
st2.start();
st3.start();
}
}public class SellTicket extends Thread{public void run(){
private static int tickets=100;public void run(){
while(true){
if(tickets>0){
System.out.println(getName()+"正在售出第"+(tickets--)+"张票");
}
}
}
}
}
//第二种方式
public class SellTickets implements Runnable{private int tickets=100;
public void run(){
while(true){
if(tickets>0){
System.out.println(getName()+"正在售出第"+(tickets--)+"张票");
}
}
}
}
public class void main(String args[]){SellTicket st=new SellTicket();Thread t1=new Thread(st,"窗口1");
Thread t2=new Thread(st,"窗口2");
Thread t1=new Thread(st,"窗口3");t1.start();
t2.start();
t3.start();
}

改进每次售出一张票延迟0.1秒

//第二种方式
public class SellTickets implements Runnable{private int tickets=100;
//创建锁对象
private Object obj =new Object();
public void run(){
synchronized(obj){
while(true){
if(tickets>0){
Thread.sleep(1000);
System.out.println(getName()+"正在售出第"+(tickets--)+"张票");
}}
}
}
}
public class void main(String args[]){SellTicket st=new SellTicket();Thread t1=new Thread(st,"窗口1");
Thread t2=new Thread(st,"窗口2");
Thread t1=new Thread(st,"窗口3");t1.start();
t2.start();
t3.start();
}

出现了问题:
①相同号码的票售出多次
②出现了负数序号的票
CPU的每一次执行必须是一个原子性(最简单基本的)操作。是由于随机性和延迟导致的

利用同步块的方式解决上述问题
同步代码块:
格式:synchronized(对象)(需要同步的代码;)
可以解决安全问题,其对象可以是任何对象。
把多条语句操作共享数据的部分给包起来
同步的好处与弊端:
好处:同步的出现解决了多线程的安全问题
弊端:当线程相当多时,因为每个线程都会去判断同步上的锁,非常耗费系统资源
前提:多个线程使用同一把锁
同步方法:把同步关键字加载到方法上
A:同步代码块的锁对象是谁呢?
* 任意对象。
*
* B:同步方法的格式及锁对象问题?
* 把同步关键字加在方法上。
*
* 同步方法是谁呢?
* this
*
* C:静态方法及锁对象问题?
* 静态方法的锁对象是谁呢?
* 类的字节码文件对象。(反射会讲)

//再次改进
public class SellTicketDemo {public static void main(String[] args) {// 创建资源对象SellTicket st = new SellTicket();// 创建三个线程对象Thread t1 = new Thread(st, "窗口1");Thread t2 = new Thread(st, "窗口2");Thread t3 = new Thread(st, "窗口3");// 启动线程t1.start();t2.start();t3.start();}
}
package cn.itcast_11;public class SellTicket implements Runnable {// 定义100张票private static int tickets = 100;// 定义同一把锁private Object obj = new Object();private Demo d = new Demo();private int x = 0;@Overridepublic void run() {while (true) {if(x%2==0){synchronized (SellTicket.class) {if (tickets > 0) {try {Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName()+ "正在出售第" + (tickets--) + "张票 ");}}}else {private static synchronized void sellTicket() {if (tickets > 0) {try {Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName()+ "正在出售第" + (tickets--) + "张票 ");}
}
}

8)线程安全的类
StringBuffered
Vector
Hashtable
collections下有很多线程安全的类

转载于:https://www.cnblogs.com/Tanqurey/p/10485332.html

JAVA自学笔记23相关推荐

  1. JAVA自学笔记24

    JAVA自学笔记24 1.能使用同步代码块就使用同步代码块,除非锁对象是this,就可以考虑使用同步方法.静态方法的锁是类的字节码对象. 2.JDK5新特性 1)接口Lock void Lock()/ ...

  2. JAVA自学笔记07

    JAVA自学笔记07 1.构造方法 1) 例如:Student s = new Student();//构造方法 System.out.println(s);// Student@e5bbd6 2)功 ...

  3. JAVA自学笔记22

    JAVA自学笔记22 1.操作基本数据类型的流 DataInputStream DataOutputStream 数据输出流允许应用程序以适当方式将基本的Java数据类型写入输出流中.然后,应用程序可 ...

  4. JAVA自学笔记21

    JAVA自学笔记21 1.转换流 由于字节流操作中文不是非常方便,因此java提供了转换流 字符流=字节流+编码表 1)编码表 由字符及其对应的数值组成的一张表 图解: 2)String类的编码和解码 ...

  5. JAVA自学笔记08

    JAVA自学笔记08 1.构造方法私有,外界就不能再创建对象 2.说明书的制作过程 1)写一个工具类,在同一文件夹下,测试类需要用到工具类,系统将自动编译工具类:工具类的成员方法一般是静态的,因此在测 ...

  6. JAVA自学笔记25

    JAVA自学笔记25 1.GUI 1)图形用户接口,以图形的方式,来显示计算机操作的界面,更方便更直观 2)CLI 命令行用户接口,就是常见的Dos,操作不直观 3) 类Dimension 类内封装单 ...

  7. Java自学笔记——Java面向对象——04.抽象类、接口、内部类

    Java 面向对象各节 Java自学笔记--Java面向对象--01.回顾方法 Java自学笔记--Java面向对象--02.构造器.类和对象 Java自学笔记--Java面向对象--03.封装.继承 ...

  8. 廖雪峰Java自学笔记------Java简洁

    廖雪峰Java自学笔记------Java简洁 这是专门针对小白的零基础Java教程. 为什么要学Java? 因为Java是全球排名第一的编程语言,Java工程师也是市场需求最大的软件工程师,选择Ja ...

  9. Java自学笔记总结04

    Java自学笔记04 一.Lambda表达式 1.1 函数式编程思想概述 1.2 体验Lambda表达式 1.3 Lambda表达式的标准格式 1.4 Lambda表达式的使用 1.5 Lambda表 ...

最新文章

  1. 2019-06-03 Java学习日记 day24 多线程
  2. java 隐藏文件_java-如何仅列出jtree中的非隐藏文件和非系...
  3. 数字谋定农村建设-农业大健康·万祥军:农业人工智能应用
  4. android 恶意广告,CheckPoint:Android恶意广告软件SimBad被下载近1.5亿次
  5. 栅格像元做地理坐标映射后的坐标值是像元中心点还是其他位置?
  6. Spring Cloud Alibaba迁移指南(一):一行代码从 Hystrix 迁移到 Sentinel
  7. Android手机编程初学遇到的问题及解决方法
  8. mysql反掩码_Linux运维:文件目录管理
  9. LVM: Logical Volume Manager 逻辑卷管理
  10. jmap 文件解析_jvm系列:dump文件深度分析
  11. js实现身份证验证功能
  12. 怎样卸载deepin系统_双系统卸载deepin的方法是什么_双系统完全卸载deepin的方法图文步骤...
  13. php输出跳转下一页,tp5页面输出时,搜索后跳转下一页的处理
  14. 激光投影仪与普通投影仪有什么区别?应该怎么购买?
  15. Python基础之列表、元组、字典、集合的使用
  16. 编译原理陈火旺第三版第七章课后题答案
  17. python合并两个列表并去重_python合并多个txt文件
  18. 游戏开发unity性能优化:DOTS导航帖
  19. python--lintcode109.数字三角形(动态规划)
  20. Android开发——Android手机屏幕适配方案总结

热门文章

  1. python之工作举例:通过复制NC文件来造数据
  2. hadoop fs 运维常用的几个命令
  3. 查询字符串某个字符的个数
  4. Adsense加入黑名单的预防办法
  5. 遇到问题了 .net项目发布到iis6,没有权限访问!?
  6. 区块链技术怎么构架落地应用?
  7. Python~win32com~Excel
  8. 使用tableView崩溃
  9. RHEL/CentOS下编译安装Nginx
  10. [转载]C#异步委托的用法 .