现在的位置: 首页 > 综合 > 正文

别样JAVA学习(七)多线程中

2018年11月02日 ⁄ 综合 ⁄ 共 2231字 ⁄ 字号 评论关闭

1、多线程的安全

通过分析,发现,打印出0,-1,-2等错票。

多线程的运行出现了安全问题。

问题的原因:

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

2、同步代码块
Java对多线程的安全问题提供了专业的解决方式。
就是同步代码块

synchronized(对象)
{
       需要被同步的代码
}
对象如同锁,持有锁的线程可以在同步中执行。
没有持有锁的线程及时获取cpu的执行权,也进不去。
比如上火车卫生间----经典。

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

必须保证同步中中只能有一个线程执行。
好处:解决了多线程的安全问题。
弊端:多个线程都需要判断锁,较为消耗资源,

3、同步函数
需求:
银行有一个金库,
有两个储户分别存300员,每次存100,存3次。

结果:

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

通过该程序进行验证。
使用两个线程来买票,
一个线程在同步代码中,
一个线程在同步函数中。
都在执行买票动作



5、静态同步函数的锁是Class对象
当在上代码中的成员变量tick和成员函数show()前面加上static时

class Ticket implements Runnable
{
	private static int tick=100;
	Object obj=new Object();
	boolean flag=true;
	public void run()
	{
		if(flag)
		{
			while(true)
			{
				synchronized(this)
				{
					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()
	{
		if(tick>0)
		{
			try{Thread.sleep(10);}catch(Exception e){}
			System.out.println(Thread.currentThread().getName()+".....show..."+tick--);
		}
	}
}

发现如果同步函数被静态修饰后,使用的锁是什么?

通过验证,发现不再是this。因为静态方法中也不可以定义this。

静态进内存是,内存中没有本类对象,但是一定有该类对应的字节码文件对象。


类名.class         该对象的类型是class

静态的同步方法,使用的锁是该方法所在类的字节码文件对象。类名.class

这程序不咋用,能看懂就行。


6、单例设计模式-懒汉式

class Single
{
	private static Single s=null;
	private Single(){}
	
	public static Single getInstance()
	{
		if(s==null)
			s=new Single();
		return s;
	}
}

多线程

class Single
{
	private static Single s=null;
	private Single(){}
	
	public static synchronized Single getInstance()
	{
		if(s==null)
			s=new Single();
		return s;
	}
}

多线程高效

class Single
{
    private static Single s=null;
    private Single(){}
    
    public static  Single getInstance()
    {
        if(s==null)
        {
            synchronized(Single.class)
            {
                if(s==null)
                    s=new Single();
            }
        }
        return s;
    }
}

7、多线程-死锁


8、线程通信-示例代码


9、线程间通信-解决安全问题

class Input implements Runnable
{
	private Res r;
	Object obj=new Object();
	Input(Res r)
	{
		this.r=r;
	}
	public void run()
	{
		int x=0;
		while(true)
		{
			synchronized(Input.class)
			{
				if(x==0)
				{
					r.name="mike";
					r.sex="man";
				}
				else
				{
					r.name="丽丽";
					r.sex="女女";
				}
				x=(x+1)%2;
			}
		}
	}
}
class Output implements Runnable
{
	private Res r;
	Object obj=new Object();
	Output(Res r)
	{
		this.r=r;
	}
	public void run()
	{
		while(true)
		{
			synchronized(Input.class)
			{
				System.out.println(r.name+"....."+r.sex);
			}
		}
	}
}

抱歉!评论已关闭.