自定义Android滑块拼图验证控件

自定义Android滑块拼图验证控件

      • 拼图认证视图
      • 默认策略
      • 工具类
      • 参考

1、继承自AppCompatImageView,兼容ImageView的scaleType设置,可设置离线/在线图片。
2、通过设置滑块模型(透明背景的图形块)设置滑块(和缺省块)样式,可修改缺省块颜色。
效果图

拼图认证视图

class PictureVerifyView @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = 0
) : AppCompatImageView(
    context, attrs, defStyleAttr
) {

    private var mState = STATE_IDEL //当前状态

    // right bottom 禁用
    private var piercedPositionInfo: RectF? = null //拼图缺块阴影的位置

    // right bottom 禁用
    private var thumbPositionInfo: RectF? = null //拼图缺块的位置
    private var thumbBlock: Bitmap? = null //拼图缺块Bitmap
    private var piercedBlock: Bitmap? = null
    private var thumbPaint: Paint? = null//绘制拼图滑块的画笔
    private var piercedPaint: Paint? = null//绘制拼图缺块的画笔
    private var startTouchTime: Long = 0 //滑动/触动开始时间
    private var looseTime: Long = 0 //滑动/触动松开时间
    private var blockSize = DEF_BLOCK_SIZE
    private var mTouchEnable = true //是否可触动
    private var callback: Callback? = null
    private var mStrategy: CaptchaStrategy? = null
    private var mMode = Captcha.MODE_BAR //Captcha验证模式
    private val xModeDstIn = PorterDuffXfermode(PorterDuff.Mode.DST_IN)
    private var isReversal = false
    private var middlewarePaint: Paint? = Paint()
    private val srcRect = Rect()
    private val dstRect = RectF()

    override fun onDetachedFromWindow() {
        mStrategy?.onDetachedFromWindow()
        thumbBlock?.recycle()
        piercedBlock?.recycle()
        thumbBlock = null
        thumbPaint = null
        piercedPositionInfo = null
        thumbPositionInfo = null
        callback = null
        piercedPaint = null
        middlewarePaint = null
        super.onDetachedFromWindow()
    }

    interface Callback {
        fun onSuccess(time: Long)
        fun onFailed()
    }

    private var tempX = 0f
    private var tempY = 0f
    private var downX = 0f
    private var downY = 0f

    init {
        setCaptchaStrategy(DefaultCaptchaStrategy(context))
    }

    private fun initDrawElements() {
        // 创建缺省镂空位置
        piercedPositionInfo ?: mStrategy?.getPiercedPosition(width, height, blockSize)
            ?.also {
                piercedPositionInfo = it
                thumbPositionInfo =
                    mStrategy?.getThumbStartPosition(width, height, blockSize, mMode, it)
            }

        // 创建滑块
        thumbBlock ?: createBlockBitmap().apply {
            thumbBlock = this
        }
    }

    private fun getBlockWidth() = if (isReversal) blockSize.height else blockSize.width
    private fun getBlockHeight() = if (isReversal) blockSize.width else blockSize.height

    private fun getRealBlockWidth() =
        getBlockWidth() + (mStrategy?.getThumbShadowInfo()?.size?.toFloat() ?: 0f)

    private fun getRealBlockHeight() =
        getBlockHeight() + (mStrategy?.getThumbShadowInfo()?.size?.toFloat() ?: 0f)

    /**
     * 生成拼图滑块和阴影图片
     */
    private fun createBlockBitmap(): Bitmap {
        // 获取背景图
        val origBitmap = getOrigBitmap()

        // 获取滑块模板
        val templateBitmap = getTempBitmap()

        if (blockSize.width != blockSize.height) {
            isReversal = templateBitmap.width == blockSize.height.toInt()
        }

        val resultBmp = Bitmap.createBitmap(
            getBlockWidth().toInt(),
            getBlockHeight().toInt(),
            Bitmap.Config.ARGB_8888
        )

        // 创建滑块画板
        middlewarePaint?.run {
            reset()
            isAntiAlias = true

            val canvas = Canvas(resultBmp)

            // 裁剪镂空位置
            val cropLeft = ((piercedPositionInfo?.left)?.toInt() ?: 0)
            val cropTop = ((piercedPositionInfo?.top)?.toInt() ?: 0)

            srcRect.set(
                cropLeft,
                cropTop,
                cropLeft + getBlockWidth().toInt(),
                cropTop + getBlockHeight().toInt()
            )
            dstRect.set(0f, 0f, getBlockWidth(), getBlockHeight())

            // 从原图上rect区间裁剪与画板上rectR区域重叠
            canvas.drawBitmap(
                origBitmap,
                srcRect,
                dstRect,
                this
            )
            srcRect.set(0, 0, getBlockWidth().toInt(), getBlockHeight().toInt())
            // 选择交集取上层图片
            xfermode = xModeDstIn
            // 绘制底层模板dst
            canvas.drawBitmap(
                templateBitmap,
                srcRect,
                dstRect,
                this
            )
        }

        return getRealThumbBitmap(resultBmp).apply {
            createPiercedBitmap(templateBitmap)
            origBitmap.recycle()
        }
    }

    // 获取缺省模板模型
    private fun getTempBitmap() = mStrategy?.getThumbBitmap(blockSize)
        ?: Utils.getBitmap(R.drawable.capt_def_puzzle, blockSize)

    private fun getOrigBitmap() =
        Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888).apply {
            val canvasOrig = Canvas(this)
            // 复原ImageView中显示操作 防止缺省位置错位
            canvasOrig.concat(imageMatrix)
            drawable.draw(canvasOrig)
        }

    /**
     * 设置带阴影的滑块
     */
    private fun getRealThumbBitmap(resultBmp: Bitmap) =
        mStrategy?.getThumbShadowInfo()?.run {
            Utils.addShadow(resultBmp, this)
        } ?: resultBmp

    /**
     * 获取滑块图片
     */
    private fun createPiercedBitmap(templateBitmap: Bitmap) {
        piercedBlock = (mStrategy?.piercedColor() ?: Color.TRANSPARENT).let {
            if (it == Color.TRANSPARENT) {
                templateBitmap
            } else {
                createColorBitmap(templateBitmap, it)
            }
        }
    }

    /**
     * 获取滑块模型形状的纯色图片
     */
    private fun createColorBitmap(templateBitmap: Bitmap, color: Int, isRecycle: Boolean = true) =
        Bitmap.createBitmap(
            getBlockWidth().toInt(),
            getBlockHeight().toInt(),
            Bitmap.Config.ARGB_8888
        ).apply {
            val c = Canvas(this)
            c.drawColor(color)
            middlewarePaint?.run {
                reset()
                xfermode = xModeDstIn
                // 从原图上rect区间裁剪与画板上rectR区域重叠
                c.drawBitmap(
                    templateBitmap,
                    srcRect,
                    dstRect,
                    this
                )
                if (isRecycle) {
                    templateBitmap.recycle()
                }
            }
        }

    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)
        initDrawElements()
        if (mState != STATE_ACCESS) {
            // 绘制缺块位置
            piercedPaint?.runWith(piercedPositionInfo, piercedBlock) { p, i, b ->
                if (mStrategy?.drawPiercedBitmap(canvas, p, i, b) != true) {
                    canvas.drawBitmap(b, i.left, i.top, p)
                }
            }
        }
        if (mState == STATE_MOVE || mState == STATE_IDEL || mState == STATE_DOWN || mState == STATE_UNACCESS) {
            // 绘制滑块
            thumbPaint?.runWith(thumbPositionInfo, thumbBlock) { p, i, b ->
                if (mStrategy?.drawThumbBitmap(canvas, p, i, b) != true) {
                    val offset = (mStrategy?.getThumbShadowInfo()?.size?.toFloat()) ?: 0f
                    canvas.drawBitmap(
                        b,
                        (i.left - offset).coerceAtLeast(0f),
                        (i.top - offset).coerceAtLeast(0f),
                        p
                    )
                }
            }
        }
    }

    private fun Paint.runWith(
        t: RectF?,
        bm: Bitmap?,
        block: (Paint, RectF, Bitmap) -> Unit
    ): Paint {
        return this.also { p ->
            t?.let { rect ->
                bm?.let { b ->
                    if (!b.isRecycled) {
                        block(p, rect, b)
                    }
                }
            }
        }
    }

    /**
     * 按下滑动条(滑动条模式)
     */
    fun down(progress: Int) {
        if (isEnabled) {
            startTouchTime = System.currentTimeMillis()
            mState = STATE_DOWN
            thumbPositionInfo?.left = progress / 100f * (width - getRealBlockWidth())
            invalidate()
        }
    }

    /**
     * 触动拼图块(触动模式)
     */
    private fun downByTouch(x: Float, y: Float) {
        if (isEnabled) {
            mState = STATE_DOWN
            thumbPositionInfo?.run {
                left = x - getRealBlockWidth() / 2f
                top = y - getRealBlockHeight() / 2f
            }
            startTouchTime = System.currentTimeMillis()
            invalidate()
        }
    }

    /**
     * 移动拼图缺块(滑动条模式)
     */
    fun move(progress: Int) {
        if (isEnabled) {
            mState = STATE_MOVE
            thumbPositionInfo?.left = progress / 100f * (width - getRealBlockWidth())
            invalidate()
        }
    }

    /**
     * 触动拼图缺块(触动模式)
     */
    private fun moveByTouch(offsetX: Float, offsetY: Float) {
        if (isEnabled) {
            mState = STATE_MOVE
            thumbPositionInfo?.run {
                left = (left + offsetX.toInt()).coerceAtMost(width - getRealBlockWidth())
                top = (top + offsetY.toInt()).coerceAtMost(height - getRealBlockHeight())
            }
            invalidate()
        }
    }

    /**
     * 松开
     */
    fun loose() {
        if (isEnabled) {
            mState = STATE_LOOSEN
            looseTime = System.currentTimeMillis()
            checkAccess()
            invalidate()
        }
    }

    /**
     * 复位
     */
    fun reset() {
        mState = STATE_IDEL
        thumbPositionInfo = null
        thumbBlock?.recycle()
        thumbBlock = null
        piercedBlock?.recycle()
        piercedBlock = null
        isReversal = false
        piercedPositionInfo = null
        invalidate()
    }

    fun unAccess() {
        mState = STATE_UNACCESS
        invalidate()
    }

    fun access() {
        mState = STATE_ACCESS
        invalidate()
    }

    fun callback(callback: Callback?) {
        this.callback = callback
    }

    fun setCaptchaStrategy(strategy: CaptchaStrategy) {
        mStrategy = strategy
        thumbPaint = strategy.thumbPaint
        piercedPaint = strategy.piercedPaint
        setLayerType(LAYER_TYPE_SOFTWARE, thumbPaint)
        if (!isInLayout) {
            invalidate()
        }
    }

    fun setBlockSize(size: SizeF) {
        blockSize = size
        reset()
    }

    fun setBitmap(bitmap: Bitmap?) {
        setImageBitmap(bitmap)
    }

    override fun setImageBitmap(bm: Bitmap?) {
        super.setImageBitmap(bm)
        reset()
    }

    override fun setImageDrawable(drawable: Drawable?) {
        super.setImageDrawable(drawable)
        reset()
    }

    override fun setImageURI(uri: Uri?) {
        super.setImageURI(uri)
        reset()
    }

    override fun setImageResource(resId: Int) {
        super.setImageResource(resId)
        reset()
    }

    fun setMode(@Captcha.Mode mode: Int) {
        mMode = mode
        isEnabled = true
        reset()
    }

    fun setTouchEnable(enable: Boolean) {
        mTouchEnable = enable
    }

    private fun getFaultTolerant() = (mStrategy?.getFaultTolerant()) ?: DEF_TOLERANCE

    /**
     * 检测是否通过
     */
    private fun checkAccess() {
        thumbPositionInfo?.let { info ->
            piercedPositionInfo?.run {
                val faultTolerant = getFaultTolerant()
                if (abs(info.left - left) < faultTolerant && abs(
                        info.top - top
                    ) < faultTolerant
                ) {
                    access()
                    callback?.onSuccess(looseTime - startTouchTime)
                } else {
                    unAccess()
                    callback?.onFailed()
                }
            }
        }
    }

    override fun dispatchTouchEvent(event: MotionEvent): Boolean {
        //触动模式下,点击超出拼图缺块的区域不进行处理
        thumbPositionInfo?.let {
            if (event.action == MotionEvent.ACTION_DOWN
                && mMode == Captcha.MODE_NONBAR
                && (event.x < it.left || event.x > it.left + getRealBlockWidth() || event.y < it.top || event.y > it.top + getRealBlockHeight())
            ) {
                return false
            }
        }
        return super.dispatchTouchEvent(event)
    }

    @SuppressLint("ClickableViewAccessibility")
    override fun onTouchEvent(event: MotionEvent): Boolean {
        if (mMode == Captcha.MODE_NONBAR && mTouchEnable && isEnabled) {
            thumbBlock?.run {
                val x = event.x
                val y = event.y
                when (event.action) {
                    MotionEvent.ACTION_DOWN -> {
                        downX = x
                        downY = y
                        downByTouch(x, y)
                    }

                    MotionEvent.ACTION_UP -> loose()
                    MotionEvent.ACTION_MOVE -> {
                        val offsetX = x - tempX
                        val offsetY = y - tempY
                        moveByTouch(offsetX, offsetY)
                    }
                }
                tempX = x:
                tempY = y
            }
        }
        return true
    }

    companion object {
        //状态码
        private const val STATE_DOWN = 1
        private const val STATE_MOVE = 2
        private const val STATE_LOOSEN = 3
        private const val STATE_IDEL = 4
        private const val STATE_ACCESS = 5
        private const val STATE_UNACCESS = 6
        internal const val DEF_TOLERANCE = 10 //验证的最大容差
        internal val DEF_BLOCK_SIZE = SizeF(50f, 50f) //验证的最大容差
    }
}

默认策略

class DefaultCaptchaStrategy(ctx: Context) : CaptchaStrategy(ctx) {
    private val degreesList = arrayListOf(0, 90, 180, 270)
    private val defBound: ShadowInfo =
        ShadowInfo(SizeUtils.dp2px(3.0f), Color.BLACK,SizeUtils.dp2px(2.0f).toFloat())

    // 滑块模型
    override fun getThumbBitmap(blockSize: SizeF): Bitmap {
        return Utils.getBitmap(
            R.drawable.capt_def_puzzle,
            blockSize,
            getDegrees()
        )
    }

    override fun getThumbShadowInfo() = defBound // 滑块阴影信息

    // 缺省位置
    override fun getPiercedPosition(width: Int, height: Int, blockSize: SizeF): RectF {
        val random = Random()
        val size =
            blockSize.width.coerceAtLeast(blockSize.height).toInt() + getThumbShadowInfo().size
        val left = (random.nextInt(width - size)
            .coerceAtLeast(size)).toFloat()
        val top = (random.nextInt(height - size)
            .coerceAtLeast(getThumbShadowInfo().size)).toFloat()
        return RectF(left, top, 0f, 0f)
    }

    private fun getDegrees(): Int {
        val random = Random()
        return degreesList[random.nextInt(degreesList.size)]
    }

    // 滑块初始位置
    override fun getThumbStartPosition(
        width: Int,
        height: Int,
        blockSize: SizeF,
        mode: Int,
        thumbPosition: RectF
    ): RectF {
        var left = 0f
        val top: Float
        val maxSize = blockSize.width.coerceAtLeast(blockSize.height).toInt()
        if (mode == Captcha.MODE_BAR) {
            top = thumbPosition.top
        } else {
            val random = Random()
            val size = maxSize + getThumbShadowInfo().size
            left = (random.nextInt(width - size)
                .coerceAtLeast(getThumbShadowInfo().size)).toFloat()
            top = (random.nextInt(height - size)
                .coerceAtLeast(getThumbShadowInfo().size)).toFloat()
        }
        return RectF(left, top, 0f, 0f)
    }

    override val thumbPaint: Paint
        get() = Paint().apply {
            isAntiAlias = true
        }

    override val piercedPaint: Paint
        get() = Paint().apply {
            isAntiAlias = true
        }

    override fun drawThumbBitmap(canvas: Canvas, paint: Paint, info: RectF, src: Bitmap): Boolean {
        return false
    }

    override fun drawPiercedBitmap(
        canvas: Canvas,
        paint: Paint,
        info: RectF,
        src: Bitmap
    ): Boolean {
        return false
    }

    // 缺省块颜色
    override fun piercedColor(): Int {
        return ResourcesUtils.getColor(R.color.black_a6)
    }

   // 验证可冗余空间
    override fun getFaultTolerant(): Int {
        return SizeUtils.dp2px(10.0f)
    }
}

工具类

object Utils {

    /**
     * 获取指定大小、指定旋转角度的图片
     */
    @JvmStatic
    fun getBitmap(@DrawableRes resId: Int, size: SizeF, degrees: Int = 0): Bitmap {
        val options = BitmapFactory.Options()
        options.inMutable = true
        val newWidth = size.width.toInt()
        val newHeight = size.height.toInt()
        return ImageUtils.scale(
            BitmapFactory.decodeResource(
                ResourcesUtils.getResources(),
                resId,
                options
            ), newWidth, newHeight, true
        ).let {
            if (degrees > 0) ImageUtils.rotate(
                it,
                degrees, newWidth / 2f, newHeight / 2f, true
            ) else it
        }
    }

    /**
     * 给图片添加阴影
     */
    @JvmStatic
    fun addShadow(
        srcBitmap: Bitmap,
        info: ShadowInfo
    ): Bitmap? {
        val w = 2 * info.size + info.dx.toInt()
        val h = 2 * info.size + info.dy.toInt()
        val dstWidth = srcBitmap.width + w
        val dstHeight = srcBitmap.height + h
        val mask = Bitmap.createBitmap(dstWidth, dstHeight, Bitmap.Config.ALPHA_8)
        val scaleToFit = Matrix()
        val src = RectF(0f, 0f, srcBitmap.width.toFloat(), srcBitmap.height.toFloat())
        val dst = RectF(
            info.size.toFloat(),
            info.size.toFloat(),
            dstWidth - info.size - info.dx,
            dstHeight - info.size - info.dy
        )
        scaleToFit.setRectToRect(src, dst, Matrix.ScaleToFit.CENTER)
        val dropShadow = Matrix(scaleToFit)
        dropShadow.postTranslate(info.dx, info.dy)
        val maskCanvas = Canvas(mask)
        val paint = Paint(Paint.ANTI_ALIAS_FLAG)
        maskCanvas.drawBitmap(srcBitmap, scaleToFit, paint)
        paint.xfermode = PorterDuffXfermode(PorterDuff.Mode.SRC_OUT)
        maskCanvas.drawBitmap(srcBitmap, dropShadow, paint)
        //设置阴影
        val filter = BlurMaskFilter(info.size.toFloat(), BlurMaskFilter.Blur.NORMAL)
        paint.reset()
        paint.isAntiAlias = true
        paint.color = info.color
        paint.maskFilter = filter
        paint.isFilterBitmap = true
        val ret = Bitmap.createBitmap(dstWidth, dstHeight, Bitmap.Config.ARGB_8888)
        val retCanvas = Canvas(ret)
        //绘制阴影
        retCanvas.drawBitmap(mask, 0f, 0f, paint)
        retCanvas.drawBitmap(srcBitmap, scaleToFit, null)
        mask.recycle()
        return ret
    }
}

参考

Android拼图滑块验证码控件:http://blog.csdn.net/sdfsdfdfa/article/details/79120665
关于android:绘制图像时绘制外部阴影:https://www.codenong.com/17783467/
Paint API之—— Xfermode与PorterDuff详解:https://www.kancloud.cn/kancloud/android-tutorial/87249

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

sql server 存储过程 set ansi_nulls set quoted_identifier,out 、output

SQL-92 标准要求在对空值(NULL) 进行等于 () 或不等于 (<>) 比较时取值为 FALSE。 当 SET ANSI_NULLS 为 ON 时&#xff0c;即使 column_name 中包含空值&#xff0c;使用 WHERE column_name NULL 的 SELECT 语句仍返回零行。即使 column_name 中包含非空值&#xff0c…

python列表笔记,python列表用法及基础操作

列表的介绍 定义100个变量&#xff0c;每个变量存放一个学生的姓名可行吗&#xff1f;有更好的办法吗&#xff1f; 答&#xff1a; 列表 一、列表的格式 定义列的格式&#xff1a;[元素1, 元素2, 元素3, ..., 元素n] 变量tmp的类型为列表 tmp [xiaoWang,180, 65.0] 列…

亿赛通电子文档安全管理系统任意文件上传漏洞复现

0x01 产品简介 亿赛通电子文档安全管理系统&#xff08;简称&#xff1a;CDG&#xff09;是一款电子文档安全加密软件&#xff0c;该系统利用驱动层透明加密技术&#xff0c;通过对电子文档的加密保护&#xff0c;防止内部员工泄密和外部人员非法窃取企业核心重要数据资产&…

【人工智能124种任务大集合】-集齐了自然语言处理(NLP),计算机视觉(CV),语音识别,多模态等任务

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下人工智能124种任务大集合&#xff0c;任务集合主要包括4大类&#xff1a;自然语言处理&#xff08;NLP&#xff09;、计算机视觉&#xff08;CV&#xff09;、语音识别、多模态任务。 我这里整理了124种应用场景任…

FPGA_学习_14_第一个自写模块的感悟和ila在线调试教程与技巧(寻找APD的击穿偏压)

前一篇博客我们提到了&#xff0c;如果要使用算法找到Vbr&#xff0c;通过寻找APD采集信号的噪声方差的剧变点去寻找Vbr是一个不错的方式。此功能的第一步是在FPGA中实现方差的计算&#xff0c;这个我们已经在上一篇博客中实现了。 继上一篇博客之后&#xff0c;感觉过了很久了…

.net连接mysql,提示找不到请求的 .Net Framework Data Provider。可能没有安装

开发完成的.net程序需要连接mysql数据库&#xff0c;在个人电脑上运行没问题&#xff0c;别人运行时提示“提示找不到请求的 .Net Framework Data Provider。可能没有安装”。经过查询&#xff0c;安装Connector/NET 8.1.0&#xff0c;下载地址如下所示&#xff1a; https://d…

AI抢饭碗!新闻集团将使用生成式AI,每周自动写3000篇新闻丨IDCF

作者&#xff1a;AIGC开放社区 8月1日&#xff0c;英国卫报消息&#xff0c;全球最大新闻媒体公司之一的新闻集团&#xff0c;将使用生成式AI每周自动创建3000篇澳大利亚本地新闻。 据悉&#xff0c;新闻集团在内部成立了一个名为“Data Local”的部门只有4名员工&#xff0c;…

c语言操作文件

1、文件存储 文件存取格式 从用户或者操作系统使用的角度&#xff08;逻辑上&#xff09;文件可以分为文本文件、二进制文件 文本文件&#xff1a;存储时&#xff0c;是将字符的ascii码值存在磁盘中&#xff0c;取的时候将ascii码翻译成对应的字符&#xff0c;这个翻译器就是…

ubuntu下gif动态图片的制作

Gif图片比视频小, 比静态JPG图片形象生动, 更适用于产品展示和步骤演示等。各种各样的gif动图为大家交流提供很大的乐趣. 这里简单介绍ubuntu系统下gif图的制作。 一、工具安装: kazam和ffmpeg kazam是linux下的一款简单但是功能强大的屏幕录制工具. 它可录制声音并选择全屏录…

【C++】一文带你初识C++继承

食用指南&#xff1a;本文在有C基础的情况下食用更佳 &#x1f340;本文前置知识&#xff1a; C类 ♈️今日夜电波&#xff1a;napori—Vaundy 1:21 ━━━━━━️&#x1f49f;──────── 3:23 …

数据库--SQL关键字的执行顺序

数据库相关链接&#xff1a; 数据库--数据类型&#xff1a;http://t.csdn.cn/RtqMD 数据库--三大范式、多表查询、函数sql&#xff1a;http://t.csdn.cn/udJSG 数据库--MySQL增删改查&#xff1a;http://t.csdn.cn/xkiti 一、一条sql语句通常包括&#xff1a; select fro…

[PaddlePaddle] [学习笔记] [上] 计算机视觉(卷积、卷积核、卷积计算、padding计算、BN、缩放、平移、Dropout)

1. 计算机视觉的发展历程 计算机视觉作为一门让机器学会如何去“看”的学科&#xff0c;具体的说&#xff0c;就是让机器去识别摄像机拍摄的图片或视频中的物体&#xff0c;检测出物体所在的位置&#xff0c;并对目标物体进行跟踪&#xff0c;从而理解并描述出图片或视频里的场…

pytest自动生成测试类 demo

一、 pytest自动生成测试类 demo # -*- coding:utf-8 -*- # Author: 喵酱 # time: 2023 - 08 -15 # File: test4.py # desc: import pytest import unittest# 动态生成测试类def create_test_class(class_name:str, test_cases:list) -> type:"""生成测试类…

根据源码,模拟实现 RabbitMQ - 通过 SQLite + MyBatis 设计数据库(2)

目录 一、数据库设计 1.1、数据库选择 1.2、环境配置 1.3、建库建表接口实现 1.4、封装数据库操作 1.5、针对 DataBaseManager 进行单元测试 一、数据库设计 1.1、数据库选择 MySQL 是我们最熟悉的数据库&#xff0c;但是这里我们选择使用 SQLite&#xff0c;原因如下&am…

JavaScript、TypeScript、ES5、ES6之间的联系和区别

ECMAScript&#xff1a; 一个由 ECMA International 进行标准化&#xff0c;TC39 委员会进行监督的语言。通常用于指代标准本身。JavaScript&#xff1a; ECMAScript 标准的各种实现的最常用称呼。这个术语并不局限于某个特定版本的 ECMAScript 规范&#xff0c;并且可能被用于…

看看安森美深力科NSI45090JDT4G 是如何点亮汽车内外照明系统解决方案

关于线性恒流调节器&#xff08;CCR&#xff09;&#xff1a;是一种用于控制电流的稳定输出。它通常由一个功率晶体管和一个参考电流源组成。CCR的工作原理是通过不断调节功率晶体管的导通时间来维持输出电流的恒定。当输出电流超过设定值时&#xff0c;CCR会减少功率晶体管的导…

Linux:shell脚本 正则表达式与AWK

一、正则表达式 由一类特殊字符及文本字符所编写的模式&#xff0c;其中有些字符&#xff08;元字符&#xff09;不表示字符字面意义&#xff0c;而表示控制或通配的功能&#xff0c;类似于增强版的通配符功能&#xff0c;但与通配符不同&#xff0c;通配符功能是用来处理文件…

八股文之框架篇(Spring Boot、SSM)

文章目录 Spring中的单例bean是线程安全的吗什么是AOP&#xff0c;项目中有没有使用到AOPSpring中的事务是如何实现的Spring中事务失效的场景有哪些Bean的生命周期Spring中的循环依赖&#xff08;循环引用&#xff09;SpringMVC的执行流程SpringBoot自动配置原理Spring、Spring…

生信豆芽菜-差异基因富集分析

网址&#xff1a;http://www.sxdyc.com/enrichmentEnrich 该工具使用R 语言的clusterProfiler包对关键基因集进行GO和KEGG富集分析&#xff0c;注意这个的关键基因集可以是差异基因&#xff0c;WGCNA的module基因&#xff0c;也可以是表型相关的基因集 1、数据准备 准备一个基因…

kubernetes企业级高可用部署

目录 1、Kubernetes高可用项目介绍 2、项目架构设计 2.1、项目主机信息 2.2、项目架构图 1、Kubernetes高可用项目介绍 2、项目架构设计 2.1、项目主机信息 2.2、项目架构图 2.3、项目实施思路 3、项目实施过程 3.1、系统初始化 3.2、配置部署keepalived服务 3.3、…