Android_三种动画(帧动画、View动画、属性动画)

一、动画分类:

Android的动画分为了三种, 分别是 帧动画、View动画、属性动画。

二、具体介绍:

1.帧动画:

简介:

  • 帧动画就是顺序播放一组预先定义好的图片,就是一张一张的图片连续播放。
  • 帧动画的实现有两种方式:
    layout的xml文件中实现和在MainActivity中用代码实现。

在layout的xml中实现:
1、在res/drawable目录下定义一个XML文件,根节点为系统提供的animation-list,然后放入定义更好的图片;
2、使用AnimationDrawable类播放第一步定义好的Drawable中的图片,形成动画效果;

属性 解释
oneshot 动画是否重复
Drawable 设置当前帧的图片
duration 每一帧的执行时间
AnimationDrawable 帧动画类
getDrawable() 获取帧动画的图片资源

注意:
1、android:oneshot=“false”:设置动画是否重复播放;
2、每个item都有Drawable和duration属性,Drawable表示我们要播放的图片;duration表示这张图播放的时间,时间以毫秒为单位;
在res的xml文件中:

<?xml version="1.0" encoding="utf-8"?>
<animation-list xmlns:android="http://schemas.android.com/apk/res/android"
    android:oneshot="false">
    <!--一个item对应一帧:一张图片
        drawable:图片
        duration:一张图片存活的时间-->
    <item android:drawable="@drawable/campfire01"
        android:duration="100"/>
    <item android:drawable="@drawable/campfire02"
        android:duration="100"/>
    <item android:drawable="@drawable/campfire03"
        android:duration="100"/>
    <item android:drawable="@drawable/campfire04"
        android:duration="100"/>
</animation-list>

在drawable的xml文件中:

<ImageView
        android:id="@+id/animation"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:scaleType="fitXY"
        android:src="@drawable/fire_animation"
        />

在MainActivity文件中,用AnimationDrawable播放动画:

boolean isStart = true;
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if(event.getAction() == MotionEvent.ACTION_DOWN){
            ImageView imageView = findViewById(R.id.animation);
//            通过控件获取动画
            AnimationDrawable animationDrawable = (AnimationDrawable)imageView.getDrawable();
//            启动动画
            if (isStart == true){
                animationDrawable.start();
            }else {
//           暂停动画
                animationDrawable.stop();
            }
            isStart = !isStart;
        }
        return super.onTouchEvent(event);
    }

图片资源:链接:https://pan.baidu.com/s/1L8q1oM_sipMIMGU5wDglMg
提取码:7qej
复制这段内容后打开百度网盘手机App,操作更方便哦

在MainActivity中用代码实现:
在layout的xml中创建一个控价:

<ImageView
        android:id="@+id/animation"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:scaleType="fitXY"
        android:src="@drawable/campfire01"
        />

在MainActivity中:

public class MainActivity extends AppCompatActivity {

    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

//       用代码创建动画
//        1.创建一个动画资源
        AnimationDrawable animationDrawable = new AnimationDrawable();
//        创建一个数组,添加每一帧的动画
        int[] resID = {R.drawable.campfire01,R.drawable.campfire02,R.drawable.campfire03,
                R.drawable.campfire04,R.drawable.campfire05,R.drawable.campfire06,
                R.drawable.campfire07,R.drawable.campfire08,R.drawable.campfire09,
                R.drawable.campfire10,R.drawable.campfire11,R.drawable.campfire12,
                R.drawable.campfire13,R.drawable.campfire14,R.drawable.campfire15,
                R.drawable.campfire16,R.drawable.campfire17};
        for(int id:resID){
            animationDrawable.addFrame(getResources().getDrawable(id,null), 100);
        }
//        使用资源
        ImageView imageView = findViewById(R.id.animation);
        imageView.setImageDrawable(animationDrawable);
//        启动动画
        animationDrawable.start();
    }
}

图片资源:链接:https://pan.baidu.com/s/1L8q1oM_sipMIMGU5wDglMg
提取码:7qej
复制这段内容后打开百度网盘手机App,操作更方便哦

2.view动画(补间动画):
  • view动画也称为补间动画,顾名思义,对于这种动画的操作,我们只需要确定好动画开始和结束的状态就行了,中间的部分系统会自动帮我们补全。
  • view动画主要运用于平移、缩放、旋转、透明度变化(渐变) 四种基本效果。
  • Animation属性详解:
属性 解释
setDuration(long) 置动画执行时间
setInterpolator(Interpolator) 设定插值器(指定的动画效果,譬如回弹等)
setFillAfter(boolean) 动画结束后是否保持结束时的状态
setFillBefore(boolean) 动画结束后是否回到开始时的状态(默认为true)
setRepeatMode(int) 重复类型有两个值,reverse表示倒序回放,restart表示从头播放
setRepeatCount 动画重复次数
setStartOffset(long) 调用start函数之后等待开始运行的时间,单位为毫秒
  • view的animation有四个子类:
    TranslateAnimation、ScaleAnimation、RotateAnimation、AlphaAnimation,在xml用translate、scale、rotate、alpha表示。
  • 在xml加多个动画,系统自动视为组合动画;在MainActivity中用代码实现的view动画,必须用AnimationSet实现组合动画。
  • view动画同样有xml文件实现和在MainActivity中用代码实现两种方法。
    xml实现:
    创建一个动画,在res/anim目录下创建一个xml文件:
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="2000"
    android:fillAfter="true"
    android:repeatMode="restart"
    >
    <translate android:fromXDelta="0"
        android:toXDelta="500" />

    <alpha android:fromAlpha="0"
        android:toAlpha="1.0"/>

    <scale android:fromXScale="0.1"
        android:toXScale="1"
        android:fromYScale="0.1"
        android:toYScale="1"
        android:pivotX="50%"/>

    <rotate android:fromDegrees="0"
        android:toDegrees="360"
        android:pivotX="50%"
        android:pivotY="50%"/>
</set>

在MainActivity中加载xml动画:

public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (event.getAction() == MotionEvent.ACTION_DOWN){
//          找到在drawable的xml文件中添加的控价            
            View view = findViewById(R.id.view_animation);
//            加载xml动画
            Animation translate = AnimationUtils.loadAnimation(this,R.anim.translate);
//            回弹效果
            translate.setInterpolator(new BounceInterpolator());
//            将动画作用到控件
            view.startAnimation(translate);
        }
        return super.onTouchEvent(event);
    }
}

MainActivity中代码实现:

public void test1(){
//      平移
        TranslateAnimation translateAnimation = new TranslateAnimation(0,100,0,100);
        translateAnimation.setDuration(1000);
        translateAnimation.setFillAfter(true);
//      缩放
        ScaleAnimation scaleAnimation = new ScaleAnimation(1,1.5f,1,1.5f);
//      旋转
        RotateAnimation rotateAnimation = new RotateAnimation(0,360);
//      改变透明度
        AlphaAnimation alphaAnimation = new AlphaAnimation(1,0.1f);
//      创建用于组合动画的数组
        AnimationSet animationSet = new AnimationSet(true);
//      添加动画
        animationSet.addAnimation(translateAnimation);
        animationSet.addAnimation(rotateAnimation);
        animationSet.addAnimation(alphaAnimation);
        animationSet.addAnimation(scaleAnimation);
//      设置动画数组的属性
        animationSet.setDuration(1000);
        animationSet.setRepeatCount(2);
//      开启动画
        imageView.startAnimation(animationSet);
    }
3.属性动画(Animator):
  • 属性动画执行过程中改变动画对象任意的属性值。一个属性动画会在一段特定长度的时间内改变一个属性的值。
  • Animator的子类:valueAnimator(记录动画过程中各个时刻动画的属性),valueAnimator的子类:objectAnimator(改变动画对象的属性、视觉效果,最常用)和TimeAnimator
  • 属性动画相关属性:
属性 解释
ObjectAnimator 对象动画执行类
ValueAnimator 值动画执行类,常配合AnimatorUpdateListener使用
PropertyValuesHolder 属性存储器,为两个执行类提供更新多个属性的功能
Keyframe 为 PropertyValuesHolder提供多个关键帧的操作值
AnimatorSet 一组动画的执行集合类:设置执行的先后顺序,时间等
AnimatorUpdateListener 动画更新监听
AnimatorListener 动画执行监听,在动画开始、重复、结束、取消时进行回调
AnimatorInflater 加载属性动画的xml文件
TypeEvaluator 类型估值,用于设置复杂的动画操作属性的值
TimeInterpolator 时间插值,用于控制动画执行过程

Java代码实现属性动画:

    public void test1(){
//        改变透明度
        ObjectAnimator alpha = ObjectAnimator.ofFloat(view,"alpha",1,0,1);
        alpha.setDuration(1000);
        alpha.start();
    }
    public void test2(){
//        旋转
        ObjectAnimator alpha = ObjectAnimator.ofFloat(view,"rotation",0,360);
        alpha.setDuration(1000);
        alpha.setRepeatCount(-1);
//        重复方式
        alpha.setRepeatMode(ValueAnimator.REVERSE);
        alpha.start();
    }
    public void test3(){
//        缩放
        ObjectAnimator alpha1 = ObjectAnimator.ofFloat(view,"scaleX",1,1.2f);
        ObjectAnimator alpha2 = ObjectAnimator.ofFloat(view,"scaleY",1,1.2f);
        alpha1.setDuration(1000);
//        alpha1.start();
        alpha2.setDuration(1000);
//        alpha2.start();

//        动画数组
        AnimatorSet set = new AnimatorSet();
        set.playTogether(alpha1,alpha2);
//        set.play(alpha1).after(alpha2);
        set.start();
    }
    public void test4(){
//        平移
//        在当前动画的基础上每次动画都增加一定的值
        ObjectAnimator alpha1 = ObjectAnimator.ofFloat(view,"translationX",view.getTranslationX() + 100);
        ObjectAnimator alpha2 = ObjectAnimator.ofFloat(view,"translationY",0,100);
        alpha1.setDuration(1000);
        alpha1.setRepeatCount(-1);
//        重复方式
        alpha1.setRepeatMode(ValueAnimator.REVERSE);
        alpha1.start();
        alpha2.setDuration(1000);
        alpha2.setRepeatCount(-1);
//        重复方式
        alpha2.setRepeatMode(ValueAnimator.REVERSE);
        alpha2.start();
    }
4、三种动画的优缺点比较:
  • 帧动画更多的依赖于完善的UI资源,他的原理就是将一张张单独的图片连贯的进行播放。所以使用帧动画时需要注意,不要使用过多特别大的图,容易导致内存不足。

  • 补间动画中,虽然使用translate将图片移动了,但是点击原来的位置,依旧可以发生点击事件,而属性动画却不是。因此我们可以确定,属性动画才是真正的实现了view的移动,补间动画对view的移动更像是在不同地方绘制了一个影子,实际的对象还是处于原来的地方。

  • 当我们把动画的repeatCount设置为无限循环时,如果在Activity退出时没有及时将动画停止,属性动画会导致Activity无法释放而导致内存泄漏,而补间动画却没有问题。因此,使用属性动画时切记在Activity执行 onStop 方法时顺便将动画停止。

  • xml 文件实现的补间动画,复用率极高。在Activity切换,窗口弹出时等情景中有着很好的效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容