多线程问题的思路与处理方式。


一般:我们把共享资源拿出来,,把各种对共享资源操作的方法与资源封装在一起,在写方法的时候,我们就要考虑到完整性和顺序性。


加锁,只是为了让一个线程能够把操作共享数据的语句连续执行完。保证完整性。

加锁的两种方式:同步synchronized   Lock lock=new ReetrentLock();

如果涉及到顺序问题,在锁里边,借助资源的flag,使用用wait,notify或者await  singnal。


总之,一想到多线程,就把操作的共享资源与操作共享资源的方法封装成一个类,对操作共享数据的方法加锁。

如果多线程涉及到顺序问题,就要在(方法)锁里边操作(await,singnal,,,)。保证顺序。


下边是详细的叙述:

 多个线程,,功能相同,就像卖票一样,Thread(一样)

这个时候的安全问题:是线程对共享数据操作的不完整性。

对操作共享数据的语句加锁,保证线程操作数据的完整性。

这种情况只要一把锁。

class Count

{

private int sum;

public synchronized void add(int n)

{

sum=sum+n;

try{Thread.sleep(10);}catch(Exception e){}

System.out.println("sum"+sum);

}

}


class Person implements Runnable

{

private Count b=new Count();

public void run()

{

for(int x=0;x<3;x++)

{

b.add(100);

}

}

}

class  CountDemo

{

public static void main(String[] args) 

{

Person p=new Person();

Thread t1=new Thread(p);

Thread t2=new Thread(p);

t1.start();

t2.start();

}

}


多个线程功能不同(执行的代码不一样),但还是在操作共享数据。

一般,把这些操作操作共享数据的语句封装成方法,与共享数据形成一个类。


1.对不同线程操作共享数据的顺序无要求:::对各个线程的执行代码加同一个锁。保证对共享数据操作的完整性。

2.对不同线程操作共享数据的顺序有要求:::使用标记(这个标记定义在资源中,共享)

只有两个线程(一对一):if

class Res

{

private String name;

private String sex;

private  boolean flag=false;


public synchronized void set(String name,String sex)

{

if(flag)

{

try{this.wait();}catch(Exception e){}

}

this.name=name;

this.sex=sex;

flag=true;

this.notify();

}

public synchronized void out(int i)

{

if(!flag)

{

try{this.wait();}catch(Exception e){}

}

System.out.println(i+"    "+name+".........."+sex);

flag=false;

this.notify();

}

}


3.有顺序要求。多对多。while(flag)

class Resource 

{

private String name;

private int count=1;

private boolean flag=false;


private Lock lock=new ReentrantLock();

private Condition condition_pro=lock.newCondition();

private Condition condition_con=lock.newCondition();

public void set(String name) throws Exception

{

lock.lock();

try

{

while(flag)

condition_pro.await();

this.name=name+"--"+count++;

System.out.println(Thread.currentThread().getName()+"...生产者..."+this.name);

flag=true;

condition_con.signal();

}

finally

{

lock.unlock();

}

}


public void out() throws Exception

{

lock.lock();

try

{

while(!flag)

condition_con.await();

System.out.println(Thread.currentThread().getName()+"......消费者......"+this.name);

flag=false;

condition_pro.signal();

}

finally

{

lock.unlock();

}

}


}

线程池:线程运行的时候,内存中会建立一个线程池。等待线程都存在线程池当中。notify唤醒线程池中的第一个线程。

线程wait后,就会进入线程池。