【Android 笔记】记移植OpenCV4.8图像人脸识别

news2025/1/13 10:08:16

前言


因业务需要,使用大屏端摄像头捕获图像,且要识别图像中人脸的数目以及从中随机抽取一人。

业务流程如下,调用摄像头预览、拍照,使用OpenCV库进行人脸识别,将识别到的人脸使用矩形框绘制出来,从识别的人脸中随机选中一人进行展示。

在这里插入图片描述

实现


一、集成OpenCV4.8版本库

本例中人脸识别功能使用OpenCV4.8.0版本来实现,进入官网,选择OpenCV-4.8.0版本,Android平台下载。

OpenCV官网

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

下载解压,将文件夹中的sdk作为module导入到工程中,以下是OpenCV的检测流程。

在这里插入图片描述

1、加载本地OpenCV

 
	 if (!OpenCVLoader.initDebug()) {
	      Log.d(TAG, "Internal OpenCV library not found. Using OpenCV Manager for initialization")
	      OpenCVLoader.initAsync(OpenCVLoader.OPENCV_VERSION, DemoApplication.context, loaderCallback)
	  } else {
	      Log.d(TAG, "OpenCV library found inside package. Using it!")
	      loaderCallback.onManagerConnected(LoaderCallbackInterface.SUCCESS)
	  }
	
	 private val loaderCallback: BaseLoaderCallback = object : BaseLoaderCallback(DemoApplication.context) {
	         override fun onManagerConnected(status: Int) {
	             when (status) {
	                 SUCCESS -> {
	                     initOnnx()
	                     Log.i(TAG, "OpenCV loaded successfully")
	                 }
	
	                 else -> {
	                     super.onManagerConnected(status)
	                 }
	             }
	         }
	 }

2、初始化检测模型

检测模型使用YuNet架构,YuNet 是一种轻量级的面部检测网络,在移动设备和嵌入式设备上实现高效、实时的面部检测。320_320: 表示模型的输入尺寸是 320x320 像素,模型通常在特定的输入尺寸下进行训练,以优化检测性能和速度。

新建res/raw,将检测模型文件放入raw中。

在这里插入图片描述


   /**
   	* 初始化检测模型
   	*/
  	private fun initOnnx() {
      	try {
	          val yyNet = resources.openRawResource(R.raw.yunet_n_320_320)
	          val cascadeDir = requireContext().getDir("cascade", Activity.MODE_PRIVATE)
	          val cascadeFile = File(cascadeDir, "yunet_n_320_320.onnx")
	          val os = FileOutputStream(cascadeFile)
	          val buffer = ByteArray(4096)
	          var bytesRead: Int
	          while (yyNet.read(buffer).also { bytesRead = it } != -1) {
	              os.write(buffer, 0, bytesRead)
	          }
	          yyNet.close()
	          os.close()
         	 //创建检测器时用到
          	modelPath = cascadeFile.absolutePath
     	 } catch (e: Exception) {
       	   e.printStackTrace()
     	 }
 	 }
    

3、创建检测器

步骤2读取检测模型,在创建检测器时作为参数传入,

 
 	public static FaceDetectorYN create(String model, String config, Size input_size, float score_threshold, float nms_threshold, int top_k, int backend_id, int target_id) {}
 

FaceDetectorYNcreate()方法参数含义如下。


	 private var backendId = 3
	
	 private var targetId = 0
	
	 private var scoreThreshold = 0.70F
	
	 private var nmsThreshold = 0.2F
	
	 private var topK = 5000


	/**
	 * @param modelPath 预训练模型文件的路径,通常是一个 .onnx 文件
	 * @param config 配置文件路径,一般为空字符串
	 * @param inputSize 输入图像的尺寸,以 Size(width, height) 的形式传入
	 * @param scoreThreshold 置信度分数阈值,默认值为 0.9
	 * @param nmsThreshold 非极大值抑制(NMS)阈值,默认值为 0.3
	 * @param topK 留的最高置信度检测结果的数量上限,默认值为 5000
	 * @param backendId 指定使用的计算后端,默认值为 0(DNN_BACKEND_DEFAULT)
	 * @param targetId 指定计算目标设备,默认值为 0(DNN_TARGET_CPU)
	 */
	    
	 private fun detectPicture() {
	     val images = org.opencv.android.Utils.loadResource(DemoApplication.context, R.mipmap.face)
	     val detector = FaceDetectorYN.create(modelPath, "", images.size(), scoreThreshold, nmsThreshold, topK, backendId, targetId)
	     detector.inputSize = images.size()
	     val faces = Mat()
	     detector.detect(images, faces)
	     
	     //自定义View,绘制人脸
	 	 mViewBinding.detectResult.setFacesData(images, faces)
	 }
 

二、自定义FacialDrawView绘制人脸

前面介绍了OpenCV的配置过程,集成导入、加载模型、初始化检测器以及识别方法。识别的结果参数中有总共识别到的数目、以及每个人脸的的坐标信息。有了信息接下来就比较简单了,自定义FacialDrawView,绘制识别到总人数以及绘制矩形框将人脸标记出来即可,下面是识别标记效果。

在这里插入图片描述
下面贴下FacialDrawView中主要的方法,绘制标记所有人脸、绘制总人数提示框。

 	
	 <!--布局文件中引入-->
   	 <com.ho.csdn.widget.FacialDrawView
        android:id="@+id/detectResult"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:visibility="visible" />


	 /**
	  * 是否是pick模式
	  */
	 private var isPick = false
	 
	 /**
	  * 识别到的人脸数据
	  */
	 private var mFaces: Mat? = null
	 
	 /**
	  * 缩放比
	  */
	 private var scale = 0f
	
	 /**
	  * 随机选人,计数使用
	  */
	 private var count = 0
	
	 /**
	  * 人脸数目
	  */
	 private var rows: Int = 0
	
	 /**
	  * 图片宽
	  */
	 private var imgWidth = 0f
	
	 /**
	  * 图片高
	  */
	 private var imgHeight = 0f
	
	 /**
	  * 设置识别数据
	  * @param[images] 识别的图片
	  * @param[faces] 图片中
	  */
	 fun setFacesData(images: Mat, faces: Mat) {
	     isPick = false
	     mFaces = faces
	     rows = faces.rows()
	     countdown = getCount()
	     imgWidth = images.width().toFloat()
	     imgHeight =  images.height().toFloat()
	     scale = (width / imgWidth).coerceAtMost(height / imgHeight)
	     invalidate()
	 }
	
	 /**
	  * onDraw方法中调用
	  * 绘制所有人脸矩形框
	  */
	 private fun drawAllRect(canvas: Canvas) {
	     paint.style = Paint.Style.STROKE
	     paint.color = ctx.getColor(R.color.color_rect)
	     for (i in 0..rows) {
	         val x1 = mFaces?.get(i, 0)?.get(0)?.times(scale)?.toInt()
	         val y1 = mFaces?.get(i, 1)?.get(0)?.times(scale)?.toInt()
	         val x2 = mFaces?.get(i, 2)?.get(0)?.times(scale)?.toInt()
	         val y2 = mFaces?.get(i, 3)?.get(0)?.times(scale)?.toInt()
	
	         if (x1 != null && x2 != null && y1 != null && y2 != null) {
	             detectRect = Rect(x1, y1, x2 + x1, y2 + y1)
	             canvas.drawRect(detectRect, paint)
	         }
	     }
	 }


 	 /**
	  * 文字的高度
	  */
	 private var textY  = 0f
	
	 /**
	  * 自定义View宽度
	  */
	 private var vWidth = 0
	
	 /**
	  * 自定义View高度
	  */
	 private var vHeight = 0
		
	 /**
	  * 圆角矩形距离顶部距离
	  */
	 private var rectTop = 60f
	
	 /**
	  * 圆角矩形的高度
	  */
	 private var rectH = 120
	  
	 /**
	  * 圆角矩形的宽度
	  */
	 private var rectW = 520

	 /**
	  * 绘制总人数
	  */
	 private fun drawNumber(canvas: Canvas) {
	    if (rows > 0) {
	        paint.apply {
	            style = Paint.Style.FILL
	            color = ctx.getColor(R.color.color_rect)
	        }
	        //绘制蓝色底圆角矩形
	        canvas.drawRoundRect(RectF((width - rectW) /  2f,rectTop,width / 2f + rectW / 2,rectH +  rectTop),20f,20f,paint)
	
	        paint.color = ctx.getColor(R.color.colorWhite)
	        val personTxt = ctx.getString(R.string.detect_student_num,rows)
	        //计算文字的宽度
	        val txtWidth = paint.measureText(personTxt,0,personTxt.length)
	        //绘制文字
	        canvas.drawText(personTxt, (width - txtWidth) /  2, textY +  rectTop + (rectH - textY) /  2, paint)
	    }
	 }


     override fun onSizeChanged(w: Int, h: Int, oldw: Int, oldh: Int) {
        super.onSizeChanged(w, h, oldw, oldh)
        vWidth = w
        vHeight = h
        calculateTextPos()
     }


	 /**
	  * 计算文字的高度
	  */
	 private fun calculateTextPos(){
	    //获取fontMetrics对象
	    val fontMetrics = paint.fontMetrics
	    //获取文本的高度的一半,取文字垂直中线高度值
	    val textHalfHeight = (abs(fontMetrics.descent) + abs(fontMetrics.ascent)) / 2
	    //将文字的向上移动Descent,再向下移动文字高度一半
	    textY = abs(fontMetrics.descent) + textHalfHeight
	 }

关于文字高度的计算,这里就不详细介绍了,可以参考之前写的一篇文章 Android自定义控件(六) Andriod仿iOS控件Switch开关


三、实现随机选人功能

识别到所有的人脸后,再从其他随机选择一个人脸进行展示就很简单了,开启定时器,在人脸总数范围内生成随机索引,每次绘制对应索引的人脸矩形,就能达到这样的效果。


	/**
     * 是否正在识别
     */
    private var isDetecting = false

  	/**
     * 随机倒计时
     */
    private var countdown = 10

	/**
     * 最终选择学生人脸矩形框对象
     */
    private lateinit var detectRect : Rect
    
    /**
     * 定时器对象
     */
    private var timer: Timer? = null

    /**
     * 定时器任务对象
     */
    private var task: TimerTask? = null

 	/**
     * 随机数
     */
    private var random = 0
  	
  	/**
     * 是否正在检测,主类中使用
     */
    fun isDetecting():Boolean = isDetecting

    /**
     * 随机选人
     */
    fun setRandomPick() {
        isPick = true
        random = 0
        //如果总人数是1,不随机,直接回调
        if(rows == 1){
            isDetecting = false
            mListener?.complete(scale,detectRect)
        }else{
            initTimer()
        }
    }

    /**
     * 开启计时器
     */
    private fun initTimer() {
        isDetecting = true
        timer = Timer()
        task = object : TimerTask() {
            override fun run() {
                //在人脸数目范围内生成随机数
                random = Random.nextInt(rows)
                count += 1
                if (count == countdown) {
                    isDetecting = false
                    //回调到主类显示人脸
                    mListener?.complete(scale,detectRect)
                    cancelTask()
                }else{
                    invalidate()
                }
            }
        }
        timer!!.schedule(task, 0, 200)
    }
    

    /**
     * 根据人脸数目获取pick时间
     */
    private fun getCount():Int{
        var countdown = 10
        if (rows < 10){
            countdown = 10
        }else if(rows in 11..30){
            countdown = 20
        }else if(rows > 30){
            countdown = 40
        }
        return countdown
    }

    /**
     * 识别完成回调
     */
    private var mListener:DetectCompleteListener ? = null

    interface DetectCompleteListener{
        fun complete(scale:Float,rect:Rect)
    }

    fun setCompleteListener(listener:DetectCompleteListener){
        mListener = listener
    }
  

四、展示

FacialDrawView实现随机点名后,将最终的选中的学生人脸坐标范围回调到主类,PickDialog中内容比较简单,实现展示头像、重新点名功能。

在这里插入图片描述

     

    /**
     * 完成随机选人后回调到主类
     */    
    mViewBinding.detectResult.setCompleteListener(object:FacialDrawView.DetectCompleteListener{
            override fun complete(scale:Float,rect: Rect) {
                val left = (rect.left / scale).toInt()
                val top = (rect.top / scale).toInt()
                //裁剪图片中指定的区域
                val cropBitmap = cropBitmap(detectBitmap!!,scale,left ,top ,rect)
                requireActivity().runOnUiThread { showPickDialog(cropBitmap) }
            }
        })
        

    /**
     * 裁剪识别到的头像
     */
    private fun cropBitmap(bitmap: Bitmap, scale:Float,left: Int, top: Int,rect:Rect): Bitmap {
        val matrix = Matrix()
        matrix.setScale(scale,scale)

        val leftX: Int = if(left - 40 < 0){ 0 }else{ left - 40 }
        val topY: Int = if(top - 40 < 0) { 0 }else { top - 40 }

        val totalW = leftX + rect.width() / 2 + 80
        var vWidth = rect.width() / 2 + 80
        if(totalW > bitmap.width){
            vWidth = bitmap.width - leftX
        }

        val totalH = topY + rect.height() / 2 + 80
        var vHeight = rect.height() / 2 + 80
        if(totalH > bitmap.height){
            vHeight = bitmap.height - topY
        }
        return Bitmap.createBitmap(bitmap, leftX, topY, vWidth , vHeight,matrix,false)
    }


    private fun showPickDialog(cropBitmap: Bitmap) {
        PickDialog
            .init(requireContext())
            .setAvatar(cropBitmap)
            .create().setPickListener(object :PickDialog.PickListener{
                override fun pickAgain(isRetry:Boolean) {
                    if(isRetry){
                    	//重新选人
                        mViewBinding.detectResult.setRandomPick()
                    }else{
             			//todo
                    }
                }

            }).show()
    }


五、UVC摄像头预览拍照

上述使用本地资源文件进行人脸识别过程,和摄像头拍照识别过程是一致的,只需要将预览、拍照、将照片文件转为为OpenCV能识别的图像格式即可。因大屏端使用的是USB Camera,尝试使用Android原生Camera1Camera2 API来调用相机,预览十分卡顿,无法正常使用,这里就借用了Github上大神的AndroidUSBCamera工程库来实现,下面介绍下识别拍照中人脸的一些重要内容。

首先是覆写getCameraRequest()方法,指定根据自身需求预览的尺寸等。


    override fun getCameraRequest(): CameraRequest {
        return CameraRequest.Builder()
            .setPreviewWidth(1920) // camera preview width
            .setPreviewHeight(1080) // camera preview height
            .setRenderMode(CameraRequest.RenderMode.OPENGL) // camera render mode
            .setDefaultRotateType(RotateType.ANGLE_0) // rotate camera image when opengl mode
            .setAudioSource(CameraRequest.AudioSource.SOURCE_AUTO) // set audio source
            .setAspectRatioShow(true) // aspect render,default is true
            .setCaptureRawImage(false) // capture raw image picture when opengl mode
            .setRawPreviewData(false)  // preview raw image when opengl mode
            .create()
    }

savePath是拍照图片保存的路径,个将拍照的图片转化成Bitmap。其他后续流程和上述识别本地资源图片过程一致。

 
     /**
     * 识别拍照图片图片
     */
    private fun detectPicture() {
        detectBitmap = ImageProcessor.resizeBitmap(ImageProcessor.compressBitmap(savePath,50),1920,1080)
        mViewBinding.facesPic.setImageBitmap(detectBitmap)
        val images = ImageProcessor.convertToMat(detectBitmap)
        val detector = FaceDetectorYN.create(modelPath, "", images.size(), scoreThreshold, nmsThreshold, topK, backendId, targetId)
        detector.inputSize = images.size()
        val faces = Mat()
        detector.detect(images, faces)
        mViewBinding.detectResult.setFacesData(images, faces)
        mViewBinding.loading.visibility = View.GONE
    }
        

转化的过程需要注意一点的是,如果直接将生成的位图交给OpenCV识别,会提示一下错误信息。Number of input channels should be multiple of 3 but got 4:表示在卷积层中期望输入的通道数是3的倍数,但实际上输入的通道数为4,这导致了错误。

输入数据格式不正确:通常,图像数据在RGB模式下有3个通道(红、绿、蓝),而在RGBA模式下有4个通道(红、绿、蓝、透明度)。如果模型或网络期望的是RGB数据,提供的是RGBA数据,就会出现这个错误。

将生成的Bitmap通过Imgproc.cvtColor 转化为3通道的数据,这样OpenCV才能正常识别。


   fun convertToMat(bitmap: Bitmap?): Mat {
        val mat = Mat()
        Utils.bitmapToMat(bitmap, mat)
        Imgproc.cvtColor(mat, mat, Imgproc.COLOR_BGR2RGB)
        return mat
    }

摄像头拍照识别效果如下:

在这里插入图片描述


结尾

~~

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

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

相关文章

Matlab绘制像素风字母颜色及透明度随机变化动画

本文是使用 Matlab 绘制像素风字母颜色及透明度随机变化动画的教程 实现效果 实现代码 如果需要更改为其他字母组合&#xff0c;在下面代码的基础上简单修改就可以使用。 步骤&#xff1a;(1) 定义字母形状&#xff1b;(2) 给出字母组合顺序&#xff1b;(3) 重新运行程序&#…

15年以来 — 战略性云平台服务的演进路径之全面呈现(含亚马逊、微软和谷歌)

Gartner每年都发布对全球IaaS平台进行评估的魔力象限报告。2023年底&#xff0c;Gartner将此项评估的名称改为“战略性云平台服务”&#xff08;Strategic cloud platform services&#xff09;&#xff0c;尽管其核心仍为IaaS&#xff0c;但是&#xff0c;毫无疑问&#xff0c…

【深度学习】什么是深度学习?

1. 前言 深度学习是机器学习的一个分支&#xff0c;它使用神经网络教计算机做人类自然而然会做的事情&#xff1a;从示例中学习。在深度学习中&#xff0c;模型会学习直接从图像、文本或声音等数据中执行分类或回归任务。深度学习模型可以达到最先进的准确率&#xff0c;通常超…

WPF自定义控件

控件模板 顾名思义就是在原有的控件上进行模版修改成自己需要的样式 把ProgressBar修改为一个水液面的进度条 <Window x:Class"XH.CustomLesson.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://s…

树莓派3B升级glibc-2.29

自从我树莓派的python升级到3.9&#xff0c;每次import numpy都会报错&#xff1a;libm.so.6: version GLIBC_2.29 not found。这又是怎么回事呢&#xff1f;查了资料才知道&#xff0c;原来是我的respbian系统版本低的原因&#xff08;如图&#xff09;。 &#xff08;可以用l…

Facebook广告投放优化思路分享,为何总是低量级

为什么谷歌优化总是不起效果&#xff1f;今天我来谈谈我的Facebook广告优化思路&#xff0c;希望对你有所帮助。感兴趣的可以点赞、收藏。关注我&#xff0c;每天分享海外推广知识。 以下是Facebook广告优化思路思维导图&#xff1a; 那么&#xff0c;今天主要来谈谈量级低的情…

哪个工具可以ai续写免费?多维度评测分享

在创意的征途中&#xff0c;你是否曾遇到过文思枯竭的困境&#xff1f;当灵感如同夜空中的流星&#xff0c;一闪而逝&#xff0c;如何捕捉并延续那份璀璨&#xff1f;答案或许就藏在ai续写软件的无限可能中。 不知道ai续写软件哪个好&#xff1f;别急&#xff0c;今天&#xff…

Win10 禁止更新-【延长更新时间】

文章目录 操作注册表 操作注册表 Win R regedit 打开注册表 粘贴&#xff1a; ** \HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings 代开目录 新建32位数据值 &#xff1a;FlightSettingsMaxPauseDays 999 然后到更新界面&#xff0c;就可以一直点

数字信号处理3:数字滤波器设计

文章目录 前言一、实验目的二、实验设备三、实验内容四、实验原理五、实验方法及要求1.用脉冲响应不变法设计巴特沃斯数字滤波器2. 用双线性变换法设计切比雪夫数字滤波器3. 用双线性变换法设计巴特沃斯数字滤波器,并将直接型结构转换成级联型结构4. 数字低通滤波器特性比较5…

GD - EmbeddedBuilder_v1.4.1.23782工程中的gdc名称必须和工程名称一致

文章目录 GD - EmbeddedBuilder_v1.4.1.23782工程中的gdc名称必须和工程名称一致概述笔记总结END GD - EmbeddedBuilder_v1.4.1.23782工程中的gdc名称必须和工程名称一致 概述 在看EmbeddedBuilder_v1.4.1.23782自带的demo工程。 看到 D:\EmbeddedBuilder_v1.4.1.23782\exam…

工业一体机立式报工台助力工厂改善生产报工效率

在现代化工业生产中&#xff0c;生产效率和管理水平是企业竞争力的关键。为了实现高效的生产管理&#xff0c;工厂需要一个可靠、稳定、便捷的报工系统。传统的报工方式存在诸多弊端&#xff0c;例如效率低、易出错、数据难以统计分析等。而工厂软件报工台立式工业一体机的出现…

家里浮毛怎么去掉最高效?这些宠物空气净化器总有一款适合你

我家是三个月大的时候被我领回家的。它简直就是个小天使&#xff0c;乖得不得了&#xff0c;既不乱尿也不掉毛。每次去朋友家&#xff0c;看到他们为猫咪掉毛头疼不已&#xff0c;我就忍不住在心里偷偷乐&#xff0c;觉得自己真是捡到了宝&#xff0c;怎么会有这么省心的猫咪呢…

上海知名泌尿外科专家常态化坐诊黄山新晨医院,让前列腺癌看得更早、更准!

继7月28日上海第四人民医院泌尿外科专家在黄山新晨医院开展义诊之后&#xff0c;8月9日和10日&#xff0c;该团队领头人周铁教授又完成了合作以来的首次坐诊&#xff0c;标志着双方合作从此进入常态化阶段。 周铁主任在查看患者的检查报告 周铁主任曾任中华医学会泌尿外科分会…

拍立淘API:商品ID与标题的快速获取

拍立淘&#xff08;Pailitao&#xff09;是阿里巴巴旗下的一个基于图像识别技术的购物应用功能&#xff0c;用户可以通过拍照或上传图片来快速找到相似的商品。然而&#xff0c;拍立淘本身并不直接提供一个公开的API接口供开发者使用来直接获取商品ID和标题等信息。 如果你想通…

C++现代教程五

#pragma once _Pragma("once")# C/C混合编程 #ifdef __cplusplus extern "C" { #endif // 一些c代码 #ifdef __cplusplus } #endif# 继承构造 struct A {A(int i) {}A(double d,int i){}A(float f,int i,const char* c){}//...等等系列的构造函数版本 }&am…

protobuf 生成 error

一、简介 响应错误时可以直接使用 errors 包中的 New 方法来声明一个 error&#xff0c;也可以直接通过 proto 预定义定义错误码&#xff0c;然后通过 proto-gen-go 生成帮助代码&#xff0c;直接返回 error。 二、使用教程 2.1 错误定义 syntax "proto3"; impor…

深入理解Java中的LocalDateTime与ChronoUnit:精确时间处理的最佳实践

个人名片 &#x1f393;作者简介&#xff1a;java领域优质创作者 &#x1f310;个人主页&#xff1a;码农阿豪 &#x1f4de;工作室&#xff1a;新空间代码工作室&#xff08;提供各种软件服务&#xff09; &#x1f48c;个人邮箱&#xff1a;[2435024119qq.com] &#x1f4f1…

人工智能战略:如何实施人工智能解决方案以实现企业成功

人工智能 (AI) 不再是一个未来概念&#xff0c;而是改变全球各行各业的切实现实。 企业正在利用人工智能来提高效率、提高生产力并获得竞争优势。然而&#xff0c;实施人工智能解决方案需要明确的战略。 本指南将引导您完成成功将人工智能融入业务运营的关键步骤&#xff0c;…

局部场电位LFP

声明&#xff1a;本文章是根据网上资料&#xff0c;加上自己整理和理解而成&#xff0c;仅为记录自己学习的点点滴滴。可能有错误&#xff0c;欢迎大家指正。 神经科学最伟大的发现之一是人脑的电活动可以用附在头皮上的电极进行无创测量。脑电图&#xff08;Electroencephalog…

四种实用办法恢复回收站清空的文件!

想要恢复回收站的文件不小心点成清空回收站&#xff0c;如果你也遇上同样的状况&#xff0c;想在回收站找回清空的文件&#xff0c;可以尝试以下这几个恢复办法。 方法一&#xff1a;借用云存储/存储设备找回丢失文件 如果丢失的文件在其他云存储软件上&#xff08;如百度网盘…