爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      Java的wait()、notify()学习三部曲之一:JVM源码分析

      首页 知识中心 软件开发 文章详情页

      Java的wait()、notify()学习三部曲之一:JVM源码分析

      2024-04-22 08:34:59 阅读次数:45

      Java,多线程

      综述

      • Java的wait()、notify()学习三部曲由三篇文章组成,内容分别是: 一、通过阅读openjdk8的源码,分析和理解wait,notify在JVM中的具体执行过程; 二、修改JVM源码,编译构建成新的JVM,把我们感兴趣的参数打印出来,结合具体代码检查和我们的理解是否一致; 三、修改JVM源码,编译构建成新的JVM,按照我们的理解去修改关键参数,看能否达到预期效果;

      • 现在,咱们一起开始既漫长又深入的wait、notify学习之旅吧!

      wait()和notify()的通常用法

      • Java多线程开发中,我们常用到wait()和notify()方法来实现线程间的协作,简单的说步骤如下:
      1. A线程取得锁,执行wait(),释放锁;
      2. B线程取得锁,完成业务后执行notify(),再释放锁;
      3. B线程释放锁之后,A线程取得锁,继续执行wait()之后的代码;

      关于synchronize修饰的代码块

      • 通常,对于synchronize(lock){...}这样的代码块,编译后会生成monitorenter和monitorexit指令,线程执行到monitorenter指令时会尝试取得lock对应的monitor的所有权(CAS设置对象头),取得后即获取到锁,执行monitorexit指令时会释放monitor的所有权即释放锁;

      一个完整的demo

      • 为了深入学习wait()和notify(),先用完整的demo程序来模拟场景吧,以下是源码:
      public class NotifyDemo {
      
          private static void sleep(long sleepVal){
              try{
                  Thread.sleep(sleepVal);
              }catch(Exception e){
                  e.printStackTrace();
              }
          }
      
          private static void log(String desc){
              System.out.println(Thread.currentThread().getName() + " : " + desc);
          }
      
          Object lock = new Object();
      
          public void startThreadA(){
              new Thread(() -> {
                  synchronized (lock){
                      log("get lock");
                      startThreadB();
                      log("start wait");
                      try {
                          lock.wait();
                      }catch(InterruptedException e){
                          e.printStackTrace();
                      }
      
                      log("get lock after wait");
                      log("release lock");
                  }
              }, "thread-A").start();
          }
      
          public void startThreadB(){
              new Thread(()->{
                  synchronized (lock){
                      log("get lock");
                      startThreadC();
                      sleep(100);
                      log("start notify");
                      lock.notify();
                      log("release lock");
      
                  }
              },"thread-B").start();
          }
      
          public void startThreadC(){
              new Thread(() -> {
                  synchronized (lock){
                      log("get lock");
                      log("release lock");
                  }
              }, "thread-C").start();
          }
      
          public static void main(String[] args){
              new NotifyDemo().startThreadA();
          }
      }
      
      • 以上就是本次实战用到的demo,代码功能简述如下:
      1. 启动线程A,取得锁之后先启动线程B再执行wait()方法,释放锁并等待;
      2. 线程B启动之后会等待锁,A线程执行wait()之后,线程B取得锁,然后启动线程C,再执行notify唤醒线程A,最后退出synchronize代码块,释放锁;
      3. 线程C启动之后就一直在等待锁,这时候线程B还没有退出synchronize代码块,锁还在线程B手里;
      4. 线程A在线程B执行notify()之后就一直在等待锁,这时候线程B还没有退出synchronize代码块,锁还在线程B手里;
      5. 线程B退出synchronize代码块,释放锁之后,线程A和线程C竞争锁;
      • 把上面的代码在Openjdk8下面执行,反复执行多次,都得到以下结果:
      thread-A : get lock
      thread-A : start wait
      thread-B : get lock
      thread-C : c thread is start
      thread-B : start notify
      thread-B : release lock
      thread-A : after wait, acquire lock again
      thread-A : release lock
      thread-C : get lock
      thread-C : release lock
      
      • 针对以上结果,问题来了:

      • 第一个问题:

      • 将以上代码反复执行多次,结果都是B释放锁之后A会先得到锁,这又是为什么呢?C为何不能先拿到锁呢?

      • 第二个问题:

      • 线程C自开始就执行了monitorenter指令,它能得到锁是容易理解的,但是线程A呢?在wait()之后并没有没有monitorenter指令,那么它又是如何取得锁的呢?

      • wait()、notify()这些方法都是native方法,所以只有从JVM源码寻找答案了,本次阅读的是openjdk8的源码;

      带上问题去看JVM源码

      • 按照demo代码执行顺序,我整理了如下问题,带着这些问题去看JVM源码可以聚焦主线,不要被一些支线的次要的代码卡住(例如一些异常处理,监控和上报等):
      1. 线程A在wait()的时候做了什么?
      2. 线程C启动后,由于此时线程B持有锁,那么线程C此时在干啥?
      3. 线程B在notify()的时候做了什么?
      4. 线程B释放锁的时候做了什么?

      源码中最重要的注释信息

      • 在源码中有段注释堪称是整篇文章最重要的说明,请大家始终记住这段信息,处处都用得上:

      ObjectWaiter对象存在于WaitSet、EntryList、cxq等集合中,或者正在这些集合中移动

      • 原文如下: Java的wait()、notify()学习三部曲之一:JVM源码分析 请务必记住这三个集合:WaitSet、EntryList、cxq

      • 好了,接下来看源码分析问题吧:

      线程A在wait()的时候做了什么

      • 打开hotspot/src/share/vm/runtime/objectMonitor.cpp,看ObjectMonitor::wait方法: Java的wait()、notify()学习三部曲之一:JVM源码分析
      • 如上图所示,有两处代码值得我们注意:
      1. 绿框中将当前线程包装成ObjectWaiter对象,并且状态为TS_WAIT,这里对应的是jstack看到的线程状态WAITING;
      2. 红框中调用了AddWaiter方法,跟进去看下: Java的wait()、notify()学习三部曲之一:JVM源码分析
      • 这个ObjectWaiter对象被放入了_WaitSet中,_WaitSet是个环形双向链表(circular doubly linked list)

      • 回到ObjectMonitor::wait方法接着往下看,会发现关键代码如下图,当前线程通过park()方法开始挂起(suspend): Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 至此,我们把wait()方法要做的事情就理清了:

      1. 包装成ObjectWaiter对象,状态为TS_WAIT;
      2. ObjectWaiter对象被放入_WaitSet中;
      3. 当前线程挂起;

      线程B持有锁的时候线程C在干啥

      • 此时的线程C无法进入synchronized{}代码块,用jstack看应该是BLOCKED状态,如下图: Java的wait()、notify()学习三部曲之一:JVM源码分析
      • 我们看看monitorenter指令对应的源码吧,位置:openjdk/hotspot/src/share/vm/interpreter/interpreterRuntime.cpp
      IRT_ENTRY_NO_ASYNC(void, InterpreterRuntime::monitorenter(JavaThread* thread, BasicObjectLock* elem))
      #ifdef ASSERT
        thread->last_frame().interpreter_frame_verify_monitor(elem);
      #endif
        if (PrintBiasedLockingStatistics) {
          Atomic::inc(BiasedLocking::slow_path_entry_count_addr());
        }
        Handle h_obj(thread, elem->obj());
        assert(Universe::heap()->is_in_reserved_or_null(h_obj()),
               "must be NULL or an object");
        if (UseBiasedLocking) {
          // Retry fast entry if bias is revoked to avoid unnecessary inflation
          ObjectSynchronizer::fast_enter(h_obj, elem->lock(), true, CHECK);
        } else {
          ObjectSynchronizer::slow_enter(h_obj, elem->lock(), CHECK);
        }
        assert(Universe::heap()->is_in_reserved_or_null(elem->obj()),
               "must be NULL or an object");
      #ifdef ASSERT
        thread->last_frame().interpreter_frame_verify_monitor(elem);
      #endif
      IRT_END
      
      • 上面的代码有个if (UseBiasedLocking)判断,是判断是否使用偏向锁的,本例中的锁显然已经不属于当前线程C了,所以我们还是直接看slow_enter(h_obj, elem->lock(), CHECK)方法吧;

      • 打开openjdk/hotspot/src/share/vm/runtime/synchronizer.cpp:

      void ObjectSynchronizer::slow_enter(Handle obj, BasicLock* lock, TRAPS) {
        markOop mark = obj->mark();
        assert(!mark->has_bias_pattern(), "should not see bias pattern here");
      
        //是否处于无锁状态
        if (mark->is_neutral()) {
          // Anticipate successful CAS -- the ST of the displaced mark must
          // be visible <= the ST performed by the CAS.
          lock->set_displaced_header(mark);
          //无锁状态就去竞争锁
          if (mark == (markOop) Atomic::cmpxchg_ptr(lock, obj()->mark_addr(), mark)) {
            TEVENT (slow_enter: release stacklock) ;
            return ;
          }
          // Fall through to inflate() ...
        } else
        if (mark->has_locker() && THREAD->is_lock_owned((address)mark->locker())) {
          //如果处于有锁状态,就检查是不是当前线程持有锁,如果是当前线程持有的,就return,然后就能执行同步代码块中的代码了
          assert(lock != mark->locker(), "must not re-lock the same lock");
          assert(lock != (BasicLock*)obj->mark(), "don't relock with same BasicLock");
          lock->set_displaced_header(NULL);
          return;
        }
      
      #if 0
        // The following optimization isn't particularly useful.
        if (mark->has_monitor() && mark->monitor()->is_entered(THREAD)) {
          lock->set_displaced_header (NULL) ;
          return ;
        }
      #endif
      
        // The object header will never be displaced to this lock,
        // so it does not matter what the value is, except that it
        // must be non-zero to avoid looking like a re-entrant lock,
        // and must not look locked either.
        lock->set_displaced_header(markOopDesc::unused_mark());
        //锁膨胀
        ObjectSynchronizer::inflate(THREAD, obj())->enter(THREAD);
      }
      
      • 线程C在上面代码中的执行顺序如下:
      1. 判断是否是无锁状态,如果是就通过Atomic::cmpxchg_ptr去竞争锁;
      2. 不是无锁状态,就检查当前锁是否是线程C持有;
      3. 不是线程C持有,调用inflate方法开始锁膨胀;
      • ObjectSynchronizer::inflate(THREAD, obj())->enter(THREAD);

      • 来看看锁膨胀的源码: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 如上图,锁膨胀的代码太长,我们这里只看关键代码吧:

      • 红框中,如果当前状态已经是重量级锁,就通过mark->monitor()方法取得ObjectMonitor指针再返回;

      • 绿框中,如果还不是重量级锁,就检查是否处于膨胀中状态(其他线程正在膨胀中),如果是膨胀中,就调用ReadStableMark方法进行等待,ReadStableMark方法执行完毕后再通过continue继续检查,ReadStableMark方法中还会调用os::NakedYield()释放CPU资源;

      • 如果红框和绿框的条件都没有命中,目前已经是轻量级锁了(不是重量级锁并且不处于锁膨胀状态),可以开始膨胀了,如下图: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 简单来说,锁膨胀就是通过CAS将监视器对象OjectMonitor的状态设置为INFLATING,如果CAS失败,就在此循环,再走前一副图中的的红框和绿框中的判断,如果CAS设置成功,会继续设置ObjectMonitor中的header、owner等字段,然后inflate方法返回监视器对象OjectMonitor;

      • 看看之前slow_enter方法中,调用inflate方法的代码如下:

      ObjectSynchronizer::inflate(THREAD, obj())->enter(THREAD);
      
      • 所以inflate方法返回监视器对象OjectMonitor之后,会立刻执行OjectMonitor的enter方法,这个方法中开始竞争锁了,方法在openjdk/hotspot/src/share/vm/runtime/objectMonitor.cpp文件中: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 如上图,红框中表示OjectMonitor的enter方法一进来就通过CAS将OjectMonitor的_owner设置为当前线程,绿框中表示设置成功的逻辑,第一个if表示重入锁的逻辑,第二个if表示第一次设置_owner成功,都意味着竞争锁成功,而我们的线程C显然是竞争失败的,会进入下图中的无线循环,反复调用EnterI方法: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 进入EnterI方法看看: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 如上图,首先构造一个ObjectWaiter对象node,后面的for(;;)代码块中来是一段非常巧妙的代码,同一时刻可能有多个线程都竞争锁失败走进这个EnterI方法,所以在这个for循环中,用CAS将_cxq地址放入node的_next,也就是把node放到_cxq队列的首位,如果CAS失败,就表示其他线程把node放入到_cxq的首位了,所以通过for循环再放一次,只要成功,此node就一定在最新的_cxq队列的首位。

      • 接下来的代码又是一个无限循环,如下图: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 从上图可以看出,进入循环后先调用TryLock方法竞争一次锁,如果成功了就退出循环,否则就调用Self->_ParkEvent->park方法使线程挂起,这里有自旋锁的逻辑,也就是park方法带了时间参数,就会在挂起一段时间后自动唤醒,如果不是自旋的条件,就一直挂起等待被其他条件唤醒,线程被唤醒后又会执行TryLock方法竞争一次锁,竞争不到继续这个for循环;

      • 到这里我们已经把线程C在BLOCK的时候的逻辑理清楚了,小结如下:

      1. 偏向锁逻辑,未命中;
      2. 如果是无锁状态,就通过CAS去竞争锁,此处由于锁已经被线程B持有,所以不是无锁状态;
      3. 不是无锁状态,而且锁不是线程C持有,执行锁膨胀,构造OjectMonitor对象;
      4. 竞争锁,竞争失败就将线程加入_cxq队列的首位;
      5. 开始无限循环,竞争锁成功就退出循环,竞争失败线程挂起,等待被唤醒后继续竞争;

      线程B在notify()的时候做了什么

      • 接下来该线程B执行notify了,代码是objectMonitor.cpp的ObjectMonitor::notify方法: Java的wait()、notify()学习三部曲之一:JVM源码分析
      • 如上图所示,首先是Policy的赋值,其次是调用DequeueWaiter()方法将_WaitSet队列的第一个值取出并返回,还记得_WaitSet么?所有wait的线程都被包装成ObjectWaiter对象然后放进来了;
      • 接下来对ObjectWaiter对象的处理方式,根据Policy的不同而不同: Policy == 0:放入_EntryList队列的排头位置; Policy == 1:放入_EntryList队列的末尾位置; Policy == 2:_EntryList队列为空就放入_EntryList,否则放入_cxq队列的排头位置;

      Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 如上图所示,请注意把ObjectWaiter的地址写到_cxq变量的时候要用CAS操作,因为此时可能有其他线程正在竞争锁,竞争失败的时候会将自己包装成ObjectWaiter对象加入到_cxq中;

      • 这里的代码有一处疑问,期待着读着您的指教:如果_EntryList为空,就把ObjectWaiter放入ObjectWaiter中,为什么要这样做呢?

      • Policy == 3:放入_cxq队列中,末尾位置;更新_cxq变量的值的时候,同样要通过CAS注意并发问题;

      • 这里有一段很巧妙的代码,现将_cxq保存在Tail中,正常情况下将ObjectWaiter赋值给Tail->_next就可以了,但是此时有可能其他线程正在_cxq的尾部追加数据了,所以此时Tail对象对应的记录就不是最后一条了,那么它的_next就非空了,一旦发生这种情况,就执行Tail = Tail->_next,这样就获得了最新的_cxq的尾部数据,如下图所示: Java的wait()、notify()学习三部曲之一:JVM源码分析

      • Policy等于其他值,立即唤醒ObjectWaiter对应的线程;

      • 小结一下,线程B执行notify时候做的事情:

      1. 执行过wait的线程都在队列_WaitSet中,此处从_WaitSet中取出第一个;
      2. 根据Policy的不同,将这个线程放入_EntryList或者_cxq队列中的起始或末尾位置;

      线程B释放锁的时候做了什么

      • 接下来到了揭开问题的关键了,我们来看objectMonitor.cpp的ObjectMonitor::exit方法; Java的wait()、notify()学习三部曲之一:JVM源码分析

      • 如上图,方法一进来先做一些合法性判断,接下来如红框所示,是偏向锁逻辑,偏向次数减一后直接返回,显然线程B在此处不会返回,而是继续往下执行;

      • 根据QMode的不同,有不同的处理方式:

      1. QMode = 2,并且_cxq非空:取_cxq队列排头位置的ObjectWaiter对象,调用ExitEpilog方法,该方法会唤醒ObjectWaiter对象的线程,此处会立即返回,后面的代码不会执行了;
      2. QMode = 3,并且_cxq非空:把_cxq队列首元素放入_EntryList的尾部;
      3. QMode = 4,并且_cxq非空:把_cxq队列首元素放入_EntryList的头部;
      4. QMode = 0,不做什么,继续往下看;
      • 只有QMode=2的时候会提前返回,等于0、3、4的时候都会继续往下执行:

      • 如果_EntryList的首元素非空,就取出来调用ExitEpilog方法,该方法会唤醒ObjectWaiter对象的线程,然后立即返回;

      • 如果_EntryList的首元素为空,就取_cxq的首元素,放入_EntryList,然后再从_EntryList中取出来执行ExitEpilog方法,然后立即返回;

      • 以上操作,均是执行过ExitEpilog方法然后立即返回,如果取出的元素为空,就执行循环继续取;

      • 小结一下,线程B释放了锁之后,执行的操作如下:

      1. 偏向锁逻辑,此处未命中;
      2. 根据QMode的不同,将ObjectWaiter从_cxq或者_EntryList中取出后唤醒;
      3. 唤醒的元素会继续执行挂起前的代码,按照我们之前的分析,线程唤醒后,就会通过CAS去竞争锁,此时由于线程B已经释放了锁,那么此时应该能竞争成功;
      • 到了现在已经将之前的几个问题搞清了,汇总起来看看:
      1. 线程A在wait() 后被加入了_WaitSet队列中;
      2. 线程C被线程B启动后竞争锁失败,被加入到_cxq队列的首位;
      3. 线程B在notify()时,从_WaitSet中取出第一个,根据Policy的不同,将这个线程放入_EntryList或者_cxq队列中的起始或末尾位置;
      4. 根据QMode的不同,将ObjectWaiter从_cxq或者_EntryList中取出后唤醒;;
      • 所以,最初的问题已经清楚了,wait()的线程被唤醒后,会进入一个队列,然后JVM会根据Policy和QMode的不同对队列中的ObjectWaiter做不同的处理,被选中的ObjectWaiter会被唤醒,去竞争锁;

      • 至此,源码分析已结束,但是因为我们不知道Policy和QMode参数到底是多少,所以还不能对之前的问题有个明确的结果,这些还是留在下一章来解答吧,下一章里我们去修改JVM源码,把参数都打印出来;

      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://blog.51cto.com/zq2599/5241146,作者:程序员欣宸,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:子域名收集 -- Findomain

      下一篇:java集合【6】——— iterable接口

      相关文章

      2025-05-14 10:03:13

      arm架构下JAVA开发

      ARM(Advanced RISC Machine)是一种基于精简指令集计算(RISC)设计的处理器架构。它以高效、节能著称,因此广泛应用 于从智能手机到物联网设备的各个领域。

      2025-05-14 10:03:13
      Java , JVM , 嵌入式 , 架构 , 设备
      2025-05-14 10:02:58

      Java实现根据概率中奖率怎么算

      要实现根据概率计算中奖率的功能,可以使用 Java 编程语言编写一个简单的程序。

      2025-05-14 10:02:58
      Java , 概率 , 模拟 , 程序
      2025-05-14 09:51:21

      Java 代码本地设置Hadoop用户名密码

      在Hadoop环境中,通常使用Kerberos进行身份验证。但在一些开发或测试环境中,我们可能需要在本地代码中设置用户名和密码来模拟或进行简单的测试。

      2025-05-14 09:51:21
      Hadoop , Java , 代码 , 使用 , 用户名 , 认证
      2025-05-14 09:51:21

      java 判断map为null或者空

      java 判断map为null或者空

      2025-05-14 09:51:21
      Java , Map , null , 方法 , 是否 , 检查 , 示例
      2025-05-14 09:51:15

      java怎么对线程池做监控

      对Java线程池进行监控是确保系统性能和稳定性的重要部分。监控线程池可以帮助我们了解线程池的状态,如当前活跃线程数、任务队列长度、已完成任务数等。

      2025-05-14 09:51:15
      Java , 方法 , 监控 , 示例 , 线程 , 队列
      2025-05-13 09:53:23

      java动态获取实体类的字段

      在Java中,我们可以使用反射(Reflection)API来动态地获取实体类的字段。

      2025-05-13 09:53:23
      API , Java , 使用 , 字段 , 实体类 , 方法 , 获取
      2025-05-13 09:53:23

      Java静态变量在静态方法内部无法改变值

      在Java中,静态变量(也称为类变量)属于类本身,而不是类的任何特定实例。它们可以在没有创建类的实例的情况下访问和修改。如果我们发现在静态方法内部无法改变静态变量的值,这通常是因为我们的代码中有一些逻辑错误或误解。

      2025-05-13 09:53:23
      Java , 变量 , 实例 , 类名 , 访问 , 静态 , 静态方法
      2025-05-12 10:19:12

      springboot学习(2)

      springboot学习(2)

      2025-05-12 10:19:12
      Java , main , springboot , web , 启动 , 方法 , 浏览器
      2025-05-08 09:03:21

      基于spring+jsp+mysql实现的Java web论坛系统【源码+数据库+指导运行】

      本项目是一套基于spring+jsp+mysql实现的Java web论坛系统,主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。

      2025-05-08 09:03:21
      Java , jsp , spring , 功能 , 源码
      2025-05-07 09:08:42

      Java中的异常体系

      Java中的异常体系

      2025-05-07 09:08:42
      Exception , Java , 异常 , 程序
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5245638

      查看更多

      最新文章

      arm架构下JAVA开发

      2025-05-14 10:03:13

      Java实现根据概率中奖率怎么算

      2025-05-14 10:02:58

      Java 代码本地设置Hadoop用户名密码

      2025-05-14 09:51:21

      java 判断map为null或者空

      2025-05-14 09:51:21

      java怎么对线程池做监控

      2025-05-14 09:51:15

      Java静态变量在静态方法内部无法改变值

      2025-05-13 09:53:23

      查看更多

      热门文章

      Java线程同步synchronized wait notifyAll

      2023-04-18 14:15:05

      JAVA多线程学习笔记

      2023-05-11 06:05:48

      Java/Android Annotation注解/注入(二)

      2023-04-13 09:37:00

      Android/Java判断字符串String是否为float浮点数或double类型

      2023-04-17 09:39:54

      Java小数点数字和百分号数字之间的转换

      2023-04-13 09:48:57

      Java的自带注解Annotation(一)

      2023-05-10 06:02:06

      查看更多

      热门标签

      java Java python 编程开发 代码 开发语言 算法 线程 Python html 数组 C++ 元素 javascript c++
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      Java设计模式:中介模式

      论JavaEE和Java的区别

      Java八种基本数据类型&字符串&运算符(图文详解)

      Java学习之this关键字(封装)

      如何在Java中实现自动化测试与CI/CD

      深入理解Java中的堆内存与栈内存分配

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号