一、进程与线程

  进程:正在执行的程序称为一个线程,主要负责内存空间的划分。

   线程:线程在一个进程中负责代码的执行,就是进程中的一个执行路径。

   多线程:在一个进程中有多个线程同时在执行不同的任务(同时指的是宏观上)。

  Q:windows为多任务操作系统,那么它可以同时运行多个应用程序吗?

  A:不可以,一次只能运行一个。单核CPU一个时间片中只能执行一个应用程序。

    宏观上:同时运行           微观上:单个执行  

  Q:线程负责了代码的执行,那么我们没用到Thread类创建对象,为什么代码可以执行?一个JAVA程序至少有几个线程?

  A:任何一个Java程序,在jvm运行的时候都会创建main线程执行main方法中的代码;

    2个:main线程,垃圾回收器线程负责垃圾回收。

  多线程的优点:1.解决一个进程可以执行多个任务的问题

         2.提高资源的利用率(不是效率)。

  多线程的缺点:1.增加CPU的负担

         2.降低了进程中单线程的执行概率        

         3.会引发线程安全问题

         4.会出现死锁现象

二、如何创建线程

方法一:

   1.自定义一个类继承Thread类

   2.该子类应重写 Thread 类的 run 方法

   3.创建该子类的实例对象,并调用start()方法来开启线程,一个线程一旦开启,

    那么线程就会执行run()方法中的代码,run()方法不能直接调用,直接调用run方法就相当于调用普通的方法,并没有开启新的线程。


  1. class PrimeThread extends Thread {
  2. long minPrime;
  3. PrimeThread(long minPrime) {
  4. this.minPrime = minPrime;
  5. }
  6.  
  7. public void run() {
  8. // 将自定义线程需要执行的代码放在run()方法中
  9.  . . .
  10. }
  11. }

然后,下列代码会创建并启动一个线程:

  1. PrimeThread p = new PrimeThread(143);
  2. p.start();

方法二:

    创建线程的另一种方法是声明实现 Runnable 接口的类。该类然后实现 run 方法。然后可以分配该类的实例,

    在创建 Thread 时作为一个参数来传递并启动。采用这种风格的同一个例子如下所示:


  1. class PrimeRun implements Runnable {
  2. long minPrime;
  3. PrimeRun(long minPrime) {
  4. this.minPrime = minPrime;
  5. }
  6.  
  7. public void run() {
  8. // compute primes larger than minPrime
  9.  . . .
  10. }
  11. }

然后,下列代码会创建并启动一个线程:

  1. PrimeRun p = new PrimeRun(143);
  2. new Thread(p).start();

  Q:为什么要重写run方法?目的是?

  A:每个线程都有自己的任务代码,JVM创建主线程的任务代码为main方法中的所有代码

   自定义线程的任务代码就要写在run方法中,自定义线程负责run方法中的代码

eg:

  1. public class Demo1 extends Thread {
  2.  
  3. @Override
  4. public void run() {
  5. for(int i = 0;i<100;i++) {
  6. System.out.println("自定义线程"+i);
  7. }
  8. }
  9.  
  10. public static void main(String[] args) {
  11.  
  12. Demo1 a = new Demo1();
  13. a.start();
  14.  
  15. for(int i = 0;i<100;i++) {
  16. System.out.println("main线程"+i);
  17. }
  18. }
  19. }

三、线程的生命周期

  图片借鉴

  运行状态下线程一旦执行了sleep或者wait方法后,该线程会进入临时阻塞状态,如果线程是调用sleep方法进入该状态,一旦超过了指定的睡眠时间,

那么就会重新进入可运行状态,如果调运个了wait方法今日阻塞状态,那么就需要其他线程唤醒该线程才可以重新进入运行状态。

四、线程常用方法

Thread(String name)     初始化线程的名字
etName(String name)    设置线程对象名
getName()             返回线程的名字
sleep()                 线程睡眠指定的毫秒数。 静态的方法, 那个线程执行了sleep方法代码那么就是那个线程睡眠。
currentThread()      返回当前的线程对象,该方法是一个静态的方法, 注意: 那个线程执行了currentThread()代码就返回那个线程 的对象。
getPriority()             返回当前线程对象的优先级   默认线程的优先级是5(主线程优先级获取Thread.CurrentThread.getPriority();)

setPriority(int newPriority) 设置线程的优先级    虽然设置了线程的优先级,但是具体的实现取决于底层的操作系统的实现(最大的优先级是10 ,最小的1 , 默认是5)。

 四、线程安全性问题

以一个程序为例:

  1. class SaleTicket extends Thread{
  2.  
  3. int num = 50;
  4.  
  5. public SaleTicket(String name) {
  6. super(name);
  7. }
  8.  
  9. public void run() {
  10. while(true){
  11. if(num>0){
  12. System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
  13. try {
  14. Thread.sleep(100);
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. num--;
  19. }else{
  20. System.out.println("售罄了..");
  21. break;
  22. }
  23. }
  24. }
  25. }
  26. public class Demo {
  27.  
  28. public static void main(String[] args) {
  29. //创建三个线程对象,模拟三个窗口
  30. SaleTicket thread1 = new SaleTicket("窗口1");
  31. SaleTicket thread2 = new SaleTicket("窗口2");
  32. SaleTicket thread3 = new SaleTicket("窗口3");
  33. //开启线程
  34. thread1.start();
  35. thread2.start();
  36. thread3.start();
  37. }
  38. }

问题1 :为什么50张票被卖出了150次?

出现的原因:因为num是非静态的,非静态的成员变量数据是在每个对象中都会维护一份数据的,三个线程对象就会有三份。

解决方案:把num票数共享出来给三个线程对象使用。使用static修饰。

问题2: 出现了线程安全问题 ?

线程安全问题的解决方案:sun提供了线程同步机制让我们解决这类问题的。
 
 java线程同步机制的方式:
  方式一:同步代码块
   同步代码块的格式:
    
    synchronized(锁对象){
     需要被同步的代码...
    }

同步代码块要注意事项:
  1. 任意的一个对象都可以做为锁对象。
  2. 在同步代码块中调用了sleep方法并不是释放锁对象的。
  3. 只有真正存在线程安全问题的时候才使用同步代码块,否则会降低效率的。
  4. 多线程操作的锁 对象必须 是唯一共享 的。否则无效。
 

 方式二:同步函数(暂时没学到,后期更新)

出现线程安全问题的根本原因:
 1. 存在两个或者两个以上 的线程对象,而且线程之间共享着一个资源。
 2. 有多个语句操作了共享资源。

正确代码:

  1. class SaleTicket extends Thread{
  2.  
  3. static int num = 50;//票数 非静态的成员变量,非静态的成员变量数据是在每个对象中都会维护一份数据的。
  4.  
  5. static Object o = new Object();
  6.  
  7. public SaleTicket(String name) {
  8. super(name);
  9. }
  10.  
  11. public void run() {
  12. while(true){
  13. //同步代码块
  14. synchronized ("锁") {
  15. if(num>0){
  16. System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
  17. try {
  18. Thread.sleep(100);
  19. } catch (InterruptedException e) {
  20. e.printStackTrace();
  21. }
  22. num--;
  23. }else{
  24. System.out.println("售罄了..");
  25. break;
  26. }
  27. }
  28. }
  29. }
  30. }
  31.  
  32. public class Demo4 {
  33.  
  34. public static void main(String[] args) {
  35. //创建三个线程对象,模拟三个窗口
  36. SaleTicket thread1 = new SaleTicket("窗口1");
  37. SaleTicket thread2 = new SaleTicket("窗口2");
  38. SaleTicket thread3 = new SaleTicket("窗口3");
  39. //开启线程售票
  40. thread1.start();
  41. thread2.start();
  42. thread3.start();
  43. }
  44. }

课后习题:一个银行账户5000块,两夫妻一个拿着 存折,一个拿着卡,开始取钱比赛,每次只能取100块,要求不准出现线程安全问题。

学习自测代码:

public class Demo2 extends Thread {
 
 static int money = 5000;
    static Object lock = new Object();
   
 public Demo2(String name) {
  super(name);
 }

public void run() {
  while(true) {
   synchronized (lock) {
    if(money>0) {
     System.out.println(Thread.currentThread().getName()+"取走100元,还剩"+(money-100)+"元!");
     money-=100;
     try {
      Thread.sleep(100);
     } catch (InterruptedException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
     }
    }else {
     System.out.println("银行卡没有存款了!");
     break;
    } 
   }
  }
 }
 
 public static void main(String[] args) {
  Demo2 men = new Demo2("男士");
  Demo2 women = new Demo2("女士");
  
  men.start();
  women.start();
 }

}

Day 1:线程与进程系列问题(一)的更多相关文章

  1. Day 2:线程与进程系列问题(二)

    补充: 线程的创建方式二: 1.自定义一个实现Runnable接口的类 2.实现Runnable接口中的run方法把自定义线程的任务写在run方法中 3.创建实现Runnable接口的对象 4.创建T ...

  2. 进程与线程 .Net Core系列-多线程

    进程与线程 进程: 狭义定义:进程是正在运行的程序的实例 广义定义:进程是一个具有一定独立功能的程序关于某个数据集合的一次运行活动.它是操作系统动态执行的基本单元,在传统的操作系统中,进程既是基本的分 ...

  3. python并发编程之Queue线程、进程、协程通信(五)

    单线程.多线程之间.进程之间.协程之间很多时候需要协同完成工作,这个时候它们需要进行通讯.或者说为了解耦,普遍采用Queue,生产消费模式. 系列文章 python并发编程之threading线程(一 ...

  4. Python 【第五章】:线程、进程和协程

    Python线程 Threading用于提供线程相关的操作,线程是应用程序中工作的最小单元. #!/usr/bin/env python # -*- coding:utf-8 -*- import t ...

  5. Python网络编程之线程,进程

    一. 线程: 基本使用 线程锁 线程池 队列(生产者消费者模型) 二. 进程:  基本使用  进程锁 进程池 进程数据共享 三. 协程: gevent greenlet 四. 缓存: memcache ...

  6. iOS开发:(线程篇-上)线程和进程

    iOS开发多线程篇—多线程简单介绍 一.进程和线程 1.什么是进程 进程是指在系统中正在运行的一个应用程序 每个进程之间是独立的,每个进程均运行在其专用且受保护的内存空间内 比如同时打开QQ.Xcod ...

  7. LR中线程和进程的区别

    LoadRunner中的进程与线程    1.进程与线程的区别: 进程和线程的区别是什么?进程和线程都是由操作系统所体会的程序运行的基本单元,系统利用该基本单元实现系统对应用的并发性.进程和线程的区别 ...

  8. iOS_线程和进程的区别与联系

    首先是线程和进程的联系: 线程和进程都是由操作系统所负责的程序运行的基本单元,系统利用该基本单元实现对应用的并发性. 接下来就是线程和进程的区别: 线程和进程最大的区别就是它们是操作系统的两种资源管理 ...

  9. python学习笔记12 ----线程、进程

    进程和线程的概念 进程和线程是操作系统中两个很重要的概念,对于一般的程序,可能有若干个进程,每一个进程有若干个同时执行的线程.进程是资源管理的最小单位,线程是程序执行的最小单位(线程可共享同一进程里的 ...

随机推荐

  1. Jmockit如何同时mock多个类的静态方法

    使用Jmockit来mock类的静态方法,网上有很多mock单个类静态方法的例子.有使用Expectations,还有使用MockUp的. 但是如果想在一个测试方法中mock多个类的静态方法,该怎么写 ...

  2. kubernetes集群全栈监控报警方案kube-prometheus

    参考文档 http://www.servicemesher.com/blog/prometheus-operator-manual/ https://github.com/coreos/prometh ...

  3. ThinkCMF框架上的任意内容包含漏洞

    一.背景 ThinkCMF是一款基于PHP+MYSQL开发的中文内容管理框架,底层采用ThinkPHP3.2.3构建. ThinkCMF提出灵活的应用机制,框架自身提供基础的管理功能,而开发者可以根据 ...

  4. python集成开发环境Anaconda的安装

    参考博文: anaconda在Linux下的安装 Linux下anaconda3的安装 Anaconda的安装.启用及停用的步骤 Python学习之Anaconda的使用及配置方法 Anaconda ...

  5. hadoop 配置问题以及HDFS下如何读写文件

    辛辛苦苦学两年 ,一举回到解放前!!! 大数据开始学真的头疼 关键是linux你玩的不6 唉难受 hadoop 配置参见博客 http://dblab.xmu.edu.cn/blog/install- ...

  6. poj 1318 Word Amalgamation

    Word Amalgamation Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 9968   Accepted: 4774 ...

  7. 八十、SAP中数据库操作之 (FOR ALL ENTRIES IN )用法,比较难明白

    一.代码如下 二.查询结果如下 三.我们把it_spfli的内表内容添加两个 四.结果如下,查询的是所有的JFK和SFO机场. *&------------------------------ ...

  8. Power BI角色控制

    Case:企业的数据分析报表经常需要进行权限控制,根据读者的部门或职位,决定他可以看到的数据.例如,A部门的人只能查看A部门的数据,B部门的人只能查看B部门的数据,而领导层则可以看到所有的数据. 1, ...

  9. Web.config中executionTimeout的单位

    executionTimeout:表示允许执行请求的最大时间限制,单位为秒

  10. Mac电脑上怎么设置环境变量

    https://jingyan.baidu.com/article/8065f87f47b29523312498e4.html 环境变量是电脑操作系统中常用的一些变量,作用类似于将一些常用命令所在的文 ...