1.程序、进程和线程的区别:
程序:一组指令的集合,指一段静态的代码;
进程:程序的一次执行过程;
线程:是一个程序内部的一条执行的路径。
2.实现多线程的方法:
①继承Thread类;
②实现Runnable接口;
哪种方式好:
Runnable好:避免了Java单继承的局限性
如果多个线程要操作同一份资源(或数据),更适合使用实现方式
3.Thread常用的一些方法:
start():启动线程并执行相应的run()方法;
run():子线程要执行的代码放入run()方法中;
currentThread():静态的,调取当前的线程;
getName():获取次线程的名字;
setName():设置此线程的名字;
yield():调用此方法的当前线程释放当前cup的执行权;
join():在A线程中调用B线程的join()方法,表示:当执行到此方法,A线程停止执行,直至B线程执行完毕
isAlive():判断当前线程是否还存活;
sleep():显示的让当前线程睡眠多少毫秒;
线程通信:wait()、notify()、notifyAll()
设置线程的优先级:setPriority() MAX_PRIORITY(10)、MIN_PRIORITY(1)、NORM_PRIORITY(5)
4.线程的安全问题
在窗口买票的实例中,会出现重票和错票的情况;
解决办法:必须让一个线程操作共享数据完毕后,其他线程菜有机会参与共享数据的操作
Java如何实现线程安全的-------线程的同步机制:
方式一:同步代码块:
synchronized(同步监视器){
//需要被同步的代码块(即操作共享数据的代码)
}
共享数据:多个线程共同操作的同一个数据(变量)
测试类:
加锁后,不会出现错票或者重票,值得注意的是,所有的线程必须是使用的同一把锁,否则仍会出现线程安全问题。
程序:一组指令的集合,指一段静态的代码;
进程:程序的一次执行过程;
线程:是一个程序内部的一条执行的路径。
2.实现多线程的方法:
①继承Thread类;
②实现Runnable接口;
哪种方式好:
Runnable好:避免了Java单继承的局限性
如果多个线程要操作同一份资源(或数据),更适合使用实现方式
3.Thread常用的一些方法:
start():启动线程并执行相应的run()方法;
run():子线程要执行的代码放入run()方法中;
currentThread():静态的,调取当前的线程;
getName():获取次线程的名字;
setName():设置此线程的名字;
yield():调用此方法的当前线程释放当前cup的执行权;
join():在A线程中调用B线程的join()方法,表示:当执行到此方法,A线程停止执行,直至B线程执行完毕
isAlive():判断当前线程是否还存活;
sleep():显示的让当前线程睡眠多少毫秒;
线程通信:wait()、notify()、notifyAll()
设置线程的优先级:setPriority() MAX_PRIORITY(10)、MIN_PRIORITY(1)、NORM_PRIORITY(5)
4.线程的安全问题
在窗口买票的实例中,会出现重票和错票的情况;
解决办法:必须让一个线程操作共享数据完毕后,其他线程菜有机会参与共享数据的操作
Java如何实现线程安全的-------线程的同步机制:
方式一:同步代码块:
synchronized(同步监视器){
//需要被同步的代码块(即操作共享数据的代码)
}
共享数据:多个线程共同操作的同一个数据(变量)
同步监视器:由任何一个类充当(所有线程必须共用同一个监视器)。
通过一个实例来理解一下:
public class TrainThread2 implements Runnable{
int ticket = 100; //共享数据
@Override
public void run() {
while (true){
synchronized (this){ //同步代码块
if (ticket>0){
try {
Thread.currentThread().sleep(10);//放大错误,出现错票,同步代码块后解决
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票号为:"+ticket--);
}else {
break;
}
}
}
}
}
测试类:
public class TrainThreadTest {
//实现Runnable接口方式实现售票
public static void main(String[] args) {
TrainThread2 t = new TrainThread2();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
Thread t3 = new Thread(t);
t1.setName("窗口1");
t2.setName("窗口2");
t3.setName("窗口3");
t1.start();
t2.start();
t3.start();
}
}
方式二:同步方法;
代码实例:
public class TrainThread3 implements Runnable{
int ticket = 100; //共享数据
@Override
public void run() {
while (true){
show();
if(ticket==0){
break;
}
}
}
public synchronized void show(){
if (ticket>0){
try {
Thread.currentThread().sleep(9);//放大错误,出现错票,同步后解决
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票号为:"+ticket--);
}
}
}
加锁后,不会出现错票或者重票,值得注意的是,所有的线程必须是使用的同一把锁,否则仍会出现线程安全问题。