并发是指什么?
我的理解是指多个线程在共用cpu使用权,如果只有一个线程在运行,就不叫并发了,因为线程占用的资源(内存...),都是该线程控制的,不受其他因素影响。对于单个cpu来说,某一时刻只能有一个线程在cpu上在执行,而并发则是指多个行程在通过时间片使用cpu,也就是一个线程执行一小段时间。
并发会发生什么问题?
并发是指多个线程都在执行,那么,这些线程可能都在使用同一个变量,而如果一个线程修改了改变量,其他线程却不知道变量的新值就会出现问题(因为线程存的是该变量的副本,修改之后需要刷新到主存中,其他线程再从主存读取才能获取最新值)。所以说这就是Java的并发包的作用了。例如:synchronized,volatile,lock等。
并发的优势?
并发的好处很多,很重要的一点是可以提高资源利用率。例如:如果只有一个线程在执行,此时该线程需要读取磁盘文件,此时cpu只能闲置,这就极大的浪费了cpu资源。如果是多线程,当有线程进行io时,其他线程可以拥有cpu执行权,此时,CPU利用率就提高了。
并发的缺点?
设计更加复杂,对于复杂的设计,出现错误时很难排查错误,所以构建并发系统需要很好的并发知识。还有就是CPU每次切换到另外一个行程执行会消耗额外的时间和资源,也就是上下文切换。
代码:
package com.concurenny.chapter.one;
import java.util.concurrent.TimeUnit;
/**
* @author Mr lebron 创建者: 创建时间:2017年11月16日 下午2:20:02
* 模拟多个线程买票,票的数量会有问题。某些票可能会被卖多次
*/
public class DemoWithProblem {
public static void main(String[] args) {
// 开启3个线程买票.
new Thread(new Seller(new Ticket())).start();
new Thread(new Seller(new Ticket())).start();
new Thread(new Seller(new Ticket())).start();
}
/**
* 买票者
*
* @author Mr lebron
*
*/
static class Seller implements Runnable {
// 买的票数
private static int count;
// 总票数
private static int total = 100;
private Ticket ticket;
public Seller(Ticket ticket) {
this.ticket = ticket;
}
@Override
public void run() {
while (count < total) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
// ++count不是原子的,也就是这步可能没执行完,其他行程就开始执行了。
System.out.println(Thread.currentThread().getName() + ":卖的第" + (++count) + "张票");
}
}
}
/**
* 票
*
* @author Mr lebron
*/
static class Ticket {
}
}
package com.concurenny.chapter.one;
import java.util.concurrent.TimeUnit;
/**
* 创建者:Mr lebron 创建时间:2017年11月16日 下午2:42:28
* 线程安全的买票
*/
public class DemoWithOk {
public static void main(String[] args) {
// 开启3个线程买票.
new Thread(new Seller(new Ticket())).start();
new Thread(new Seller(new Ticket())).start();
new Thread(new Seller(new Ticket())).start();
}
/**
* 买票者
*
* @author Mr lebron
*
*/
static class Seller implements Runnable {
// 买的票数
private static int count;
// 总票数
private static int total = 100;
private Ticket ticket;
public Seller(Ticket ticket) {
this.ticket = ticket;
}
@Override
public void run() {
while (count < total) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 这里保证只能有一个线程执行,即保证了原子性和可见性
synchronized (Seller.class) {
// ++count不是原子的,也就是这步可能没执行完,其他行程就开始执行了。
System.out.println(Thread.currentThread().getName() + ":卖的第" + (++count) + "张票");
}
}
}
}
/**
* 票
*
* @author Mr lebron
*/
static class Ticket {
}
}