Android中 activity怎么调用自定义view中重绘的方法?

android中,在activity里调用自定义view的postInvalidate()和invalidate()方法,自定义view中并没有重新绘制,当然在自定义view本类中调用是可以实现重新绘制的。求大神指导,是我调用方法不对? 还是有其他重新绘制方法可以供activity调用?

7个回答

应该是你没有把这个View放到布局中的缘故,你可以尝试在主布局中添加自定义的这个View,然后在onCreate方法中使用findViewById的方式获取对象,应该就正常了,用你发的代码 做我上面的修改是OK的

你看看你操作的view是不是其它的view,对象错了

One_Sun
One_Sun 你好,没有调错对象,我打Log都看了,我担心是线程间问题,也打印线程id了,是在一个线程执行也不行,而且线程也起了,也试着用handler进行处理了,还是不行。。
大约 3 年之前 回复

这个没重新绘制原因有很多,比如,android系统没有检测到你的控件发生变化,他就不会执行onDraw进行重绘。或者在内部的缓存中,发现缓存的大小并
没改变,也不会导致重绘。所以,我们可以强制清除系统缓存等数据,让view去重绘。
可调用如下方法进行试验:
forceLayout();
measure(0, 0);
requestLayout();
invalidate();

One_Sun
One_Sun 你好,我刚刚试了试你提的这几种方法,但是还是没有效果,目前这边情况是我在自定义view类中类构建时起一个线程,然后让线程停止一秒,改变标志位可以停止绘制。再然后让线程停止几秒,再重新调用postInvalidate()方法,也能实现重新绘制。只是在activity中向自定义view发送消息通知自定义view重新绘制时没有效果,不用handler发送消息,写方法调用也没有效果。你一会再看评论,我给你截几张图你看看,谢谢。
大约 3 年之前 回复

代码发出来 我帮你看看

poorkick
onlyloveyd 回复One_Sun: 应该是你没有把这个View放到布局中的缘故,你可以尝试在主布局中添加自定义的这个View,然后在onCreate方法中使用findViewById的方式获取对象,应该就正常了
大约 3 年之前 回复
One_Sun
One_Sun 你好,代码我也发评论了,你可以看一下,谢谢。
大约 3 年之前 回复
One_Sun
One_Sun 你好,我截了几张图,你看一下,在评论里,我感觉图片比较明了,如果不够详细,你告诉我我给你发代码。
大约 3 年之前 回复

activity中的OnCreate自定义View中onDraw方法里加的判断自定义view中接收activity中发送消息的handler

package com.example.windows.myapplication;

import android.app.Activity;
import android.os.Bundle;
import android.os.Message;
import android.util.Log;
import android.view.View;
import android.widget.Button;

public class MainActivity extends Activity {

private DynamicWaveTwo dynamicWaveTwo;
private Message message;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    dynamicWaveTwo = new DynamicWaveTwo(MainActivity.this,null);


    //在此处发送消息给自定义view,是我想开始时让自定义view先绘制一秒之后然后再停止绘制。
    //这个通知消息传给自定义view之后是可以停止绘制的。是ok的。
    message=dynamicWaveTwo.drawHandler.obtainMessage();
    message.what = 1 ;
    dynamicWaveTwo.drawHandler.sendMessageDelayed(message,1000);
    Log.i("TAG", "onCreate 已发送"+Thread.currentThread().getId());



    Button bt = (Button)findViewById(R.id.bt1);
    bt.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            //问题在这里,我在点击按钮之后发送一条消息,自定义view也能收到消息,标志位也修改成功。
            //但是在自定义view中调用重新绘制方法就是绘制不成功。
            message=dynamicWaveTwo.drawHandler.obtainMessage();
            message.what = 0 ;
            dynamicWaveTwo.drawHandler.sendMessage(message);
            Log.i("TAG", "已点击"+Thread.currentThread().getId());


            // Log.i("TAG", "已点击");
            //DynamicWaveTwo.a = 0;


            //new DynamicWaveTwo(MainActivity.this,null).continueDraw(0);

            //如何在activity里调用自定义view的重绘方法

          /*  new Thread(new Runnable() {
                @Override
                public void run() {
                    Log.i("TAG", "已点击");
                    new DynamicWaveTwo(MainActivity.this,null).continueDraw(0);
                }
            }).start();*/

/* new Thread(new Runnable() {
@Override
public void run() {
Message message=dynamicWaveTwo.drawHandler.obtainMessage();
message.what = 0 ;
dynamicWaveTwo.drawHandler.sendMessage(message);
//dynamicWaveTwo.drawHandler.sendMessageDelayed();
}
}).start();*/

        }
    });
}

}

全部代码,这是Activity中的

package com.example.windows.myapplication;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.DrawFilter;
import android.graphics.Paint;
import android.graphics.Paint.Style;
import android.graphics.PaintFlagsDrawFilter;
import android.os.Handler;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;

public class DynamicWaveTwo extends View {

// 波纹颜色
private static final int WAVE_PAINT_COLOR = 0x880000aa;
// y = Asin(wx+b)+h
private static final float STRETCH_FACTOR_A = 23;//影响振幅,数值越大,波纹振幅越深

private static final int OFFSET_Y = 0;
// 第一条水波移动速度
private static final int TRANSLATE_X_SPEED_ONE = 16;//原值7
// 第二条水波移动速度
private static final int TRANSLATE_X_SPEED_TWO = 8;//原值5
private float mCycleFactorW;

private int mTotalWidth, mTotalHeight;
private float[] mYPositions;
private float[] mResetOneYPositions;
private float[] mResetTwoYPositions;
private int mXOffsetSpeedOne;
private int mXOffsetSpeedTwo;
private int mXOneOffset;
private int mXTwoOffset;
private int new_STRETCH_FACTOR_A;

private Paint mWavePaint;
private Paint mWaveTowPaint;

private DrawFilter mDrawFilter;

public static int a = 0;



//自定义view类中
public  Handler drawHandler = new Handler() {

    public void handleMessage(android.os.Message msg) {

        int b = msg.what;

        //接收activity的消息,并修改标志位,实现控制。
        //停止绘制的判读。没有问题,可以停止绘制。
        if(b == 1){

            Log.i("TAG","handleMessage 1"+Thread.currentThread().getId());
            a = b;


            //问题在这里,修改完标志位,调用重新绘制方法,自定义view不能实现重绘。
            //Log 打印标志位也修改成功,就是不能重新绘制。
        }else if(b == 0){

            Log.i("TAG","handleMessage 0"+Thread.currentThread().getId());
            a = b;
            // continueDraw();
            // postInvalidate();
            // invalidate();

            //forceLayout();
            //measure(0, 0);
            //requestLayout();
            //invalidate();
            postInvalidate();
        }

    }

};


public DynamicWaveTwo(Context context, AttributeSet attrs) {
    super(context, attrs);
    // 将dp转化为px,用于控制不同分辨率上移动速度基本一致
    mXOffsetSpeedOne = UiUtils.dipToPx(context, TRANSLATE_X_SPEED_ONE);
    mXOffsetSpeedTwo = UiUtils.dipToPx(context, TRANSLATE_X_SPEED_TWO);

    // 将dp转化为px,用于控制不同分辨率上振幅高度基本一致
    new_STRETCH_FACTOR_A = UiUtils.dipToPx(context, (int)STRETCH_FACTOR_A);

    // 初始绘制波纹的画笔
    mWavePaint = new Paint();
    mWaveTowPaint = new Paint();
    // 去除画笔锯齿
    mWavePaint.setAntiAlias(true);
    mWaveTowPaint.setAntiAlias(true);
    // 设置风格为实线
    mWavePaint.setStyle(Style.FILL);
    mWaveTowPaint.setStyle(Style.FILL);
    // 设置画笔颜色
    //mWavePaint.setColor(WAVE_PAINT_COLOR);
    //mWavePaint.setColor(Color.RED);

    mWavePaint.setColor(Color.rgb(41, 182, 246));
    //设置透明度,注意一定要在设置颜色之后设置透明度,不然没有效果。
    mWavePaint.setAlpha(0x80);//其取值范围是0---255,数值越小,越透明,颜色上表现越淡。实际上当设成10以下就会有透明的效果了。
    //这里设置成0x80,0x代表转换为十六进制,设置为八十,证明透明度就是二十,设置为0,证明透明度就是百分百,设置的数值越小,越透明
    // 这个项目UI要求的透明度为%20,所以就设置为80。

    mWaveTowPaint.setColor(Color.rgb(235, 98, 55));
    mWaveTowPaint.setAlpha(0x90);

    mDrawFilter = new PaintFlagsDrawFilter(0, Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG);


  /*  new Thread(new Runnable() {
        @Override
        public void run() {

            try {
                Thread.sleep(1000);
                a = 1;
               *//* Thread.sleep(9000);
                a = 0;
                //postInvalidate();

                continueDraw();

//

            } catch (InterruptedException e) {
                e.printStackTrace();
            }

        }
    }).start();*/




}

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    // 从canvas层面去除绘制时锯齿
    canvas.setDrawFilter(mDrawFilter);
    resetPositonY();
    for (int i = 0; i < mTotalWidth; i++) {

        // 减400只是为了控制波纹绘制的y的在屏幕的位置,大家可以改成一个变量,然后动态改变这个变量,从而形成波纹上升下降效果
        // 绘制第一条水波纹     (demo想法)

        //实际上就是绘制两个不同的动画,然后进行移动,重叠处像是产生了波纹动画的效果,我们可以把第一个减400改为减100,第二个还是减400,运行就可以看出效果。
        //两个都减100,产生的效果就是整个动画高度没有那么高了。  (个人想法)
        canvas.drawLine(i, mTotalHeight - mResetOneYPositions[i] - 100, i,
                mTotalHeight,
                mWavePaint);

        // 绘制第二条水波纹
        canvas.drawLine(i, mTotalHeight - mResetTwoYPositions[i] - 100, i,
                mTotalHeight,
                mWaveTowPaint);
    }

    // 改变两条波纹的移动点
    mXOneOffset += mXOffsetSpeedOne;
    mXTwoOffset += mXOffsetSpeedTwo;

    // 如果已经移动到结尾处,则重头记录
    if (mXOneOffset >= mTotalWidth) {
        mXOneOffset = 0;
    }
    if (mXTwoOffset > mTotalWidth) {
        mXTwoOffset = 0;
    }

    // 引发view重绘,一般可以考虑延迟20-30ms重绘,空出时间片
    //postInvalidate();
    // postInvalidateDelayed(100); //延时30毫秒重新绘制


    //在onDraw方法中,为了方便activity中对绘制实现控制,
    //所以加的判断条件
    if(a == 0){
        postInvalidate();
    }



}

private void resetPositonY() {
    // mXOneOffset代表当前第一条水波纹要移动的距离
    int yOneInterval = mYPositions.length - mXOneOffset;
    // 使用System.arraycopy方式重新填充第一条波纹的数据
    System.arraycopy(mYPositions, mXOneOffset, mResetOneYPositions, 0, yOneInterval);
    System.arraycopy(mYPositions, 0, mResetOneYPositions, yOneInterval, mXOneOffset);

    int yTwoInterval = mYPositions.length - mXTwoOffset;
    System.arraycopy(mYPositions, mXTwoOffset, mResetTwoYPositions, 0,
            yTwoInterval);
    System.arraycopy(mYPositions, 0, mResetTwoYPositions, yTwoInterval, mXTwoOffset);

}

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    // 记录下view的宽高
    mTotalWidth = w;
    mTotalHeight = h;
    // 用于保存原始波纹的y值
    mYPositions = new float[mTotalWidth];
    // 用于保存波纹一的y值
    mResetOneYPositions = new float[mTotalWidth];
    // 用于保存波纹二的y值
    mResetTwoYPositions = new float[mTotalWidth];

    // 将周期定为view总宽度
    mCycleFactorW = (float) (2 * Math.PI / mTotalWidth);

    // 根据view总宽度得出所有对应的y值
    for (int i = 0; i < mTotalWidth; i++) {
        mYPositions[i] = (float) (new_STRETCH_FACTOR_A * Math.sin(mCycleFactorW * i) + OFFSET_Y);
    }
}

public void continueDraw(){

    //a = a1;
    //Log.i("TAG","continueDraw     " + String.valueOf(a).toString());
    Log.i("TAG", "continueDraw     " + String.valueOf(a).toString());
    //this.postInvalidate();
    //invalidate();
    //requestLayout();
    postInvalidate();
}

}

全部代码,这是自定义View。 谢谢。

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
其他相关推荐
Android中 activity调用自定义view中的重绘方法不起作用?
-
Android如何在activity中设置自定义view中的数据
-
Android 自定义view中可以改变activity的数据吗?或者说传值给Activity
-
Activity中调用 自定义LinearLayout布局文件方法
-
Android在自定义View控制Activity里控件
-
Android中能在一个自定义view中注册广播并发出广播吗
-
Android 动态切换标题栏里用户自定义view , view里button的点击事件不响应
-
Android 自定义View 参数
-
android 关于自定义view的问题
-
Android自定义控件在ListView中不显示
-
Android自定义view失败
-
是用一个Activity对应一个界面,还是一个个自定义的View对应一个界面
-
自定义组件:在使用自定义组件的Activity修改Paint画笔画的长方形颜色
-
Android 自定义View点击事件
-
如何从主activity打开我自定义的类
-
Android自定义Button的OnKeyDown方法没反应
-
在一个自定义的对话框中如何使用一个 list view?
-
Android Studio自定义布局中的Button不响应点击事件
-
android 中自定义GridView的使用
-
程序员实用工具网站
目录 1、搜索引擎 2、PPT 3、图片操作 4、文件共享 5、应届生招聘 6、程序员面试题库 7、办公、开发软件 8、高清图片、视频素材网站 9、项目开源 10、在线工具宝典大全 程序员开发需要具备良好的信息检索能力,为了备忘(收藏夹真是满了),将开发过程中常用的网站进行整理。 1、搜索引擎 1.1、秘迹搜索 一款无敌有良心、无敌安全的搜索引擎,不会收集私人信息,保...
我花了一夜用数据结构给女朋友写个H5走迷宫游戏
起因 又到深夜了,我按照以往在csdn和公众号写着数据结构!这占用了我大量的时间!我的超越妹妹严重缺乏陪伴而 怨气满满! 而女朋友时常埋怨,认为数据结构这么抽象难懂的东西没啥作用,常会问道:天天写这玩意,有啥作用。而我答道:能干事情多了,比如写个迷宫小游戏啥的! 当我码完字准备睡觉时:写不好别睡觉! 分析 如果用数据结构与算法造出东西来呢? ...
别再翻了,面试二叉树看这 11 个就够了~
写在前边 数据结构与算法: 不知道你有没有这种困惑,虽然刷了很多算法题,当我去面试的时候,面试官让你手写一个算法,可能你对此算法很熟悉,知道实现思路,但是总是不知道该在什么地方写,而且很多边界条件想不全面,一紧张,代码写的乱七八糟。如果遇到没有做过的算法题,思路也不知道从何寻找。面试吃了亏之后,我就慢慢的做出总结,开始分类的把数据结构所有的题型和解题思路每周刷题做出的系统性总结写在了 Github...
让程序员崩溃的瞬间(非程序员勿入)
今天给大家带来点快乐,程序员才能看懂。 来源:https://zhuanlan.zhihu.com/p/47066521 1. 公司实习生找 Bug 2.在调试时,将断点设置在错误的位置 3.当我有一个很棒的调试想法时 4.偶然间看到自己多年前写的代码 5.当我第一次启动我的单元测试时 ...
接私活必备的 10 个开源项目!
点击蓝色“GitHubDaily”关注我加个“星标”,每天下午 18:35,带你逛 GitHub!作者 | SevDot来源 | http://1t.click/VE8W...
GitHub开源的10个超棒后台管理面板
目录 1、AdminLTE 2、vue-Element-Admin 3、tabler 4、Gentelella 5、ng2-admin 6、ant-design-pro 7、blur-admin 8、iview-admin 9、material-dashboard 10、layui 项目开发中后台管理平台必不可少,但是从零搭建一套多样化后台管理并不容易,目前有许多开源、免费、...
Git 天天用 但是 Git 原理你了解吗?
Git 原理 做技术一定要知其然知其所以然,意思就是:知道它是这样的,更知道它为什么是这样的。我主要通过4块内容来简单介绍 Git 是原理是什么样的。这4块内容如下: Git 存储目录结构介绍 Git 是如何存储的 Git 的对象 Git引用 当然 Git 原理不仅仅包含这些,想要更深入了解请查看官方教程 https://git-scm.com/book/zh/v2/。 本文内容是我在 Git...
终于明白阿里百度这样的大公司,为什么面试经常拿ThreadLocal考验求职者了
点击上面↑「爱开发」关注我们每晚10点,捕获技术思考和创业资源洞察什么是ThreadLocalThreadLocal是一个本地线程副本变量工具类,各个线程都拥有一份线程私...
100 个网络基础知识普及,看完成半个网络高手
欢迎添加华为云小助手微信(微信号:HWCloud002或HWCloud003),输入关键字“加群”,加入华为云线上技术讨论群;输入关键字“最新活动”,获取华为云最新特惠促销。华为云诸多技术大咖、特惠活动等你来撩! 1)什么是链接? 链接是指两个设备之间的连接。它包括用于一个设备能够与另一个设备通信的电缆类型和协议。 2)OSI 参考模型的层次是什么? 有 7 个 OSI 层:物理...
为什么面向对象糟透了?
又是周末,编程语言“三巨头”Java, Lisp 和C语言在Hello World咖啡馆聚会。服务员送来咖啡的同时还带来了一张今天的报纸, 三人寒暄了几句, C语言翻开了...
分享靠写代码赚钱的一些门路
作者 mezod,译者 josephchang10如今,通过自己的代码去赚钱变得越来越简单,不过对很多人来说依然还是很难,因为他们不知道有哪些门路。今天给大家分享一个精彩...
对计算机专业来说学历真的重要吗?
我本科学校是渣渣二本,研究生学校是985,现在毕业五年,校招笔试、面试,社招面试参加了两年了,就我个人的经历来说下这个问题。 这篇文章很长,但绝对是精华,相信我,读完以后,你会知道学历不好的解决方案,记得帮我点赞哦。 先说结论,无论赞不赞同,它本质就是这样:对于技术类工作而言,学历五年以内非常重要,但有办法弥补。五年以后,不重要。 目录: 张雪峰讲述的事实 我看到的事实 为什么会这样 ...
世界上最好的学习法:费曼学习法
你是否曾幻想读一遍书就记住所有的内容?是否想学习完一项技能就马上达到巅峰水平?除非你是天才,不然这是不可能的。对于大多数的普通人来说,可以通过笨办法(死记硬背)来达到学习的目的,但效率低下。当然,也可以通过优秀的学习法来进行学习,比如今天讲的“费曼学习法”,可以将你的学习效率极大的提高。 费曼学习法是由加拿大物理学家费曼所发明的一种高效的学习方法,费曼本身是一个天才,13岁自学微积分,24岁加入曼...
学Linux到底学什么
来源:公众号【编程珠玑】 作者:守望先生 网站:https://www.yanbinghu.com/2019/09/25/14472.html 前言 ​我们常常听到很多人说要学学Linux或者被人告知说应该学学Linux,那么学Linux到底要学什么? 为什么要学Linux 在回答学什么之前,我们先看看为什么要学。首先我们需要认识到的是,很多服务器使用的是Linux系统,而作为服务器应...
深入理解C语言指针
一、指针的概念 要知道指针的概念,要先了解变量在内存中如何存储的。在存储时,内存被分为一块一块的。每一块都有一个特有的编号。而这个编号可以暂时理解为指针,就像酒店的门牌号一样。 1.1、变量和地址 先写一段简单的代码: void main(){ int x = 10, int y = 20; } 这段代码非常简单,就是两个变量的声明,分别赋值了 10、20。我们把内存当做一个酒店,而每个房间就...
C语言实现推箱子游戏
很早就想过做点小游戏了,但是一直没有机会动手。今天闲来无事,动起手来。过程还是蛮顺利的,代码也不是非常难。今天给大家分享一下~ 一、介绍 开发语言:C语言 开发工具:Dev-C++ 5.11 日期:2019年9月28日 作者:ZackSock 也不说太多多余的话了,先看一下效果图: 游戏中的人物、箱子、墙壁、球都是字符构成的。通过wasd键移动,规则的话就是推箱子的规则,也就不多说了。 二、代...
面试官:兄弟,说说基本类型和包装类型的区别吧
Java 的每个基本类型都对应了一个包装类型,比如说 int 的包装类型为 Integer,double 的包装类型为 Double。基本类型和包装类型的区别主要有以下 4 点。
8000字干货:那些很厉害的人是怎么构建知识体系的
本文约8000字,正常阅读需要15~20分钟。读完本文可以获得如下收益: 分辨知识和知识体系的差别 理解如何用八大问发现知识的连接点; 掌握致用类知识体系的构建方法; 能够应用甜蜜区模型找到特定领域来构建知识体系。 1. 知识体系?有必要吗? 小张准备通过跑步锻炼身体,可因为之前听说过小腿变粗、膝盖受伤、猝死等等与跑步有关的意外状况,有点担心自己会掉进各种坑里,就在微信上问朋友圈一直晒跑步...
Android完整知识体系路线(菜鸟-资深-大牛必进之路)
前言 移动研发火热不停,越来越多人开始学习Android 开发。但很多人感觉入门容易成长很难,对未来比较迷茫,不知道自己技能该怎么提升,到达下一阶段需要补充哪些内容。市面上也多是谈论知识图谱,缺少体系和成长节奏感,特此编写一份 Android 研发进阶之路,希望能对大家有所帮助。 由于篇幅过长,有些问题的答案并未放在文章当中,不过我都整理成了一个文档归纳好了,请阅读到文末领取~ Ja...
网易云音乐你喜欢吗?你自己也可以做一个
【公众号回复 “1024”,免费领取程序员赚钱实操经验】今天我章鱼猫给大家带来的这个开源项目,估计很多喜欢听音乐的朋友都会喜欢。就目前来讲,很多人对这款音乐 App 都抱...
前端开发必备网站推荐
本人是一个纯正的小白,在学习的过程中搜集了一些关于前端开发的网站,希望对大家能够有所帮助!授人以鱼不如授人以渔! 1.MDN开发者文档网址:https://developer.mozilla.org/zh-CN/ 2.菜鸟教程:https://www.runoob.com/html/html-tutorial.html 3.CSS教程:https://www.schoolw3c.com/html-...
相关热词 c# 去除空格 c#读取tiff未bmp c# 识别回车 c#生成条形码ean13 c#子控制器调用父控制器 c# 写大文件 c# 浏览pdf c#获取桌面图标的句柄 c# list反射 c# 句柄 进程