Java多线程编程

来源:互联网 发布:人工智能机器人 编辑:程序博客网 时间:2024/05/23 19:17

目的:为了防止多个线程同时访问一个数据对象,对数据造成破坏
作用:确保方法里面的代码一次只由一个线程访问,当被线程访问会加锁,所以其他线程也无法访问,只能等待,当这个线程访问之后(同步方法运行完之后),释放锁,下一个线程(从等待的线程中选取)继续访问,加锁,其他线程等待。

对于同步,一般而言在java代码中需要完成两个操作:
-把竞争访问的资源表示为private。
-同步那些访问资源的代码,使用synchronized关键字来修饰方法或代码块(使用两种方式来实现线程的同步:方法和代码块)

尽量避免嵌套同步块,因为容易引起死锁

多线程同步的例子:银行取钱,一张银行卡500元,一个人在柜台取,一个人在ATM里面取,同步执行,则两个人只有一个人能去除400元,代码中getMoney方法有synchronized代表同步方法,若不加synchronized,则两个人都能取到钱,会出现负数

package cn.com.jlu.demo;public class SynchronizedDemo {    /**     * @param args     */    public static void main(String[] args) {        Bank bank=new Bank();        ThreadDemo t1=new ThreadDemo(bank);        t1.start();        ThreadDemo t2=new ThreadDemo(bank);        t2.start();    }}//创建线程类class ThreadDemo extends Thread{    private Bank bank=null;    public ThreadDemo(Bank bank) {        super();        this.bank = bank;    }    @Override    public void run() {        System.out.println("取钱:"+bank.getMoney(400));    }}class Bank{    //把竞争访问的资源表示为private。    private int money=500;    //num代表要取得钱数,money代表你存的钱数    //方法加上了synchronized关键字,代表同步方法,线程访问是会上锁,其他的线程需等待    public synchronized int getMoney(int num)      {        if(num<0)        {            return -1;        }else if(money<0){            return -2;        }else if(money-num<0){            return -3;//取的钱比存的钱多时,报错        }else        {            try {                Thread.sleep(1000);//模拟取钱的时间1s            } catch (InterruptedException e) {                // TODO Auto-generated catch block                e.printStackTrace();            }            System.out.println("剩余:"+(money-=num));            return num;        }    }}

运行结果
这里写图片描述

同步进程死锁:
(1) 互斥条件:一个资源每次只能被一个进程使用。
(2) 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
(3) 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
(4) 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。

package cn.com.jlu.demo;public class DieThreadDemo {    /**     * @param args     */    public static void main(String[] args) {        Example example=new Example();        Thread1 t1=new Thread1(example);        t1.start();        Thread2 t2=new Thread2(example);        t2.start();    }}class Thread1 extends Thread{    Example example=null;    public Thread1(Example example) {        super();        this.example = example;    }    @Override    public void run() {        example.method1();    }}class Thread2 extends Thread{    public Thread2(Example example) {        super();        this.example = example;    }    Example example=null;    @Override    public void run() {        example.method2();    }}class Example{    //两个线程,第一个线程有资源a,但是等待资源b,而第二个线程拥有资源b等待资源a    Object obja=new Object();    Object objb=new Object();    public void method1()    {        //synchronized修饰代码块,锁定obja,申请objb        synchronized (obja) {            try {                Thread.sleep(1000);            } catch (InterruptedException e) {                // TODO Auto-generated catch block                e.printStackTrace();            }            synchronized (objb) {                System.out.println("method1");              }        }        }        public void method2()        {            //synchronized修饰代码块,锁定obja,申请objb            synchronized (objb) {                try {                    Thread.sleep(1000);                } catch (InterruptedException e) {                    // TODO Auto-generated catch block                    e.printStackTrace();                }                synchronized (obja) {                    System.out.println("method2");                  }            }    }}
0 0
原创粉丝点击