线程概念 :
一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。
栈中的方法: 是拷贝过来的 称之为 (栈帧)
两个线程拷贝 内部声明变量 堆中公用 栈中拷贝
各自拷贝 进行加操作;
线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程包含以下内容:
-
一个指向当前被执行指令的指令指针;
-
一个栈;
-
一个寄存器值的集合,定义了一部分描述正在执行线程的处理器状态的值
-
一个私有的数据区。
锁: 锁方法里的变量 共享变量时加锁
引用类型加锁要注意 :
例题: 售票问题
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();
}
}
}
}