黑马程序员—JAVA多线程

------------- android培训java培训、期待与您交流! -------------

多线程学习总结:

线程与进程的区别:

进程:是一个正在执行中的程序。每一个进程执行都有一个执行顺序。该顺序是一个执行路径,或者叫一个控制单元。

线程:就是进程中的一个独立的控制单元。线程在控制着进程的执行。一个进程中至少有一个线程。

Java VM  启动的时候会有一个进程java.exe.该进程中至少一个线程负责java程序的执行。而且这个线程运行的代码存在于main方法中。该线程称之为主线程。jvm,jvm启动不止一个线程,还有负责垃圾回收机制的线程。

 

创建线程方法:

第一种方式:继承Thread类。

步骤

1,定义类继承Thread。

2,复写Thread类中的run方法。

         目的:将自定义代码存储在run方法。让线程运行。

3,调用线程的start方法,

         该方法两个作用:启动线程,调用run方法。

代码示例:

class Demo extends Thread//定义类继承Thread

{

         public void run()

         {

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

                          System.out.println("demo run----"+x);

         }

}

class ThreadDemo

{

         public static void main(String[] args)

         {

                 Demo d = new Demo();//创建好一个线程。

                 d.start();//开启线程并执行该线程的run方法。

                 //d.run();仅仅是对象调用方法。而线程创建了,并没有运行。

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

                          System.out.println("Hello World!--"+x);               

         }

}

知识总结:

覆盖run方法的原因该类就定义了一个功能,用于存储线程要运行的代码。该存储功能就是run方法。也就是说Thread类中的run方法,用于存储线程要运行的代码。

多线程的一个特性:随机性。谁抢到执行权谁执行,至于执行多长,cpu说的算。

线程都有自己默认的名称,Thread-编号 该编号从0开始。 
static Thread currentThread():获取当前线程对象。 
getName(): 获取线程名称。 
设置线程名称:setName或者构造函数。 
代码示例: 
class Test extends Thread 
{ 
         //private String name; 
         Test(String name) 
         { 
                 //this.name = name; 
                 super(name);//调用父类构造函数,设置线程名称 
         } 
         public void run() 
         { 
                 for(int x=0; x<60; x++) 
                       System.out.println((Thread.currentThread().getName()+" run..."+x);// Thread.currentThread()获取当前线程引用 
                 } 
         }   
}   
class ThreadTest  
{ 
         public static void main(String[] args)  
         { 
                 //开启两个线程 
                 Test t1 = new Test("one---"); 
                 Test t2 = new Test("two+++"); 
                 t1.start(); 
                 t2.start(); 
  
                 for(int x=0; x<60; x++) 
                 { 
                          System.out.println("main....."+x); 
                 } 
         } 
} 
创建线程的第二种方式:实现Runable接口 
  
步骤: 
1,定义类实现Runnable接口 
2,覆盖Runnable接口中的run方法。 
         将线程要运行的代码存放在该run方法中。 
  
3,通过Thread类建立线程对象。 
4,将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数。 
         为什么要将Runnable接口的子类对象传递给Thread的构造函数。 
         因为,自定义的run方法所属的对象是Runnable接口的子类对象。 
         所以要让线程去指定指定对象的run方法。就必须明确该run方法所属对象。 
    
5,调用Thread类的start方法开启线程并调用Runnable接口子类的run方法。 
  
实现方式和继承方式的区别: 
实现方式好处:避免了单继承的局限性。在定义线程时,建立使用实现方式。 
两种方式区别: 
继承Thread:线程代码存放Thread子类run方法中。 
实现Runnable,线程代码存在接口的子类的run方法。 
  
代码示例:(以卖票为例) 
class Ticket implements Runnable//实现接口 
{ 
         private  int tick = 100;//定义100张票 
         public void run() 
         { 
                 while(true) 
                 { 
                          synchronized(obj)//解决安全问题 
                          { 
                                   if(tick>0) 
                                   { 
                                            System.out.println(Thread.currentThread().getName()+"....sale : "+ tick--); 
                                   } 
                          }   
                 } 
         } 
}     
class  TicketDemo 
{ 
         public static void main(String[] args)  
         {   
                 Ticket t = new Ticket();   
                 Thread t1 = new Thread(t);//创建了一个线程; 
                 Thread t2 = new Thread(t);//创建了一个线程; 
                 Thread t3 = new Thread(t);//创建了一个线程; 
                 Thread t4 = new Thread(t);//创建了一个线程; 
                 t1.start(); 
                 t2.start(); 
                 t3.start(); 
                 t4.start();  
         } 
} 
  
线程安全问题的原因分析: 
         当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行。导致共享数据的错误。 
  
解决办法: 
         对多条操作共享数据的语句,只能让一个线程都执行完。在执行过程中,其他线程不可以参与执行。Java对于多线程的安全问题提供了专业的解决方式。 
就是同步代码块,同步函数。 
  
synchronized(对象) 
{ 
         需要被同步的代码   
} 
对象如同锁。持有锁的线程可以在同步中执行。没有持有锁的线程即使获取cpu的执行权,也进不去,因为没有获取锁。 
  
同步的前提: 
1,必须要有两个或者两个以上的线程。 
2,必须是多个线程使用同一个锁。 
进行同步操作好处与弊端: 
好处:解决了多线程的安全问题。 
弊端:多个线程需要判断锁,较为消耗资源, 
同步函数介绍: 
函数需要被对象调用,那么函数都有一个所属对象引用。所以同步函数使用的锁就是this。注意:静态的同步方法,使用的锁是该方法所在类的字节码文件对象。 类名.class 
  
代码示例: 
class Ticket implements Runnable 
{ 
         private  int tick = 100; 
         Object obj = new Object(); 
         boolean flag = true; 
         public  void run() 
         { 
                 if(flag) 
                 { 
                          while(true) 
                          { 
                                   synchronized(this)//若如不使用this则出现安全问题,这也证明同步函数使用的锁是this。如果下面是静态同步函数,则用类名.class实现同步。 
                                   { 
                                            if(tick>0) 
                                            { 
                                                    try{Thread.sleep(10);}catch(Exception e){} 
                                                     System.out.println(Thread.currentThread().getName()+"....code : "+ tick--); 
                                            } 
                                   } 
                          } 
                 } 
                 else 
                          while(true) 
                                   show(); 
         } 
         public synchronized void show()//this 
         { 
                 if(tick>0) 
                 { 
                          try{Thread.sleep(10);}catch(Exception e){} 
                          System.out.println(Thread.currentThread().getName()+"....show.... : "+ tick--); 
                 } 
         } 
}  
class  ThisLockDemo 
{ 
         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();   
         } 
} 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值