一、进程与线程

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

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

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

  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方法就相当于调用普通的方法,并没有开启新的线程。


     class PrimeThread extends Thread {
long minPrime;
PrimeThread(long minPrime) {
this.minPrime = minPrime;
} public void run() {
// 将自定义线程需要执行的代码放在run()方法中
 . . .
}
}

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

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

方法二:

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

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


     class PrimeRun implements Runnable {
long minPrime;
PrimeRun(long minPrime) {
this.minPrime = minPrime;
} public void run() {
// compute primes larger than minPrime
 . . .
}
}

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

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

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

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

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

eg:

public class Demo1 extends Thread {

    @Override
public void run() {
for(int i = 0;i<100;i++) {
System.out.println("自定义线程"+i);
}
} public static void main(String[] args) { Demo1 a = new Demo1();
a.start(); for(int i = 0;i<100;i++) {
System.out.println("main线程"+i);
}
}
}

三、线程的生命周期

  图片借鉴

  运行状态下线程一旦执行了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)。

 四、线程安全性问题

以一个程序为例:

class SaleTicket extends Thread{

     int num = 50;

     public SaleTicket(String name) {
super(name);
} public void run() {
while(true){
if(num>0){
System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
num--;
}else{
System.out.println("售罄了..");
break;
}
}
}
}
public class Demo { public static void main(String[] args) {
//创建三个线程对象,模拟三个窗口
SaleTicket thread1 = new SaleTicket("窗口1");
SaleTicket thread2 = new SaleTicket("窗口2");
SaleTicket thread3 = new SaleTicket("窗口3");
//开启线程
thread1.start();
thread2.start();
thread3.start();
}
}

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

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

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

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

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

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

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

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

正确代码:

class SaleTicket extends Thread{

     static int num = 50;//票数  非静态的成员变量,非静态的成员变量数据是在每个对象中都会维护一份数据的。

     static    Object o = new Object();

     public SaleTicket(String name) {
super(name);
} public void run() {
while(true){
//同步代码块
synchronized ("锁") {
if(num>0){
System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
num--;
}else{
System.out.println("售罄了..");
break;
}
}
}
}
} public class Demo4 { public static void main(String[] args) {
//创建三个线程对象,模拟三个窗口
SaleTicket thread1 = new SaleTicket("窗口1");
SaleTicket thread2 = new SaleTicket("窗口2");
SaleTicket thread3 = new SaleTicket("窗口3");
//开启线程售票
thread1.start();
thread2.start();
thread3.start();
}
}

课后习题:一个银行账户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. flutter如何使用配置文件pubspec.yaml(位于项目根目录)来管理第三方依赖包

    官方文档 在软件开发中,很多时候有一些公共的库或SDK可能会被很多项目用到,因此,将这些代码单独抽到一个独立模块,然后哪个项目需要使用时再直接集成这个模块,便可大大提高开发效率.很多编程语言或开发工具 ...

  2. 第2节 网站点击流项目(下):3、流量统计分析,分组求topN

    四. 模块开发----统计分析 select * from ods_weblog_detail limit 2;+--------------------------+---------------- ...

  3. day01-Python运维开发基础

    还是用思维导图来一遍,印象更深!

  4. 027、MySQL字符串替换函数,文本替换函数,字符串填充函数

    #文本填充 ,'); #ABC12121212121212121 #文本替换 SELECT REPLACE('田攀520','攀','ABC'); #田ABC520 不忘初心,如果您认为这篇文章有价值 ...

  5. 7.9 规划Varnish缓存

    ./varnishlog -i VCL_LOG

  6. springboot启动微服务项目时,启动后没有端口号信息,也访问不了

    2018-06-05 13:43:42.282 [localhost-startStop-1] DEBUG org.apache.catalina.core.ContainerBase - Add c ...

  7. Redis 详解 (三) redis的五大数据类型详细用法

    目录 1.string 数据类型 2.hash 数据类型 3.list 数据类型 4.set 数据类型 5.zset 数据类型 6.系统相关命令 7.key 相关命令 我们说 Redis 相对于 Me ...

  8. Windows平台整合SpringBoot+KAFKA_第1部分_环境配置部分

    项目需要,需要整合 SpringBoot+KAFKA 我调查了一下,发现Linux中,要先装zoomkeeper,再装KAFKA,如  https://blog.csdn.net/zhangcongy ...

  9. 【转】ASP.NET Core WebAPI JWT Bearer 认证失败返回自定义数据 Json

    应用场景:当前我们给微信小程序提供服务接口,接口中使用了权限认证这一块,当我使用 JWT Bearer 进行接口权限认证的时候,返回的结果不是我们客户端想要的,其它我们想要给客户端返回统一的数据结构, ...

  10. centos 7 打开端口

    查看防火墙状态 systemctl status firewalld 或者 firewall-cmd --state 临时关闭防火墙测试是否是端口问题 systemctl stop firewalld ...