线程的写法:
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
class ticket{
private int number=30;
Lock lock=new ReentrantLock();
public void sale(){
lock.lock();
try {
if(number>0) {
System.out.println(Thread.currentThread().getName()+"卖出第"+number--+"还剩下"+number);
Thread.sleep(10);
}
}
catch (Exception e){
e.printStackTrace();
}
finally {
lock.unlock();
}
}
}
public class Solution{
public static void main(String[] args) {
ticket ticket=new ticket();
new Thread(()->{for (int i=0;i<30;i++) ticket.sale();},"A").start();
new Thread(()->{for (int i=0;i<30;i++) ticket.sale();},"B").start();
new Thread(()->{for (int i=0;i<30;i++) ticket.sale();},"C").start();
new Thread(new Runnable() {
@Override
public void run() {
for (int i=0;i<30;i++) ticket.sale();
}
},"D").start();
}
}
这个一个买票的线程例子:
(1)new Thread(()->{for (int i=0;i<30;i++) ticket.sale();},"A").start();
(2)new Thread(new Runnable() {
@Override
public void run() {
for (int i=0;i<30;i++) ticket.sale();
}
},"D").start();
}
是线程的两种写法
线程的安全问题:
(1)
List<String> list= new ArrayList<>();
for (int i=1;i<=30;i++){
new Thread(()->{
list.add(UUID.randomUUID().toString().substring(0,8));
System.out.println(list);
System.out.println(Thread.currentThread().getName());
},String.valueOf(i)).start();
}
通过源码可知ArrayList线程是不安全的
容易报出以下的异常
(2)
List<String> list= new Vector<>();
for (int i=1;i<=30;i++){
new Thread(()->{
list.add(UUID.randomUUID().toString().substring(0,8));
System.out.println(list);
System.out.println(Thread.currentThread().getName());
},String.valueOf(i)).start();
}
通过源码可知Vector线程是安全的
(3)
List<String> list= new CopyOnWriteArrayList<>();
for (int i=1;i<=30;i++){
new Thread(()->{
list.add(UUID.randomUUID().toString().substring(0,8));
System.out.println(list);
System.out.println(Thread.currentThread().getName());
},String.valueOf(i)).start();
}
通过源码可知CopyOnWriteArrayList线程是安全的
上面的可以看出是因为加了synchronized才使得线程安全。
一个线程访问一个对象中的synchronized(this)同步代码块时,其他试图访问该对象的线程将被阻塞。