玩转Android之加速度传感器的使用,模仿微信摇一摇

Android系统带的传感器有很多种,最常见的莫过于微信的摇一摇了,那么今天我们就来看看Anroid中传感器的使用,做一个类似于微信摇一摇的效果。

OK ,废话不多说,我们就先来看看效果图吧:

当我摇动手机的时候这里的动画效果基本和微信上的动画效果一致,这里请大家自行脑补微信摇一摇画面。

那我们就动手吧。

1.布局文件

好,那我们先来看看布局文件吧,在布局文件的正中央是一个花的图片,上图大家看到的手机图片实际上是两张图片拼接在一起,将花的那张图片遮住了,当摇一摇的时候,这两张图片分别向上或者向下移动,然后花的图片就可以显示出来。OK,基本原理就是这样,我们来看看代码:

[java] view plaincopy print?
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout  
  3.     xmlns:android="http://schemas.android.com/apk/res/android"  
  4.     xmlns:tools="http://schemas.android.com/tools"  
  5.     android:layout_width="match_parent"  
  6.     android:layout_height="match_parent"  
  7.     android:background="#1f1f1f">  
  8.   
  9.     <ImageView  
  10.         android:layout_width="wrap_content"  
  11.         android:layout_height="wrap_content"  
  12.         android:src="@drawable/flower"/>  
  13.   
  14.     <LinearLayout  
  15.         android:layout_width="wrap_content"  
  16.         android:layout_height="wrap_content"  
  17.         android:layout_centerInParent="true"  
  18.         android:orientation="vertical">  
  19.   
  20.         <ImageView  
  21.             android:id="@+id/up"  
  22.             android:layout_width="wrap_content"  
  23.             android:layout_height="wrap_content"  
  24.             android:src="@drawable/up"/>  
  25.   
  26.         <ImageView  
  27.             android:id="@+id/down"  
  28.             android:layout_width="wrap_content"  
  29.             android:layout_height="wrap_content"  
  30.             android:src="@drawable/down"/>  
  31.     </LinearLayout>  
  32. </RelativeLayout>  



2.传感器监听手机晃动

既然要监听手机加速度的变化,那我首先需要获取系统的传感器:

[java] view plaincopy print?
  1. //获取到一个传感器管理器  
  2.         sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);  
  3.         //获得一个加速度传感器  
  4.         Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);  

这两行代码首先是获取一个传感器管理器,然后获取加速度传感器,因为关于传感器的API 有很多,这里你需要指明自己要获取的是哪一个传感器。拿到传感器之后,需要注册监听,如下:
[java] view plaincopy print?
  1. sensorManager.registerListener(listener, sensor, SensorManager.SENSOR_DELAY_GAME);  

注册传感器的监听一共需要三个参数,第一个是监听器,第二个是加速度传感器,第三个是传感器的灵敏度,传感器的灵敏度一共分为四级,如下,从上往下灵敏度依次降低:
    1. SENSOR_DELAY_FASTEST
   2. SENSOR_DELAY_GAME
   3. SENSOR_DELAY_UI
   4. SENSOR_DELAY_NORMAL

OK ,注册完之后,我们还是来看看这个监听器是什么吧:

[java] view plaincopy print?
  1. private SensorEventListener listener = new SensorEventListener() {  
  2.         //当手机的加速度发生变化时调用  
  3.         @Override  
  4.         public void onSensorChanged(SensorEvent event) {  
  5.             //获取手机在不同方向上加速度的变化  
  6.             float valuesX = Math.abs(event.values[0]);  
  7.             float valuesY = Math.abs(event.values[1]);  
  8.             float valuesZ = Math.abs(event.values[2]);  
  9.   
  10.             if (valuesX > 17 || valuesY > 17 || valuesZ > 17) {  
  11.                 startAnimation();  
  12.                 playSound();  
  13.             }  
  14.         }  
  15.   
  16.         @Override  
  17.         public void onAccuracyChanged(Sensor sensor, int accuracy) {  
  18.   
  19.         }  
  20.     };  

这个listener中一共就两个方法,一个是当手机的加速度发生改变的时候调用,还有一个是当传感器的灵敏度发生改变的时候调用,当手机的加速度发生改变的时候,我们可以获取到手机在X 、Y、Z 三个维度上的变化值,拿到这个值之后,我们只需要进行简单的比较即可,如果有任意一个方向的值大于17,则认为有人在晃动手机,这个时候开启动画和声音的播放。

3.开启动画和声音

动画实际上就是两个平移动画,我们来看看:

[java] view plaincopy print?
  1. private void startAnimation() {  
  2.     //如果两次晃动手机的时间小于1秒,则只执行一次动画  
  3.     long currentTimeMillis = System.currentTimeMillis();  
  4.     if (currentTimeMillis - lastTime < 1000) {  
  5.         return;  
  6.     }  
  7.     lastTime = currentTimeMillis;  
  8.     AnimationSet upSet = new AnimationSet(true);  
  9.     TranslateAnimation upUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  10.             0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);  
  11.     upUp.setDuration(1000);  
  12.     TranslateAnimation upDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  13.             0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);  
  14.     upDown.setDuration(1000);  
  15.     upDown.setStartOffset(1000);  
  16.     upSet.addAnimation(upUp);  
  17.     upSet.addAnimation(upDown);  
  18.     up.startAnimation(upSet);  
  19.     AnimationSet downSet = new AnimationSet(true);  
  20.     TranslateAnimation downUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  21.             0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);  
  22.     downUp.setDuration(1000);  
  23.     TranslateAnimation downDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  24.             0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);  
  25.     downDown.setDuration(1000);  
  26.     downDown.setStartOffset(1000);  
  27.     downSet.addAnimation(downUp);  
  28.     downSet.addAnimation(downDown);  
  29.     down.startAnimation(downSet);  
  30. }  

至于声音,由于我这里只是播放比较短小的音效而已,所以并没有必要使用MediaPlayer,我可以通过一个声音池来解决这个问题,代码如下:
[java] view plaincopy print?
  1. /** 
  2.  * 初始化声音池 
  3.  */  
  4. private void initSoundPool() {  
  5.     if (Build.VERSION.SDK_INT > 20) {  
  6.         SoundPool.Builder builder = new SoundPool.Builder();  
  7.         //1.最大并发流数  
  8.         builder.setMaxStreams(3);  
  9.         AudioAttributes.Builder aaBuilder = new AudioAttributes.Builder();  
  10.         aaBuilder.setLegacyStreamType(AudioManager.STREAM_MUSIC);  
  11.         builder.setAudioAttributes(aaBuilder.build());  
  12.         soundPool = builder.build();  
  13.     } else {  
  14.         soundPool = new SoundPool(3, AudioManager.STREAM_MUSIC, 0);  
  15.     }  
  16.     //加载一个音频文件  
  17.     sound1 = soundPool.load(this, R.raw.awe, 1);  
  18. }  

在创建一个声音池的时候我采取了两种不同的方案,如果系统的版本大于20,则是用第一种方式获取声音池,否则使用第二种方式获取声音池。获取声音池之后,再通过声音池加载一个音频文件。加载完成之后,我就可以对这个音频文件进行播放了,如下:
[java] view plaincopy print?
  1. //1.声音的id  
  2. //2.3.表示左右声道的音量  
  3. //4.优先级  
  4. //5.是否循环  
  5. //6.声音播放速率  
  6. soundPool.play(sound1, 1, 1, 0, 0, 1);  

每个参数的含义都写的很清楚了,大家又不清楚的地方可以直接看源码,这里的源码注释很好懂。

 

最后一步就是开启手机震动了,开启手机震动,我需要首先获取震动服务,如下:

[java] view plaincopy print?
  1. //获取手机震动服务  
  2.        vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);  

然后调用Vibrator类中的vibrator方法执行震动,如下:
[java] view plaincopy print?
  1. //1.表示震动的节奏off/on/off/on/off/on......  
  2.         //2.表示是否重复震动,-1表示不重复  
  3.         vibrator.vibrate(new long[]{100, 200, 100, 200, 100, 200}, -1);  

手机震动一定要记得添加震动权限哦,如下:
[java] view plaincopy print?
  1. <uses-permission android:name="android.permission.VIBRATE" />  

OK ,最后,在销毁Activity的时候要解除对传感器的监听,同时释放声音池资源,如下:
[java] view plaincopy print?
  1. @Override  
  2. protected void onDestroy() {  
  3.     super.onDestroy();  
  4.     //解除对加速度传感器的监听  
  5.     sensorManager.unregisterListener(listener);  
  6.     if (soundPool != null) {  
  7.         //声音池释放资源  
  8.         soundPool.release();  
  9.     }  
  10. }  

完整的Activity 代码如下:
[java] view plaincopy print?
  1. public class MainActivity extends AppCompatActivity {  
  2.     private ImageView up;  
  3.     private ImageView down;  
  4.     //上一次晃动手机的时间  
  5.     private long lastTime;  
  6.     private SoundPool soundPool;  
  7.     private int sound1;  
  8.     private Vibrator vibrator;  
  9.     private SensorEventListener listener = new SensorEventListener() {  
  10.         //当手机的加速度发生变化时调用  
  11.         @Override  
  12.         public void onSensorChanged(SensorEvent event) {  
  13.             //获取手机在不同方向上加速度的变化  
  14.             float valuesX = Math.abs(event.values[0]);  
  15.             float valuesY = Math.abs(event.values[1]);  
  16.             float valuesZ = Math.abs(event.values[2]);  
  17.   
  18.             if (valuesX > 17 || valuesY > 17 || valuesZ > 17) {  
  19.                 startAnimation();  
  20.                 playSound();  
  21.             }  
  22.         }  
  23.   
  24.         @Override  
  25.         public void onAccuracyChanged(Sensor sensor, int accuracy) {  
  26.   
  27.         }  
  28.     };  
  29.     private SensorManager sensorManager;  
  30.   
  31.     private void playSound() {  
  32.         //1.声音的id  
  33.         //2.3.表示左右声道的音量  
  34.         //4.优先级  
  35.         //5.是否循环  
  36.         //6.声音播放速率  
  37.         soundPool.play(sound1, 1, 1, 0, 0, 1);  
  38.         //手机震动  
  39.         //1.表示震动的节奏off/on/off/on/off/on......  
  40.         //2.表示是否重复震动,-1表示不重复  
  41.         vibrator.vibrate(new long[]{100, 200, 100, 200, 100, 200}, -1);  
  42.     }  
  43.   
  44.     private void startAnimation() {  
  45.         //如果两次晃动手机的时间小于1秒,则只执行一次动画  
  46.         long currentTimeMillis = System.currentTimeMillis();  
  47.         if (currentTimeMillis - lastTime < 1000) {  
  48.             return;  
  49.         }  
  50.         lastTime = currentTimeMillis;  
  51.         AnimationSet upSet = new AnimationSet(true);  
  52.         TranslateAnimation upUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  53.                 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);  
  54.         upUp.setDuration(1000);  
  55.         TranslateAnimation upDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  56.                 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);  
  57.         upDown.setDuration(1000);  
  58.         upDown.setStartOffset(1000);  
  59.         upSet.addAnimation(upUp);  
  60.         upSet.addAnimation(upDown);  
  61.         up.startAnimation(upSet);  
  62.         AnimationSet downSet = new AnimationSet(true);  
  63.         TranslateAnimation downUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  64.                 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);  
  65.         downUp.setDuration(1000);  
  66.         TranslateAnimation downDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,  
  67.                 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);  
  68.         downDown.setDuration(1000);  
  69.         downDown.setStartOffset(1000);  
  70.         downSet.addAnimation(downUp);  
  71.         downSet.addAnimation(downDown);  
  72.         down.startAnimation(downSet);  
  73.     }  
  74.   
  75.     @Override  
  76.     protected void onCreate(Bundle savedInstanceState) {  
  77.         super.onCreate(savedInstanceState);  
  78.         setContentView(R.layout.activity_main);  
  79.         up = ((ImageView) findViewById(R.id.up));  
  80.         down = ((ImageView) findViewById(R.id.down));  
  81.         initSensor();  
  82.         initSoundPool();  
  83.         //获取手机震动服务  
  84.         vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);  
  85.     }  
  86.   
  87.     /** 
  88.      * 初始化声音池 
  89.      */  
  90.     private void initSoundPool() {  
  91.         if (Build.VERSION.SDK_INT > 20) {  
  92.             SoundPool.Builder builder = new SoundPool.Builder();  
  93.             //1.最大并发流数  
  94.             builder.setMaxStreams(3);  
  95.             AudioAttributes.Builder aaBuilder = new AudioAttributes.Builder();  
  96.             aaBuilder.setLegacyStreamType(AudioManager.STREAM_MUSIC);  
  97.             builder.setAudioAttributes(aaBuilder.build());  
  98.             soundPool = builder.build();  
  99.         } else {  
  100.             soundPool = new SoundPool(3, AudioManager.STREAM_MUSIC, 0);  
  101.         }  
  102.         //加载一个音频文件  
  103.         sound1 = soundPool.load(this, R.raw.awe, 1);  
  104.     }  
  105.   
  106.     /** 
  107.      * 初始化传感器 
  108.      */  
  109.     private void initSensor() {  
  110.         //获取到一个传感器管理器  
  111.         sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);  
  112.         //获得一个加速度传感器  
  113.         Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);  
  114.         //注册传感器监听,  
  115.         //1.监听器  
  116.         //2.加速度传感器  
  117.         //3.传感器灵敏度  
  118.         //传感器灵敏度分为四级,从上往下灵敏度依次降低  
  119.         //SENSOR_DELAY_FASTEST  
  120.         //SENSOR_DELAY_GAME  
  121.         //SENSOR_DELAY_UI  
  122.         //SENSOR_DELAY_NORMAL  
  123.         sensorManager.registerListener(listener, sensor, SensorManager.SENSOR_DELAY_GAME);  
  124.     }  
  125.   
  126.     @Override  
  127.     protected void onDestroy() {  
  128.         super.onDestroy();  
  129.         //解除对加速度传感器的监听  
  130.         sensorManager.unregisterListener(listener);  
  131.         if (soundPool != null) {  
  132.             //声音池释放资源  
  133.             soundPool.release();  
  134.         }  
  135.     }  
  136. }  

以上。

 

转载于:https://www.cnblogs.com/Free-Thinker/p/6544174.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/372710.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java两种设计模式_23种设计模式(11)java策略模式

23种设计模式第四篇&#xff1a;java策略模式定义&#xff1a;定义一组算法&#xff0c;将每个算法都封装起来&#xff0c;并且使他们之间可以互换。类型&#xff1a;行为类模式类图&#xff1a;策略模式是对算法的封装&#xff0c;把一系列的算法分别封装到对应的类中&#xf…

JAXB自定义绑定– Java.util.Date / Spring 3序列化

JaxB可以处理Java.util.Date序列化&#xff0c;但是需要以下格式&#xff1a; “ yyyy-MM-ddTHH&#xff1a;mm&#xff1a;ss ”。 如果需要将日期对象格式化为另一种格式怎么办&#xff1f; 我有同样的问题时&#xff0c;我正在同春MVC 3和Jackson JSON处理器 &#xff0c;最…

双足机器人简单步态生成

让机器人行走最简单的方法是先得到一组步态曲线&#xff0c;即腿部每个关节随时间运动的角度值。可以在ADAMS或3D Max、Blender等软件中建立好机构/骨骼模型&#xff0c;设计出脚踝和髋关节的运动曲线&#xff0c;然后进行逆运动学解算&#xff0c;测量每个关节在运动过程中的转…

重新访问了访客模式

访客模式是面向对象设计中最被高估但又被低估的模式之一。 高估了它&#xff0c;因为它常常被选择得太快&#xff08; 可能是由建筑宇航员选择的 &#xff09;&#xff0c;然后以错误的方式添加时会膨胀本来非常简单的设计。 如果您不遵循教科书示例&#xff0c;那么它可能会非…

ASP.NET—013:实现带控件的弹出层(弹出框)

http://blog.csdn.net/yysyangyangyangshan/article/details/38458169 在页面中用到弹出新页面的情况比较多的&#xff0c;一般来说都是使用JS方法showModalDialog("新页面相对路径?参数1&参数2",window,"新页面样式");然后会新弹出一个模态的page页。…

java 二进制 归属权限_【Java EE 学习 75 上】【数据采集系统第七天】【二进制运算实现权限管理】【权限分析和设计】...

一、权限计算相关分析1.如何存储权限首先说一下权限保存的问题&#xff0c;一个系统中最多有多少权限呢&#xff1f;一个大的系统中可能有成百上千个权限需要管理。怎么保存这么多的权限&#xff1f;首先&#xff0c;我们使用一个数字中的一位保存一种权限&#xff0c;那么如果…

GWT和HTML5 Canvas演示

这是我对GWT和HTML5 Canvas的第一个实验。 我的第一个尝试是创建矩形&#xff0c;仅用几行代码就得出了这样的内容&#xff1a; 码&#xff1a; public class GwtHtml5 implements EntryPoint {static final String canvasHolderId "canvasholder";static final St…

Solr管理界面详解

转载于:https://www.cnblogs.com/gslblog/p/6553813.html

一个实用的却被忽略的命名空间:Microsoft.VisualBasic:

当你看到这个命名空间的时候&#xff0c;别因为是vb的东西就匆忙关掉网页&#xff0c;那将会是您的损失&#xff0c;此命名空间中的资源最初目的是为了简化vb.net开发而创建的&#xff0c;所以microsoft.visualbasic并不属于system命名空间&#xff0c;而是独立存在的。虽然是为…

Linux基础之命令练习Day2-useradd(mod,del),groupadd(mod,del),chmod,chown,

作业一&#xff1a; 1) 新建用户natasha&#xff0c;uid为1000&#xff0c;gid为555&#xff0c;备注信息为“master” 2) 修改natasha用户的家目录为/Natasha 3) 查看用户信息配置文件的最后一行 4) 为natasha用户设置密码“123” 5) 查看用户密码配置文件的最后一行 6) 将nat…

动态表单,JSF世界早已等待

新的PrimeFaces扩展版本0.5.0带来了新的DynaForm组件。 通常&#xff0c;如果知道行/列的数量&#xff0c;元素的位置等&#xff0c;则可以通过h&#xff1a;panelGrid或p&#xff1a;panelGrid来构建非常简单的表单。 对于静态表单&#xff0c;这是正确的。 但是&#xff0c;如…

C# 定时器事件(设置时间间隔,间歇性执行某一函数,控制台程序)

定时器事件代码 static void Main(string[] args) {Method();#region 定时器事件 Timer aTimer new Timer();aTimer.Elapsed new ElapsedEventHandler(TimedEvent);aTimer.Interval seconds * 1000; //配置文件中配置的秒数aTimer.Enabled true;#endregionstring strLi…

Vmware安装Centos NAT方式设置静态IP

【Vmware中在搭建集群环境等&#xff0c;DHCP自动获取IP方式不方便&#xff0c;为了固定IP减少频繁更改配置信息&#xff0c;建议使用静态IP来配置&#xff0c;网络连接主要有三种方式 1.nat 2.桥接&#xff0c;3主机模式 &#xff0c;在这里主要介NAT方式&#xff0c; 为什么使…

1 TB /节点时快速,可预测且高度可用

世界正每秒从移动设备&#xff0c;Web和各种小工具向应用程序推送大量数据。 如今&#xff0c;更多的应用程序必须处理此数据。 为了保持性能&#xff0c;这些应用程序需要快速访问数据层。 在过去的几年中&#xff0c;RAM价格下降了&#xff0c;我们现在可以便宜得多地获得具有…

java jni 内存_Android开发之JNI内存模型

Java 与JNI 内存管理是怎样的想要弄清楚Java与JNI的内存管理的关系&#xff0c;首先要弄清楚JVM的内存模型JVM内存模型.png其中本地方法栈就是运行时调用native 方法的数据保存区。本地方法栈的大小可以设置成固定的或者是动态扩展。Java中的内存泄露JAVA 编程中的内存泄漏&…

04 linux用户群组和权限

作业一&#xff1a; 1)新建用户natasha&#xff0c;uid为1000&#xff0c;gid为555&#xff0c;备注信息为“master” 2)修改natasha用户的家目录为/Natasha 3)查看用户信息配置文件的最后一行 4)为natasha用户设置密码“123” 5)查看用户密码配置文件的最后一行 6)将natasha用…

基于 CoreText 实现的高性能 UITableView

引起UITableView卡顿比较常见的原因有cell的层级过多、cell中有触发离屏渲染的代码&#xff08;譬如&#xff1a;cornerRadius、maskToBounds 同时使用&#xff09;、像素是否对齐、是否使用UITableView自动计算cell高度的方法等。本文将从cell层级出发&#xff0c;以一个仿朋友…

Web Magic 总体架构

1.2 总体架构 WebMagic的结构分为Downloader、PageProcessor、Scheduler、Pipeline四大组件&#xff0c;并由Spider将它们彼此组织起来。这四大组件对应爬虫生命周期中的下载、处理、管理和持久化等功能。WebMagic的设计参考了Scapy&#xff0c;但是实现方式更Java化一些。 而S…

SpringMVC搭建+实例

想做一点自己喜欢的东西&#xff0c;研究了一下springMVC,所以就自己搭建一个小demo,可供大家吐槽。 先建一个WEB工程&#xff0c;这个相信大家都会&#xff0c;这里不在多说。去网上下载spring jar包&#xff0c;然后在WEB-INF下新建一个lib文件&#xff0c;将下载的jar包放进…

php8更新,PHP 8 中新特性以及重大调整

PHP 8&#xff0c;PHP 的一个新的大版本&#xff0c;预计将于2020年12月3日发布&#xff0c;这意味着将不会有 PHP 7.5 版本。PHP8目前正处于非常活跃的开发阶段&#xff0c;所以在接下来的几个月里&#xff0c;情况可能会发生很大的变化。在这篇文章中&#xff0c;我会维持一个…