Java多线程String做锁的测试

尽量不要使用 synchronized(String a) 因为 JVM 中,字符串常量池具有缓存功能!

例子1

package ThreadTest;

//String常量做synchronized锁的问题
public class synchronizedDemo01 {
    //这两个字符串变量r1和r2实际上指向的是字符串常量池中的同一个数据
    private static String r1 = "a";
    private static String r2 = "a";
    public static void main(String[] args) {
        //定义线程1
        new Thread(()->{
            synchronized (r1){
                System.out.println("获取资源1");
                while (true){
                    try {
                        System.out.println(Thread.currentThread().getName()+"执行中");
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        },"线程1").start();

        //定义线程2
        new Thread(()->{
            synchronized (r2){
                System.out.println("获取资源2");
            }
        },"线程2").start();
    }
}

结果:

获取资源1
线程1执行中
线程1执行中
线程1执行中
线程1执行中
线程1执行中
线程1执行中
...

例子2

package ThreadTest;

//String常量做synchronized锁的问题
public class synchronizedDemo02 {
    //这两个字符串变量r1和r2实际上指向的是字符串常量池中的同一个数据
    private static String r1 = "a";
    private static String r2 = "b";
    public static void main(String[] args) {
        //定义线程1
        new Thread(()->{
            synchronized (r1){
                System.out.println("获取资源1");
                while (true){
                    try {
                        System.out.println(Thread.currentThread().getName()+"执行中");
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        },"线程1").start();

        //定义线程2
        new Thread(()->{
            synchronized (r2){
                System.out.println("获取资源2");
            }
            while (true){
                try {
                    System.out.println(Thread.currentThread().getName()+"执行中");
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        },"线程2").start();
    }
}

结果:

获取资源1
获取资源2
线程1执行中
线程2执行中
线程2执行中
线程1执行中
线程2执行中
线程1执行中
...

总结: 尽量不要使用 synchronized(String a) 因为 JVM 中,字符串常量池具有缓存功能; 如果非要用,则需要注意做好对象区分

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个Java线程调用HTTP的测试代码示例,使用了HttpClient库: ```java import org.apache.http.HttpEntity; import org.apache.http.client.methods.CloseableHttpResponse; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.impl.client.HttpClients; import org.apache.http.util.EntityUtils; import org.junit.Test; import java.io.IOException; import java.util.ArrayList; import java.util.List; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class MultiThreadHttpTest { private static final int THREAD_POOL_SIZE = 10; private static final String URL = "https://www.example.com"; @Test public void testMultiThreadHttpCall() throws InterruptedException { ExecutorService executorService = Executors.newFixedThreadPool(THREAD_POOL_SIZE); List<Runnable> tasks = new ArrayList<>(); for (int i = 0; i < THREAD_POOL_SIZE; i++) { tasks.add(new HttpTask()); } executorService.invokeAll(tasks); executorService.shutdown(); } private static class HttpTask implements Runnable { private final CloseableHttpClient httpClient; public HttpTask() { this.httpClient = HttpClients.createDefault(); } @Override public void run() { HttpGet httpGet = new HttpGet(URL); try (CloseableHttpResponse response = httpClient.execute(httpGet)) { HttpEntity entity = response.getEntity(); if (entity != null) { String result = EntityUtils.toString(entity); System.out.println("Thread " + Thread.currentThread().getName() + " got result: " + result); } } catch (IOException e) { e.printStackTrace(); } } } } ``` 在上面的示例中,我们使用了一个固定大小的线程池来执行多个HTTP调用任务。每个任务都是一个HTTP GET请求,使用 Apache HttpClient 库实现。在 `HttpTask` 类中,我们创建了一个 `CloseableHttpClient` 对象并执行 HTTP GET 请求。请求完成后,我们从响应实体中提取内容并将结果打印到控制台。 使用这个测试代码可以测试线程并发调用 HTTP 服务的情况,注意要根据实际情况修改 URL 和线程池大小。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值