处理java多线程时线程安全问题 - ThreadLocal和Synchronized
多线程在自动化测试中用的不多,也就是说我们用单线程可以完成大部分的自动化测试脚本。
主要有两个原因,首先是因为自动化测试首要考虑的是脚本的稳定性,所以一般会牺牲效率以保证脚本稳定,其次是由于局限于我们自动化测试工程师的开发功底。
如果我们想提升测试效率,同时也提升自己的脚本开发水平,还有需要处理一些单线程处理不了的需求,那就可以考虑使用多线程了。
我们在自动化测试中有哪些场景可以用到多线程呢?
1. 处理大量的数据,比如同时从多个数据库读取数据。(可以使用单线程实现)
2. 在一台工作机上并行运行多个测试用例。(不能用单线程实现)
3. 模拟抢单,秒杀。(不能用单线程实现)
4. 进行一些简单的压力测试。(可以使用单线程实现)
既然我们有了使用多线程的动力,那看看什么是多线程,以及如何使用。
是多线程。线程与进程都有五个状态:创建、就绪、运行、阻塞、终止。通俗点将,多线程就是为了提高处理任务的效率,操作系统同时开启几个线程来并行执行。比如有100块砖需要搬到仓库,1个人搬是单线程,10个人同时搬就是多线程。
public class TestThread extends Thread{
//重写父类的run方法 @Override public void run() { for (int i = 0; i < 2; i++) { System.out.println(Thread.currentThread().getName()); } } public static void main(String[] args) {
//实例化三个线程 t1,t2,t3 TestThread t1 = new TestThread(); TestThread t2 = new TestThread(); TestThread t3 = new TestThread();
//设置线程名字 t1.setName("this is thread t1");
//启动线程 t1.start(); t2.setName("this is thread t2"); t2.start(); t3.setName("this is thread t3"); t3.start(); } }
线程安全问题:
如果同时开启的线程需要同时访问同一个变量,那就会造成混乱,最终出现错误的结果。
对于这种情况,我们可以使用ThreadLocal或Synchronized来处理。
当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。
Java中的synchronized是一个保留字,它依靠JVM的锁机制来实现临界区的函数或者变量在访问中的原子性
举例代码 - 连接DB时保证每次都是用同一个connection对象
public class ConnectionUtil { private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>(); private static Connection initConn = null; static { try { initConn = DriverManager.getConnection("url, name and password"); } catch (SQLException e) { e.printStackTrace(); } } public Connection getConn() { Connection c = tl.get(); if(null == c) tl.set(initConn); return tl.get(); } }
public class ConnectionUtil {
private static Connection initConn = null;
static {
try {
initConn = DriverManager.getConnection("url, name and password");
} catch (SQLException e) {
e.printStackTrace();
}
}
public static synchronized Connection getConn() {
return initConn;
}
}