java--多线程:synchronized

多线程的运行有安全问题。


问题的原因:
当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,
另一个线程参与进来执行。导致共享数据的错误。
解决办法:
对多条操作共享数据的语句,只能让一个线程都执行完。在执行过程中,其他线程不可以参与执行。

Java对于多线程的安全问题提供了专业的解决方式。
就是同步代码块。
synchronized(对象)
{
需要被同步的代码
}
synchronized:锁,监视器。默认值是0,有程序进入锁中,默认值改为1,程序完成出去后,默认值改成0;
对象如同锁。持有锁的线程可以在同步中执行。
没有持有锁的线程即使获取cpu的执行权,也进不去,因为没有获取锁。

同步的前提:
1,必须要有两个或者两个以上的线程。
2,必须是多个线程使用同一个锁。

必须保证同步中只能有一个线程在运行。

如何找到同步代码:

1,明确哪些代码是多线程运行代码。
2,明确共享数据。
3,明确多线程运行代码中哪些语句是操作共享数据的。

synchronized:
好处:解决了多线程的安全问题。

弊端:多个线程需要判断锁,较为消耗资源。


synchronized:它包括两种用法:synchronized 方法和 synchronized 块。 
1.synchronized方法:通过在方法声明中加入 synchronized关键字来声明 synchronized 方法。如: 
public synchronized void mathod(int a); 
synchronized 方法的缺陷:若将一个大的方法声明为synchronized 将会大大影响效率,典型地,若将线程类的方法 run() 声明为
synchronized ,由于在线程的整个生命期内它一直在运行,因此将导致它对本类任何 synchronized 方法的调用都永远不会成功。当然我们可以通过将访问类成员变量的代码放到专门的方法中,将其声明为 synchronized ,并在主方法中调用来解决这一问题,但是 Java 为我们提供了更好的解决办法,那就是 synchronized 块。 

class  Test
{
	public static void main(String[] args)
	{
		Demo d=new Demo();
		Thread d1=new Thread(d);
		Thread d2=new Thread(d);
		d1.start();
		d2.start();
	}
}
class Demo implements Runnable
{
	int tick=10;
	public synchronized void run()//synchronized方法
	{
		while(tick>0)
		{
			{
				try{Thread.sleep(10);}catch(Exception e){}
				System.out.println(Thread.currentThread()+"......run......."+tick--);
			}
			
		}
	}

}
2. synchronized 块:通过 synchronized关键字来声明synchronized 块。语法如下: 
  synchronized(Object obj) { 
  //允许访问控制的代码 
  } 
synchronized 块是这样一个代码块,其中的代码必须获得对象obj(如前所述,可以是类实例或类)的锁方能执行,具体机制同前所述。由于可以针对任意代码块,且可任意指定上锁的对象,故灵活性较高。 

class  Test
{
	public static void main(String[] args)
	{
		Demo d=new Demo();
		Thread d1=new Thread(d);
		Thread d2=new Thread(d);
		d1.start();
		d2.start();
	}
}
class Demo implements Runnable
{
	int tick=10;
	Object o=new Object();
	public void run()
	{
		while(tick>0)
		{
			synchronized(o)//synchronized块,也可以是synchronized(this)
			{
				try{Thread.sleep(10);}catch(Exception e){}
				System.out.println(Thread.currentThread()+"......run......."+tick--);
			}
			
		}
	}

}

同步函数用的是哪一个锁呢?
函数需要被对象调用。那么函数都有一个所属对象引用。就是this。
所以同步函数使用的锁是this。

如果同步函数被静态修饰后,使用的锁是什么呢?
通过验证,发现不在是this。因为静态方法中也不可以定义this。
静态进内存是,内存中没有本类对象,但是一定有该类对应的字节码文件对象。
类名.class  该对象的类型是Class
静态的同步方法,使用的锁是该方法所在类的字节码文件对象。 类名.class

class  Test
{
	public static void main(String[] args)
	{
		Ticket t = new Ticket();

		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		t1.start();
		try{Thread.sleep(10);}catch(Exception e){}
		t.flag=false;
		t2.start();
	}
}
class Ticket implements Runnable
{
	private static int tick = 100;
	Object obj = new Object();
	boolean flag = true;
	public  void run()
	{
		if(flag)
		{
			while(true)
			{
				synchronized(Ticket.class)
				{
					if(tick>0)
					{
						try{Thread.sleep(10);}catch(Exception e){}
						System.out.println(Thread.currentThread().getName()+"....code : "+ tick--);
					}
				}
			}
		}
		else
			while(true)
				show();
	}
	public static synchronized void show()//Ticket.class
	{
		if(tick>0)
		{
			try{Thread.sleep(10);}catch(Exception e){}
			System.out.println(Thread.currentThread().getName()+"....show.... : "+ tick--);
		}
	}
}

死锁:同步中嵌套同步。

package Test;
//死锁
class Demo10 implements Runnable
{
	private boolean flag;
	Demo10(boolean flag)
	{
		this.flag = flag;
	}
	public void run()
	{
		if(flag)
		{
			while(true)
			{
				synchronized(MyLock.locka)
				{
					System.out.println(Thread.currentThread().getName()+"...if locka ");
					synchronized(MyLock.lockb)
					{
						System.out.println(Thread.currentThread().getName()+"..if lockb");					
					}
				}
			}
		}
		else
		{
			while(true)
			{
				synchronized(MyLock.lockb)
				{
					System.out.println(Thread.currentThread().getName()+"..else lockb");
					synchronized(MyLock.locka)
					{
						System.out.println(Thread.currentThread().getName()+".....else locka");
					}
				}
			}
		}
	}
}
class MyLock
{
	static Object locka = new Object();
	static Object lockb = new Object();
}
class  Test10
{
	public static void main(String[] args) 
	{
		Thread t1 = new Thread(new Demo10(true));
		Thread t2 = new Thread(new Demo10(false));
		t1.start();
		t2.start();
	}
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值