您的位置:首页 > 编程语言 > Java开发

Java多线程(一) 多线程的基本使用

2016-07-15 14:16 127 查看

  在总结JDBC数据库连接池的时候,发现Java多线程这块掌握得不是很好,因此回头看了下多线程的内容。做一下多线程模块的学习和总结,稳固一下多线程这块的基础。关于多线程的一些理论知识,这里不想啰嗦太多,可以进行下搜索了解。

1. 如何使用Java创建多线程

  使用Java多线程,总的来说方法有两种:①继承Thread类,重写run方法  ②把相关的类实现Runnable(可运行)接口,重写run方法。③实现Callable接口(相对用得较少)

 

package com.scl.thread;

public class TestThread
{
public static void main(String[] args) throws Exception
{
String mainName = Thread.currentThread().getName();
System.out.println(mainName + "线程开始运行");

// 方法1 使用继承对线程进行调用
MyThread myThread = new MyThread();
myThread.setName("wft");
myThread.start();
myThread.join();

// 方法2 继承Runnable接口进行调用
Thread thread = new Thread(new MyRunnable());
thread.start();

System.out.println(mainName + "线程结束");
}
}

// 使用runnable开启线程
class MyRunnable implements Runnable
{
@Override
public void run()
{
// 获取调用线程的名字
String curName = Thread.currentThread().getName();
System.out.println(curName + "线程启动");
try
{
Thread.sleep(1000);
System.out.println(curName + "正在运行");
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println(curName + "线程结束");
}
}

// 使用Thread进行run方法的编写,开启线程
class MyThread extends Thread
{
// Thread.currentThread()调用这个方法的线程名称。
String s = Thread.currentThread().getName(); // 调用这个类的是main方法调用,run方法交由新线程操作

public MyThread()
{
System.out.println("Thread.currentThread().getname()=" + Thread.currentThread().getName());
// 线程没运行前初始化的名字.因为类的初始化跟线程运行无关,这里返回的是处世话的名字
System.out.println("This.getName=" + this.getName());
}

@Override
public void run()
{
// run方法交由新线程操作,所以调用的名称不是main
String strName = Thread.currentThread().getName();
System.out.println("this.getName " + this.getName());
// main
System.out.println("do u no s?" + s);

System.out.println(strName + " 开始数数");
for (int i = 0; i < 100; i++)
{
System.out.println(strName + " " + i);
}
System.out.println(strName + " 数数完毕");
}

// 返回设置的name
public String bName()
{
return this.getName();
}
}

 

      最应该注意的是给线程标注名称的时候,应该多使用Thread.currentThread().getname() 来获取线程名称。使用this.getName方法很容易出错。

     主要该注意的有如下两个:

     ① 获取名称不在run方法里面,永远返回名称为main。因为在构造线程类的时候,主线程main会进行代码的初始化。run方法由新创建起来的线程去调用,这样run方法里面的名称才能被正确获取到

     ② 在继承类的构造函数内,如上面MyThread的构造函数内调用getName()方法。该方法是由JDK 源码Thread类继承而来。new一个Thread实例时,系统会调用Thread

下的方法,代码如下

//Thread 类构造函数
public Thread() {
init(null, null, "Thread-" + nextThreadNum(), 0);
}

// 私有的init方法
private void init(ThreadGroup g, Runnable target, String name,long stackSize) {
init(g, target, name, stackSize, null);
}

//init 方法. 更详细内容见JDK源码Thread类363行
private void init(ThreadGroup g, Runnable target, String name,long stackSize, AccessControlContext acc)

      所以在构造器内的初始化的时候调用init方法给线程设置name字段。导致参数名称修改且以"Thread-"开头。初始化完毕后,如果用户设置过相关的线程名称,系统会把该字段进行修改,可以运用上述代码中的bName()函数进行验证。

  在使用Java多线程的时候,不要直接调用某个实例下的run方法,如果直接使用run方法。则直接调用,不会开辟新的线程进行功能操作。

       继承Thread和实现Runnable接口的区别:

  ① 继承Thread能够简单地实现线程开辟,但由于Java只支持单继承。继承Thread方法实现线程会有比较多的局限性

       ② 实现Runnable接口能够很好的将线程代码和业务逻辑代码分离,但代码结构相对复杂。一般都会使用实现Runnable方法创建线程。

2. Java多线程下常用方法

  ① join( )  多线程是有cpu分配执行时间片段,如果要控制线程操作流程,则需要使用到join方法。join方法会阻塞主线程,主线程会等待join方法的线程执行完毕再继续执行后续的操作。同时join方法可以添加时间,让主线程等待一段时间,如果时间过了。则不再等待继续执行。如:thread.join( ),当主线程编译到该代码时,主线程会停止下来等待thread线程执行完毕后,主线程再继续后续操作

  ② sleep( )  不释放锁资源,让调用的线程进行等待。如:thread.sleep( ) ,但代码执行到这一句时,thread会停止下来等待,直到被唤醒或者等到到sleep到的时间点后再进行后续操作。

  ③ wait( )和notify( )、notifyAll( )  。这三个方法主要处理线程同步数据时使用。数据同步,也就是几条线程在同时修改一份共享数据。如火车站5名销售员同时在售卖火车票,火车票是固定数量的前提下,如何实现这5名销售员同时知道火车票库存、同一路段火车票售卖、车票信息共享等。

      wait方法使当前线程暂停执行并释放对象锁标志。notify和notifyall方法会唤醒几个等待的线程,让线程重新获取锁代码以便后续进行。线程状态图表如下:

       

 

      关于使用锁以及线程状态内容,将在下一篇博文进行总结,上述内容若有问题,烦请指出纠正。 

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: