Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 611|回复: 0

父组件中vuex方法更新state,子组件不能及时更新并渲染的解决方法

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-14 22:46:32 | 显示全部楼层 |阅读模式

    场景:

    我实际用到的是这样的,我父组件引用子组件related,父组件调用获取页面详情的方法,更新了state值related,子组件根据该related来渲染相关新闻内容,但是页面打开的时候总是先加载子组件,子组件在渲染的时候还没有获取到更新之后的related值,即使在子组件中watch该值的变化依然不能渲染出来子组件的相关新闻内容。

    我的解决办法:

    父组件像子组件传值,当父组件执行了获取页面详情的方法之后,state值related更新,然后传给子组件,子组件再进行渲染,可以正常获取到。

    父组件代码:

    <template>
      <div id="newsDetails">
        <mt-header title="详情">
          <router-link to="/" slot="left">
            <mt-button icon="back"></mt-button>
          </router-link>
        </mt-header>
        <div class="details clearfloat">
          <h1 class="titleFont">
            {{ title }}
          </h1>
          <div class="clearfloat sourceWrap">
            <ul class="sourceFont">
              <li v-if="(pubNews==true)">
                <span class="source">{{pubName}}</span>
              </li>
              <li>
                <span class="authorName">{{authorName}}</span>
                <span class="time">{{createAt|formatTime}}</span>
              </li>
            </ul>
            <span  v-if="(pubNews==true)" class='btnFollow' @click="follow">关注</span>
          </div>
          <div class="bodyFont clearfloat" id="bodyFont" ref="bodyFont" :class="{bodyHeight:contentStatus}">
            <div v-html="content"></div>
            <div class="editor" v-if="editorName">责任编辑:{{editorName}}</div>
          </div>
          <div class="contentToggle" @click="contentStatus=!contentStatus" v-if="contentStatus">阅读全文</div>
          <Related :related="related"></Related>
        <!--重点是这里  父组件向子组件传值-->
     </div> </div> </template>
    import { Toast } from 'mint-ui';
      import {mapState} from 'vuex'
      import Related from './Related.vue'
      import moment from 'moment';
      export default{
        name:"NewsDetails",
        components:{
          Related,
        },
        data(){
          return {
            id:this.$route.params.id,
            topicType:"news",
            contentStatus:false,
            curHeight:0,
            bodyHeight:5000,
            hotCommentScrollTop:0
          }
        },
        created(){
          this.id=this.$route.params.id;
          this.fetchData();
          moment.locale('zh-cn');
        },
        mounted(){
          setTimeout(()=>{
            this.contentToggle();
          },500)
        },
        watch: {
          '$route'(to,from){
            this.id=this.$route.params.id;
            this.fetchData();
          }
        },
        computed: {
          ...mapState({
            title: state => state.newsDetails.title,
            authorName: state => state.newsDetails.authorName,
            pubNews: state => state.newsDetails.pubNews,
            pubName: state => state.newsDetails.pubName,
            editorName: state => state.newsDetails.editorName,
            createAt: state => state.newsDetails.createAt,
            content: state => state.newsDetails.content,
            myFavourite: state => state.newsDetails.myFavourite,
            related: state => state.newsDetails.related,
          })
        },
        filters:{
          formatTime(time){
            return moment(time).fromNow();
          },
        },
        methods:{
          fetchData(){
            this.$store.dispatch('getDetails',this.id);
          },
          follow(){
            Toast('登录后进行关注');
            this.$router.push("/login");
          },
          contentToggle(){
            this.curHeight=this.$refs.bodyFont.offsetHeight;
            if(parseFloat(this.curHeight)>parseFloat(this.bodyHeight)){
              this.contentStatus=true;
            }else{
              this.contentStatus=false;
            }
    //        this.hotCommentScrollTop=this.$refs.hotComment.height;
            console.log(this.hotCommentScrollTop);
          },
        }
      }

    子组件related.vue

    <template>
        <div v-if="lists.length>0">
            <div class="tagTitle"><span>相关新闻</span></div>
            <div class="listItem" v-if="(item.type=='little')" v-for="(item,index) in lists" :to="{name:'details',params:{id:item.id}}" :key="index" @click="browserDetection()">
              <div class="listImg1">
                <!--<img :src="{lazy==loaded?item.thumb[0]:lazy==loading?'../../assets/images/little_loading.png':lazy==error?'../../assets/images/little_loading.png'}" alt="" v-lazy="item.thumb[0]">-->
                <img :src="item.thumb[0]" alt="" v-lazy="item.thumb[0]">
              </div>
              <div class='titleBox1'>
                <p class="listTitle">{{item.title}}</p>
                <div class="titleInfo">
                  <span class="openApp">打开唐人家</span>
                  <span v-if="item.top==true" class="toTop">置顶</span>
                  <!--<svg class="icon" aria-hidden="true">
                    <use xlink:href="#icon-dianzan"></use>
                  </svg>-->
                  <span class="like">阅读 {{item.read}}</span>
                  <span class="time">{{item.createAt|formatTime}}</span>
                </div>
            </div>
          </div>
        </div>
    </template>
    <script>
      import {mapActions, mapState, mapGetters} from 'vuex'
      import moment from 'moment'
      export default{
        data(){
          return {
            lists: [],
            id:this.$route.params.id,
          }
        },
        props:{
            related:Array   //重点是这里
        },
        created(){
          moment.locale('zh-cn');
        },
        /*computed: {
          ...mapState({
            related: state => state.newsDetails.related,
          })
        },*/
        filters:{
          formatTime(time){
            return moment(time).fromNow();
          },
        },
        methods:{
        },
        watch: {
          related (val) {
            this.lists = val;
          },
          '$route'(to,from){
            this.id=this.$route.params.id
          }
        }
      }
    </script>

    效果如图:

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-23 15:10 , Processed in 0.074665 second(s), 27 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表