• 处理java多线程时线程安全问题


    多线程在自动化测试中用的不多,也就是说我们用单线程可以完成大部分的自动化测试脚本。

    主要有两个原因,首先是因为自动化测试首要考虑的是脚本的稳定性,所以一般会牺牲效率以保证脚本稳定,其次是由于局限于我们自动化测试工程师的开发功底。

    如果我们想提升测试效率,同时也提升自己的脚本开发水平,还有需要处理一些单线程处理不了的需求,那就可以考虑使用多线程了。

    我们在自动化测试中有哪些场景可以用到多线程呢?

    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;
        }
        
    }
  • 相关阅读:
    JavaScript异步编程1——Promise的初步使用
    Pailler
    ElGamal
    RSA
    密码基础
    博客园中:为文章添加版权保护
    DCT实现水印嵌入与提取(带攻击)
    量子:基于EPR块对的两步量子直接通信
    量子:拜占庭协议和测谎问题的量子协议的实验证明
    liunx:网络命令
  • 原文地址:https://www.cnblogs.com/clarke157/p/6817229.html
Copyright © 2020-2023  润新知