java中多线程与锁的理解及运用

线程概念 :

一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。
栈中的方法: 是拷贝过来的 称之为 (栈帧)

两个线程拷贝 内部声明变量 堆中公用 栈中拷贝

各自拷贝 进行加操作;
线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程包含以下内容:

  1. 一个指向当前被执行指令的指令指针;

  2. 一个栈;

  3. 一个寄存器值的集合,定义了一部分描述正在执行线程的处理器状态的值

  4. 一个私有的数据区。

锁: 锁方法里的变量 共享变量时加锁
引用类型加锁要注意 :

例题: 售票问题

public class ThreadDemo extends Thread{
	int tickets = 100;
    public void run() {
      while (tickets > 0) {
          sale();
      }
}
    public synchronized void sale() {
        if (tickets > 0) {
        	 try
        	      {
        		 Thread.sleep(100);
        	      }   
        	 catch (InterruptedException e)
        	  {
        		 e.printStackTrace();
        	  }
            System.out.println(Thread.currentThread().getName() + "卖第"
                    + (100 - tickets + 1) + "张票"); //获取执行当前线程的名字并打印输出
            tickets--;
        }
      }
}
public class Test {
	 public static void main(String[] args) {  
		    ThreadDemo t= new ThreadDemo();
	        Thread t1 = new Thread(t,"一号售票窗口");
	        Thread t2 = new Thread(t,"二号售票窗口");
	        Thread t3 = new Thread(t,"三号售票窗口");
	        t1.start();
	        t2.start();
	        t3.start();  
		    }
	}

1.类锁(synchronized(静态对象)):类的所有对象都要竞争锁。
互相
2.方法锁((也是对象锁) 在方法前加synchronized): 同一对象同一方法需要竞争锁。

3.对象锁(synchronized(对象)):同一对象代码块竞争锁。

类锁与对象锁 互相锁不住

静态锁 --------->类锁
非静态 -------->对象锁
.
并发(加锁) 多个进程共享一个变量时加锁
并行 (不加锁)

首先的明白Java中锁的机制

synchronized

在修饰代码块的时候需要一个reference对象作为锁的对象.
在修饰方法的时候默认是当前对象作为锁的对象.
在修饰类时候默认是当前类的Class对象作为锁的对象.

方法锁(synchronized修饰方法时)

通过在方法声明中加入 synchronized关键字来声明 synchronized 方法。
synchronized 方法控制对类成员变量的访问:
每个类实例对应一把锁,每个 synchronized 方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。这种机制确保了同一时刻对于每一个类实例,其所有声明为 synchronized 的成员函数中至多只有一个处于可执行状态,从而有效避免了类成员变量的访问冲突。

对象锁(synchronized修饰方法或代码块)
  当一个对象中有synchronized method的时候调用此对象的同步方法或进入其同步区域时,就必须先获得对象锁。如果此对象的对象锁已被其他调用者占用,则需要等待此锁被释放。(方法锁也是对象锁)       
java的所有对象都含有1个互斥锁,这个锁由JVM自动获取和释放。线程进入synchronized方法的时候获取该对象的锁,当然如果已经有线程获取了这个对象的锁,那么当前线程会等待;synchronized方法正常返回或者抛异常而终止,JVM会自动释放对象锁。这里也体现了用synchronized来加锁的1个好处,方法抛异常的时候,锁仍然可以由JVM来自动释放。 
对象锁的两种形式:

public class Test
{
// 对象锁:形式1(方法锁)
public synchronized void Method1()
{
System.out.println(“我是对象锁也是方法锁”);
try
{
Thread.sleep(500);
} catch (InterruptedException e)
{
e.printStackTrace();
}
}

// 对象锁:形式2(代码块形式)

public void Method2()
{
    synchronized (this)
    {
        System.out.println("我是对象锁");
        try
        {
            Thread.sleep(500);
        } catch (InterruptedException e)
        {
            e.printStackTrace();
        }
    }

}


}

类锁(synchronized 修饰静态的方法或代码块)
  由于一个class不论被实例化多少次,其中的静态方法和静态变量在内存中都只有一份。所以,一旦一个静态的方法被申明为synchronized。此类所有的实例化对象在调用此方法,共用同一把锁,我们称之为类锁。  
对象锁是用来控制实例方法之间的同步,类锁是用来控制静态方法(或静态变量互斥体)之间的同步。 
类锁只是一个概念上的东西,并不是真实存在的,它只是用来帮助我们理解锁定实例方法和静态方法的区别的。 
java类可能会有很多个对象,但是只有1个Class对象,也就是说类的不同实例之间共享该类的Class对象。Class对象其实也仅仅是1个java对象,只不过有点特殊而已。由于每个java对象都有1个互斥锁,而类的静态方法是需要Class对象。所以所谓的类锁,不过是Class对象的锁而已。获取类的Class对象有好几种,最简单的就是[类名.class]的方式。

public class Test
{
   // 类锁:形式1
public static synchronized void Method1()
{
System.out.println("我是类锁一号");
try
{
Thread.sleep(500);
} catch (InterruptedException e)
{
e.printStackTrace();
}
}

// 类锁:形式2

public void Method2()
{
    synchronized (Test.class)
    {
        System.out.println("我是类锁二号");
        try
        {
            Thread.sleep(500);
        } catch (InterruptedException e)
        {
            e.printStackTrace();
        }
    }
}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值