处理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;
    }
    
}

 

posted @ 2017-05-06 16:59  测试人生-  阅读(817)  评论(0编辑  收藏  举报