Android前台服务和通知

news2024/11/7 7:43:50

前台服务

Android 13及以上系统需要动态获取通知权限。

//android 13及以上系统动态获取通知权限
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
    checkPostNotificationPermission();
}
private void checkPostNotificationPermission() {
    if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.POST_NOTIFICATIONS)
            != PackageManager.PERMISSION_GRANTED) {
        ActivityCompat.requestPermissions((Activity) this, new String[]{
                Manifest.permission.POST_NOTIFICATIONS}, 200);
    } else if (manager != null) {
        //通知ID设置(您可以使用任何您想要的ID,相同ID通知只会显示一个)。
        manager.notify(2, builder.build());
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if (requestCode == 200) {
        if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            //允许了通知权限
        } else {
            Toast.makeText(this, "您拒绝了通知权限", Toast.LENGTH_SHORT).show();
        }
    }
}

别忘记添加通知权限

<!--发布通知权限-->
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />

开启前台服务需要添加前台服务权限

<!--前台服务权限-->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />

通知震动需要添加震动权限

<!--振动器权限-->
<uses-permission android:name="android.permission.VIBRATE" />

前台服务代码

import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.graphics.Color;
import android.net.Uri;
import android.os.Build;
import android.os.IBinder;

import androidx.annotation.Nullable;
import androidx.core.app.NotificationCompat;

import com.example.javatest.MainActivity;
import com.example.javatest.R;

/**
 * created by cwj on 2023-10-19
 * Description: 前台服务
 */
public class ForegroundService extends Service {

    @Override
    public void onCreate() {
        super.onCreate();
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            CharSequence name = getString(R.string.app_name);
            int importance = NotificationManager.IMPORTANCE_DEFAULT;
            NotificationChannel channel = new NotificationChannel("CHANNEL_ID", name, importance);
            channel.setDescription("前台服务守护进程");
            NotificationManager notificationManager = getSystemService(NotificationManager.class);
            notificationManager.createNotificationChannel(channel);
        }

        //设置服务跳转
        Intent intent = new Intent(this, MainActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 1, intent, PendingIntent.FLAG_ONE_SHOT | PendingIntent.FLAG_IMMUTABLE);

        NotificationCompat.Builder builder = new NotificationCompat.Builder(this, "CHANNEL_ID")
                // 设置通知的小图标。
                .setSmallIcon(R.mipmap.ic_launcher)
                //设置通知的标题。
                .setContentTitle(getString(R.string.app_name))
                //设置通知的内容
                .setContentText("进程守护中")
                //设置通知的优先级。
                .setPriority(NotificationCompat.PRIORITY_DEFAULT)
                //震动模式的通知在Android O及以上。
                .setVibrate(new long[]{0, 1000, 500, 1000})
                //在Android O及以上的系统上使用LED光色和微信号。
                .setLights(Color.RED, 1000, 1000)
                //设置Android O及以上版本的通知声音。
                .setSound(Uri.parse("android.resource://" + this.getPackageName() + "/raw/notification_sound"))
                //如果需要,在Android O及以上版本设置较大的通知标题。
                // .setLargeTitle("Large Title Text")
                //如果需要,在Android O及以上版本设置通知的子文本。
                // .setSubText("Sub Text")
                //如果需要,在Android O及以上版本设置大字体通知。
                // .setStyle(new NotificationCompat.BigTextStyle().bigText("Big Text"))
                //如果需要,在Android O及以上版本设置通知摘要文本(当有多个具有相同优先级的通知时显示摘要文本)。
                // .setSummaryText("Summary Text")
                //如果需要,在通知中添加动作按钮(可用于启动活动或发送广播)。
                // .addAction(R.drawable.ic_action, "Action Title", PendingIntent)
                //设置可点击跳转
                .setContentIntent(pendingIntent);
        // 将通知设置为前台服务
        startForeground(1, builder.build());
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        // 在这里执行需要在前台运行的任务

        // 返回START_STICKY表示服务在被杀死后会自动重启
        return START_STICKY;
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();

        // 停止前台服务
        stopForeground(true);
        stopSelf();
    }
}

application中添加服务

<service
    android:name=".service.ForegroundService"
    android:enabled="true"
    android:exported="false" />

activity中启动服务

Intent intent = new Intent(this, ForegroundService.class);
// Android 8.0使用startForegroundService在前台启动新服务
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
    startForegroundService(intent);
} else {
    startService(intent);

创建通知

 模拟发送通知

binding.iv.setOnClickListener(v ->
        showNotification(getString(R.string.app_name), dateToString(System.currentTimeMillis(), "yyyy-MM-dd HH:mm:ss") + "您收到一条消息")
);

创建通知及通知点击取消通知

private NotificationCompat.Builder builder;
private NotificationManagerCompat manager;

private void showNotification(String title, String content) {
    // 创建通知频道,如果用户没有创建,则会自动创建。
    String CHANNEL_ID = "message";
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        if (getSystemService(NotificationManager.class).getNotificationChannel(CHANNEL_ID) == null) {
            NotificationChannel channel = new NotificationChannel(CHANNEL_ID, "消息", NotificationManager.IMPORTANCE_DEFAULT);
            //通知的振动模式。
            channel.setVibrationPattern(new long[]{0, 1000, 500, 1000});
            //为通知通道启用振动。
            channel.enableVibration(true);
            //设置通道的颜色。
            channel.setLightColor(Color.RED);
            //设置通道的锁屏可见性。
            channel.setLockscreenVisibility(NotificationCompat.VISIBILITY_PUBLIC);
            getSystemService(NotificationManager.class).createNotificationChannel(channel);
        }
    }
    //设置服务跳转
    Intent intent = new Intent(this, MainActivity.class);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
    PendingIntent pendingIntent = PendingIntent.getActivity(this, 2, intent, PendingIntent.FLAG_ONE_SHOT | PendingIntent.FLAG_IMMUTABLE);
    // 创建并分发通知。
    builder = new NotificationCompat.Builder(this, CHANNEL_ID)
            // 设置通知的小图标。
            .setSmallIcon(R.mipmap.ic_launcher)
            //设置通知的标题。
            .setContentTitle(title)
            //设置通知的内容
            .setContentText(content)
            //设置通知的优先级。
            .setPriority(NotificationCompat.PRIORITY_DEFAULT)
            //震动模式的通知在Android O及以上。
            .setVibrate(new long[]{0, 1000, 500, 1000})
            //在Android O及以上的系统上使用LED光色和微信号。
            .setLights(Color.RED, 1000, 1000)
            //设置Android O及以上版本的通知声音。
            .setSound(Uri.parse("android.resource://" + this.getPackageName() + "/raw/notification_sound"))
            //如果需要,在Android O及以上版本设置较大的通知标题。
            // .setLargeTitle("Large Title Text")
            //如果需要,在Android O及以上版本设置通知的子文本。
            // .setSubText("Sub Text")
            //如果需要,在Android O及以上版本设置大字体通知。
            // .setStyle(new NotificationCompat.BigTextStyle().bigText("Big Text"))
            //如果需要,在Android O及以上版本设置通知摘要文本(当有多个具有相同优先级的通知时显示摘要文本)。
            // .setSummaryText("Summary Text")
            //如果需要,在通知中添加动作按钮(可用于启动活动或发送广播)。
            // .addAction(R.drawable.ic_action, "Action Title", PendingIntent)
            //设置可点击跳转
            .setContentIntent(pendingIntent)
            //开启点按通知后自动移除通知
            .setAutoCancel(true);
    manager = NotificationManagerCompat.from(this);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        checkPostNotificationPermission();
    } else {
        // 在 Android 10 或更早版本中,不需要请求此权限。
        //ID要和前台服务ID区分开,相同ID只能显示一条通知。
        manager.notify(2, builder.build());
    }

activity完整代码

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.app.NotificationCompat;
import androidx.core.app.NotificationManagerCompat;

import android.Manifest;
import android.app.Activity;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.graphics.Color;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.widget.Toast;

import com.example.javatest.databinding.ActivityMainBinding;
import com.example.javatest.service.ForegroundService;

import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {

    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = ActivityMainBinding.inflate(getLayoutInflater());
        setContentView(binding.getRoot());

        initView();
        initData();
    }

    private void initView() {
        //android 13及以上系统动态获取通知权限
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
            checkPostNotificationPermission();
        }

        Intent intent = new Intent(this, ForegroundService.class);
        // Android 8.0使用startForegroundService在前台启动新服务
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            startForegroundService(intent);
        } else {
            startService(intent);
        }
    }

    private void initData() {
        binding.iv.setOnClickListener(v ->
                showNotification(getString(R.string.app_name), dateToString(System.currentTimeMillis(), "yyyy-MM-dd HH:mm:ss") + "您收到一条消息")
        );
    }

    private String dateToString(long time, String format) {
        Date date = new Date(time);
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat(format);
        return simpleDateFormat.format(date);
    }

    private NotificationCompat.Builder builder;
    private NotificationManagerCompat manager;

    private void showNotification(String title, String content) {
        // 创建通知频道,如果用户没有创建,则会自动创建。
        String CHANNEL_ID = "message";
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            if (getSystemService(NotificationManager.class).getNotificationChannel(CHANNEL_ID) == null) {
                NotificationChannel channel = new NotificationChannel(CHANNEL_ID, "消息", NotificationManager.IMPORTANCE_DEFAULT);
                //通知的振动模式。
                channel.setVibrationPattern(new long[]{0, 1000, 500, 1000});
                //为通知通道启用振动。
                channel.enableVibration(true);
                //设置通道的颜色。
                channel.setLightColor(Color.RED);
                //设置通道的锁屏可见性。
                channel.setLockscreenVisibility(NotificationCompat.VISIBILITY_PUBLIC);
                getSystemService(NotificationManager.class).createNotificationChannel(channel);
            }
        }

        //设置服务跳转
        Intent intent = new Intent(this, MainActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 2, intent, PendingIntent.FLAG_ONE_SHOT | PendingIntent.FLAG_IMMUTABLE);

        // 创建并分发通知。
        builder = new NotificationCompat.Builder(this, CHANNEL_ID)
                // 设置通知的小图标。
                .setSmallIcon(R.mipmap.ic_launcher)
                //设置通知的标题。
                .setContentTitle(title)
                //设置通知的内容
                .setContentText(content)
                //设置通知的优先级。
                .setPriority(NotificationCompat.PRIORITY_DEFAULT)
                //震动模式的通知在Android O及以上。
                .setVibrate(new long[]{0, 1000, 500, 1000})
                //在Android O及以上的系统上使用LED光色和微信号。
                .setLights(Color.RED, 1000, 1000)
                //设置Android O及以上版本的通知声音。
                .setSound(Uri.parse("android.resource://" + this.getPackageName() + "/raw/notification_sound"))
                //如果需要,在Android O及以上版本设置较大的通知标题。
                // .setLargeTitle("Large Title Text")
                //如果需要,在Android O及以上版本设置通知的子文本。
                // .setSubText("Sub Text")
                //如果需要,在Android O及以上版本设置大字体通知。
                // .setStyle(new NotificationCompat.BigTextStyle().bigText("Big Text"))
                //如果需要,在Android O及以上版本设置通知摘要文本(当有多个具有相同优先级的通知时显示摘要文本)。
                // .setSummaryText("Summary Text")
                //如果需要,在通知中添加动作按钮(可用于启动活动或发送广播)。
                // .addAction(R.drawable.ic_action, "Action Title", PendingIntent)
                //设置可点击跳转
                .setContentIntent(pendingIntent)
                //开启点按通知后自动移除通知
                .setAutoCancel(true);

        manager = NotificationManagerCompat.from(this);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
            checkPostNotificationPermission();
        } else {
            // 在 Android 10 或更早版本中,不需要请求此权限。
            //显示ID为1的通知(您可以使用任何您想要的ID)。
            manager.notify(2, builder.build());
        }
    }

    private void checkPostNotificationPermission() {
        if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.POST_NOTIFICATIONS)
                != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions((Activity) this, new String[]{
                    Manifest.permission.POST_NOTIFICATIONS}, 200);
        } else if (manager != null) {
            //显示ID为1的通知(您可以使用任何您想要的ID)。
            manager.notify(2, builder.build());
        }
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == 200) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                //允许了通知权限
            } else {
                Toast.makeText(this, "您拒绝了通知权限", Toast.LENGTH_SHORT).show();
            }
        }
    }
}

完整配置文件AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">

    <!--网络权限-->
    <uses-permission android:name="android.permission.INTERNET" />
    <!--前台服务权限-->
    <uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
    <!--振动器权限-->
    <uses-permission android:name="android.permission.VIBRATE" />
    <!--全屏意图权限-->
    <uses-permission android:name="android.permission.USE_FULL_SCREEN_INTENT" />
    <!--发布通知权限-->
    <uses-permission android:name="android.permission.POST_NOTIFICATIONS" />

    <application
        android:allowBackup="true"
        android:dataExtractionRules="@xml/data_extraction_rules"
        android:fullBackupContent="@xml/backup_rules"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.JavaTest"
        tools:targetApi="31">
        <activity
            android:name=".MainActivity"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <service
            android:name=".service.ForegroundService"
            android:enabled="true"
            android:exported="false" />
    </application>

</manifest>

build.gradle.kts文件中添加viewBinding支持

buildFeatures{
    viewBinding = true
}

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

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

相关文章

xhadmin多应用Saas框架如何下载安装?

xhadmin是什么&#xff1f; xhadmin 是一套基于最新技术的研发的多应用 Saas 框架&#xff0c;支持在线升级和安装模块及模板&#xff0c;拥有良好的开发框架、成熟稳定的技术解决方案、提供丰富的扩展功能。为开发者赋能&#xff0c;助力企业发展、国家富强&#xff0c;致力于…

小白学java--垃圾回收机制(Garbage Collection)

压测过程中&#xff0c;作为测试会时不时听到研发说命中gc了&#xff0c;如果一头雾水&#xff0c;来看看什么是gc。 1、什么是垃圾回收机制 垃圾回收的执行过程会导致一些额外的开销&#xff0c;例如扫描和标记对象、回收内存空间等操作。这些开销可能会导致一定的性能损失和…

云安全(2)--CAP_SYS_MODULE逃逸

文章目录 测试环境配置实际环境利用 测试环境配置 docker run -it --cap-addSYS_MODULE ubuntu:18.04实际环境利用 cat /proc/self/status|grep Cap然后使用capsh decode一下 capsh --decode00000000a80525fb发现有CAP_SYS_MODULE权限&#xff0c;那么直接往内核注入恶意mo…

Streamlit库开发python交互式Web应用程序

Streamlit是一个开源的Python库&#xff0c;旨在帮助开发人员轻松创建数据科学和机器学习的Web应用程序。它允许您只需几行Python代码即可构建交互式Web应用程序&#xff0c;因此常用于创建数据驱动的应用程序、仪表板和原型。Streamlit以其简单性而闻名&#xff0c;通常被数据…

光影之梦:影视动画渲染的魅力

在动画世界中&#xff0c;光影与色彩是赋予生命与灵魂的魔法。它们将虚无的想象变为具象的画面&#xff0c;让故事情感跃然于屏幕之上。影视动画渲染&#xff0c;正是这一魔法的精妙施展&#xff0c;它以光影为笔&#xff0c;色彩为墨&#xff0c;勾勒出一个个绚丽多姿的梦境。…

洗地机哪个好用?2023年洗地机推荐指南

说到提高家庭幸福生活的家电&#xff0c;洗地机肯定是少不了的&#xff0c;特别对于现在快节奏的生活来说&#xff0c;高效率的解决家务活&#xff0c;而且能够大幅度的提高生活质量。在市场上&#xff0c;消费者面临着选择合适洗地机的难题&#xff0c;因为有各种型号、功能和…

德国大陆博世 ars 548 4D 毫米波雷达 window 系统或者 Ubuntu 系统通讯以及数据解析和显示程序

德国大陆博世 ars 548 4D 毫米波雷达 window 系统或者 Ubuntu ROS 系统通讯以及数据解析和显示程序

这件事,准备考PMP的都必须知道

大家好&#xff0c;我是老原。 新的一月&#xff0c;新的困惑。最近接到的咨询很多&#xff0c;但的确出现了差异化的特质。 以前的粉丝朋友上来就问&#xff0c;我现在是项目经理&#xff0c;主要负责产品研发&#xff0c;我是考PMP还是NPDP好&#xff1f; 现在的粉丝朋友会…

AGC电路,模拟乘法器

文章目录 AGC电路乘法器 AGC电路 注&#xff1a;下面三个没试过 乘法器 TI只有这一种乘法器&#xff0c;跟ADI的AD534一模一样 这个报告里有很多错误 做乘法器最厉害的是ADI

JUC并发编程——各种锁的理解(基于狂神说的学习笔记)

各种锁的理解 公平锁与非公平锁 公平锁&#xff1a;非常公平&#xff0c;不能够插队&#xff0c;先来后到 非公平锁&#xff1a;可以插队&#xff0c;比较灵活&#xff08;默认都是非公平&#xff0c;如&#xff1a;synchronized,lock&#xff09; // Lock lock new Reent…

从一次性销售到持续收益:低代码服务商的转型之路

随着低代码市场快速发展&#xff0c;低代码代理商也成为近年来快速崛起的一种新型IT服务提供商&#xff0c;其以敏捷、灵活和高效的优势可以有效地帮助企业加速数字化转型。然而&#xff0c;尽管低代码代理商们的潜力巨大&#xff0c;却共同面临着一个关键的挑战——与客户的合…

瑞芯微RKNN开发·yolov7

官方预训练模型转换 下载yolov7源码解压到本地&#xff0c;并配置基础运行环境。下载官方预训练模型 yolov7-tiny.ptyolov7.pt… 进入yolov7-main目录下&#xff0c;新建文件夹weights&#xff0c;并将步骤2中下载的权重文件放进去。修改models/yolo.py文件 def forward(sel…

【C++】:类和对象(中)之类的默认成员函数——构造函数and析构函数

1.类的6个默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为空类 空类中真的什么都没有吗&#xff1f;并不是&#xff0c;任何类在什么都不写时&#xff0c;编译器会自动生成以下6个默认成员函数 默认成员函数&#xff1a;用户没有显式实现&#xff0c;编译器会生成…

系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第一部分:通信协议(2)

本心、输入输出、结果 文章目录 系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第一部分&#xff1a;通信协议&#xff08;2&#xff09;前言SOAP vs REST vs GraphQL vs RPC代码优先与 API 优先HTTP 状态代码API 网关有什么作用步骤说明 我们如何设计有效和安全的 API弘…

Nautilus Chain 与 Coin98 生态达成合作,加速 Zebec 生态亚洲战略进程

目前&#xff0c;行业内首个模块化Layer3架构公链Nautilus Chain已经上线主网&#xff0c;揭示了模块化区块链领域迎来了全新的进程。在主网上线后&#xff0c;Nautilus Chain将扮演Zebec生态中最重要的底层设施角色&#xff0c;并将为Zebec APP以及Zebec Payroll规模性的采用提…

驱动day2作业

编写应用程序控制三盏灯亮灭 head.h #ifndef __HEAD_H__ #define __HEAD_H__ #define PHY_LED1_MODER 0x50006000 #define PHY_LED2_MODER 0x50007000 #define PHY_LED1_ODR 0x50006014 #define PHY_LED2_ODR 0x50007014 #define PHY_RCC 0x50000A28#endif demo1.c #includ…

海外调查问卷赚钱是真的吗?

海外问卷赚钱是真实的吗&#xff1f;我是橙河网络&#xff0c;一家问卷公司的老板&#xff0c;做这个行业已经2年时间了&#xff0c;首先给大家一个明确的回答&#xff1a;海外问卷调查赚钱是真实的&#xff01; 海外问卷调查项目&#xff0c;在国内已经存在一二十年的时间了&…

21.3 Python 使用DPKT分析数据包

dpkt项目是一个Python模块&#xff0c;主要用于对网络数据包进行解析和操作。它可以处理多种协议&#xff0c;例如TCP、UDP、IP等&#xff0c;并提供了一些常用的网络操作功能&#xff0c;例如计算校验和、解析DNS数据包等。由于其简单易用的特性&#xff0c;dpkt被广泛应用于网…

[C语言]排序的大乱炖——喵喵的成长记

宝子&#xff0c;你不点个赞吗&#xff1f;不评个论吗&#xff1f;不收个藏吗&#xff1f; 最后的最后&#xff0c;关注我&#xff0c;关注我&#xff0c;关注我&#xff0c;你会看到更多有趣的博客哦&#xff01;&#xff01;&#xff01; 喵喵喵&#xff0c;你对我真的很重要…