当前位置: 首页 > 编程日记 > 正文

使用 Python 和 OpenCV 构建 SET 求解器

60d5d2341b8226230acb56641e04189e.gif

作者 | 小白

来源 | 小白学视觉

小伙伴们玩过 SET 吗?SET 是一种游戏,玩家在指定的时间竞相识别出十二张独特纸牌中的三张纸牌(或 SET)的模式。每张 SET 卡都有四个属性:形状、阴影/填充、颜色和计数。下面是一个带有一些卡片描述的十二张卡片布局示例。

e08a5752ecdc6e147dd8cba4b4227fe5.png

带有一些卡片描述的标准十二张卡片布局

请注意,卡片的四个属性中的每一个都可以通过三个变体之一来表达。

因为没有两张牌是重复的,所以一副套牌包含 3⁴ = 81 张牌(每个属性 3 个变体,4 个属性)。一个有效的 SET 由三张卡片组成,对于四个属性中的每一个,要么全部共享相同的变量,要么都具有不同的变量。为了直观地演示,以下是三个有效 SET 示例:

28bca0ada1808d4f3e0fe9af91a4edcb.png

(1) 形状:全部不同 (2) 阴影:全部不同 (3) 颜色:全部不同 (4) 计数:全部相同

4dbc13bbb76a5ac657f0dd1ef7a9e867.png

(1) 形状:全部不同 (2) 阴影:全部相同 (3) 颜色:全部不同 (4) 计数:全部相同

73d8d7f7772fc60738a64290f3d2b618.png

(1) 形状:全部相同 (2) 阴影:全部不同 (3) 颜色:全部相同 (4) 计数:全部不同

构建一个 SET 求解器:一个计算机程序,该程序获取 SET 卡的图像并返回所有有效的 SET,我们使用 OpenCV(一个开源计算机视觉库)和 Python。为了时自己熟悉,我们可以浏览图书馆的文档并和观看一系列教程。此外,我们还可以阅读一些类似项目的博客文章和 GitHub 存储库。我们将项目分解为四项任务:

  1. 在输入图像中定位卡片 (CardExtractor.py)

  2. 识别每张卡片的唯一属性 (Card.py)

  3. 评估已识别的 SET 卡 (SetEvaluator.py)

  4. 向用户显示 SET (set_utils.display_sets)

我们为前三个任务中的每一个创建了一个专用类,我们可以在下面的类型提示 main 方法中看到。

import cv2# main method takes path to input image of cards and displays SETs
def main():input_image = 'PATH_TO_IMAGE'original_image = cv2.imread(input_image)extractor: CardExtractor = CardExtractor(original_image)cards: List[Card] = extractor.get_cards()evaluator: SetEvaluator = SetEvaluator(cards)sets: List[List[Card]] = evaluator.get_sets()display_sets(sets, original_image)cv2.destroyAllWindows()


在输入图像中定位卡片

1. 图像预处理

在导入OpenCV和Numpy(开源数组和矩阵操作库)之后,定位卡片的第一步是应用图像预处理技术来突出卡片的边界。具体来说,这种方法涉及将图像转换为灰度,应用高斯模糊并对图像进行阈值处理。简要地:

  • 转换为灰度可通过仅保留每个像素的强度或亮度(RGB 色彩通道的加权总和)来消除图像的着色。

  • 对图像应用高斯模糊会将每个像素的强度值转换为该像素邻域的加权平均值,权重由以当前像素为中心的高斯分布确定。这样可以消除噪声并 “平滑” 图像。经过实验后,我们决定高斯核大小设定 (3,3) 。

  • 阈值化将灰度图像转换为二值图像——一种新矩阵,其中每个像素具有两个值(通常是黑色或白色)之一。为此,使用恒定值阈值来分割像素。因为我们预计输入图像具有不同的光照条件,所以我们使用 cv2.THRESH_OTSU 标志来估计运行时的最佳阈值常数。

OpenCV 使这三个步骤变得很简单:

# Convert input image to greyscale, blurs, and thresholds using Otsu's binarization
def preprocess_image(image):greyscale_image = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)blurred_image = cv2.GaussianBlur(greyscale_image, (3, 3), 0)_, thresh = cv2.threshold(blurred_image, 0, 255, cv2.THRESH_OTSU)return thresh

def48c582de6d151367cedeb63e933e3.png  d71c3d453f4177e9525d6e217ab91e97.png  f4d555a612993f23f5878cbec1a7aa91.png

原始 → 灰度和模糊 → 阈

2. 查找卡片轮廓

接下来,我使用 OpenCV 的 findContours() 和 approxPolyDP() 方法来定位卡片。利用图像的二进制值属性,findContours() 方法可以找到 “ 连接所有具有相同颜色或强度的连续点(沿边界)的曲线。”² 第一步是对预处理图像使用以下函数调用:

contours, hierarchy = cv2.findContours(processed_image, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)

cv2.RETR_TREE 标志检索所有找到的轮廓以及描述给定轮廓嵌套或嵌入其他轮廓的级别的层次结构。cv2.CHAIN_APPROX_SIMPLE 标志仅通过编码轮廓端点来压缩轮廓信息。在进行了一些错误检查以排除非卡片之后,我们使用approxPolyDP ()方法使用轮廓端点来估计多边形曲线。以下是一些已识别的卡片轮廓,它们叠加在原始图像上。

9663dfd72e90e15f9265e9db7bd07c43.png

轮廓以绘制为红色

3. 重构卡片图像

识别轮廓后,必须重构卡片的边界以标准化原始图像中卡片的角度和方向。这可以通过仿射扭曲变换来完成,仿射扭曲变换是一种几何变换,可以保留图像上线条之间的共线点和平行度。我们可以在示例图像中看到下面的代码片段。

# Performs an affine transformation and crop to a set of card vertices
def refactor_card(self, bounding_box, width, height):bounding_box = cv2.UMat(np.array(bounding_box, dtype=np.float32))frame = [[449, 449], [0, 449], [0, 0], [449, 0]]if height > width:frame = [[0, 0],  [0, 449], [449, 449], [449, 0]]affine_frame = np.array(frame, np.float32)affine_transform = cv2.getPerspectiveTransform(bounding_box, affine_frame)refactored_card = cv2.warpPerspective(self.original_image, affine_transform, (450, 450))cropped_card = refactored_card[15:435, 15:435]return cropped_card

647589911fd9cd7a9743a10aec481bbf.png

仿射变换叠加在原始图像上以展示标准化的角度和方向

然后我们将每个重构的卡片图像及其坐标作为参数传递给 Card 类构造函数。这是构造函数的简化版本:

class Card:def __init__(self, card_image, original_coord):self.image = card_imageself.processed_image = self.process_card(card_image)self.processed_contours = self.processed_contours()self.original_coord = reorient_coordinates(original_coord) #standardize coordinate orientationself.count = self.get_count()self.shape = self.get_shape()self.shade = self.get_shade()self.color = self.get_color()


识别卡片属性

作为第一步,一种名为process_card的静态方法应用了上述相同的预处理技术,以及对重构后的卡片图像进行二进制膨胀和腐蚀。简要说明和示例:

  • 膨胀是其中像素 P 的值变成像素 P 的 “邻域” 中最大像素的值的操作。腐蚀则相反:像素 P 的值变成像素 P 的 “邻域” 中最小像素的值。

  • 该邻域的大小和形状(或“内核”)可以作为输入传递给 OpenCV(默认为 3x3 方阵)。

  • 对于二值图像,腐蚀和膨胀的组合(也称为打开和关闭)用于通过消除落在相关像素 “范围” 之外的任何像素来去除噪声。在下面的例子中可以看到这一点。

#Close card image (dilate then erode)
dilated_card = cv2.dilate(binary_card, kernel=(x,x), iterations=y)
eroded_card = cv2.erode(dilated_card, kernel=(x,x), iterations=y)

0f57be7eb387c2b6843efe4c0ffc16ad.pnge16966feb3e71508d26db8747db432d0.pngc6556a76c544e9ebc611a515f859e332.png

带有噪声的卡片 → 预处理后的图像 → 膨胀+腐蚀的“闭合”图像,注意噪声消除

我获取了生成的图像,并使用不同的方法从处理后的卡片中提取每个属性——形状、阴影、颜色和计数。我使用了 Github 上@piratefsh 的 set-solver 存储库中的代码来识别卡片颜色和阴影,并设计了我自己的形状和计数方法。

形状

  • 为了识别卡片上显示的符号的形状,我们使用卡片最大轮廓的面积。这种方法假设最大的轮廓是卡片上的一个符号——这一假设在排除非极端照明的情况下几乎总是正确的。

阴影

  • 识别卡片阴影或 “填充” 的方法使用卡片最大轮廓内的像素密度。

颜色

  • 识别卡片颜色的方法包括评估三个颜色通道 (RGB) 的值并比较它们的比率。

计数

  • 为了识别卡片上的符号数量,我们首先找到了四个最大的轮廓。尽管实际上计数从未超过三个,但我们选择了四个,然后进行了错误检查以排除非符号。在使用 cv2.drawContours 填充轮廓后,为了避免重复计算后,我们需要检查一下轮廓区域的值以及层次结构(以确保轮廓没有嵌入到另一个轮廓中)。

350098f311c68605332f2fed85d3db8e.pngb7538fe2eff0a50963d9e2dbb5054d72.png

填充原始符号以确保没有内部边界被视为轮廓。

另外:识别卡片属性的另一种方法可能是将有监督的 ML 分类模型应用于卡片图像。根据一些快速研究,似乎可以使用 Scikit 的 SVM 或 KNN 和 Keras ImageDataGenerator 来增强数据集。

然后每个变体都被编码为一个整数,这样任何卡片都可以用四个整数的数组表示。例如,带有两个空菱形符号的紫色卡片可以表示为 [1,1,3,2]。

现在卡片表示为数组,让我们评估一下 SET!

评估 SET

为了检查已识别卡片中的集合,将卡片对象数组传递给 SetEvaluator 类。

方法一:所有可能的组合

至少有两种方法可以评估卡的数组表示形式是否为有效集。第一种方法需要评估所有可能的三张牌组合。例如,当显示 12 张牌时,有 ₁₂C₃ =(12!)/(9!)(3!) = 660 种可能的三张牌组合。使用 Python 的 itertools 模块,可以按如下方式计算:

import itertools SET_combinations = list(combinations(cards: List[Card], 3))

请记住,对于每个属性,SET 中的三张卡片的变化必须相同或不同。如果三个卡片阵列彼此堆叠,则给定列/属性中的所有值必须显示全部相同的值或全部不同的值。

可以通过对该列中的所有值求和来检查此特性。如果所有三张卡片对于该属性具有相同的值,则根据定义,所得总和可被三整除。类似地,如果所有三个值都不同(即等于 1、2 和 3 的排列),则所得的总和 6 也可以被 3 整除。如果没有余数,这些值的任何其他总和都不能被3整除。

我们将这种方法应用于所有 660 种组合,保存了有效的组合。快看,我们有了我们的 SET!下面是一个简单演示此方法的代码片段(在可能的情况下不使用生成器尽早返回 False):

# Takes 3 card objects and returns Boolean: True if SET, False if not SET
@staticmethod
def is_set(trio):count_sum = sum([card.count for card in trio])shape_sum = sum([card.shape for card in trio])shade_sum = sum([card.shade for card in trio])color_sum = sum([card.color for card in trio])set_values_mod3 = [count_sum % 3, shape_sum % 3, shade_sum % 3, color_sum % 3]return set_values_mod3 == [0, 0, 0, 0]

但是有一个更好的方法......

方法 2:验证 SET Key

请注意,对于一副牌中的任意两张牌,只有一张牌(并且只有一张牌)可以完成 SET,我们称这第三张卡为SET Key。方法 1 的一种更有效的替代方法是迭代地选择两张卡片,计算它们的 SET 密钥,并检查该密钥是否出现在剩余的卡片中。在 Python 中检查 Set() 结构的成员资格的平均时间复杂度为 O (1)。

这将算法的时间复杂度降低到 O( n²),因为它减少了需要评估的组合数量。考虑到只有少量 n 次输入的事实(在游戏中有12 张牌在场的 SET 有 96.77% 的机会,15 张牌有 99.96% 的机会,16 张牌有 99.9996% 的机会⁴),效率并不是最重要的。使用第一种方法,我在我的中端笔记本电脑上对程序计时,发现它在我的测试输入上平均运行 1.156 秒(渲染最终图像)和 1.089 秒(不渲染)。在一个案例中,程序在 1.146 秒内识别出七个独立的集合。

向用户显示 SETS

最后,我们跟随 piratefsh 和 Nicolas Hahn 的引导,通过在原始图像上用独特的颜色圈出各自 SET 的卡片,向用户展示 SET。我们将每张卡片的原始坐标列表存储为一个实例变量,该变量用于绘制彩色轮廓。

# Takes List[List[Card]] and original image. Draws colored bounding boxes around sets.
def display_sets(sets, image, wait_key=True):for index, set_ in enumerate(sets):set_card_boxes = set_outline_colors.pop()for card in set_:card.boundary_count += 1expanded_coordinates = np.array(expand_coordinates(card.original_coord, card.boundary_count), dtype=np.int64)cv2.drawContours(image, [expanded_coordinates], 0, set_card_boxes, 20)

属于多个 SET 的卡片需要多个轮廓。为了避免在彼此之上绘制轮廓,expanded_coordinates() 方法根据卡片出现的 SET 数量迭代扩展卡片的轮廓。这是使用 cv2.imshow() 的操作结果:

c3fa1cd1d5037e0e312d50d6f9615ebb.png  71f622d165bc41a4849a83d0fb70771b.png  50d3ac5e169ea02e02ccade74f2afd3e.png

就是这样——一个使用 Python 和 OpenCV 的 SET 求解器!这个项目很好地介绍了 OpenCV 和计算机视觉基础知识。特别是,我们了解到:

  • 图像处理、降噪和标准化技术,如高斯模糊、仿射变换和形态学运算。

  • Otsu 的自动二元阈值方法。

  • 轮廓和 Canny 边缘检测。

  • OpenCV 库及其一些用途。

引文和资源

  • Piratefsh’s set-solver on Github was particularly informative. After finding that her approach to color identification very accurate, I ended up simply copying the method. Arnab Nandi’s card game identification project was also a useful starting point, and Nicolas Hahn’s set-solver also proved useful. Thank you Sherr, Arnab, and Nicolas, if you are reading this!

  • Here’s a basic explanation of contours and how they work in OpenCV. I initially implement the program with Canny Edge Detection, but subsequently removed it because it did not improve card identification accuracy for test cases.

  • You can find a more detailed description of morphological transformations on the OpenCV site here.

  • Some interesting probabilities related to the game SET.

e475b49c0332f8370721f5e59c12e106.gif

e8a0fdf725f4cd1cb9c4c7744988ac72.png

技术

6种常用的绘制地图的方法,码住!

资讯

DeepMind 打造AI游戏系统

资讯

全球首个活体机器人,能生娃

资讯

机器人Ameca苏醒瞬间逼真到令人...

1947f54fb8419a37da25cbbfc4d2ac19.png

分享

49a10e39929c0f18195fd7d019ddff32.png

点收藏

f089bca3992f30e73b5c1a5bd278f6cf.png

点点赞

8e3b7e9eea70bb350ded5defc380d3b6.png

点在看

相关文章:

Delphi XE5 常用功具与下载

1.Delphi XE5 正式版http://altd.embarcadero.com/download/radstudio/xe5/delphicbuilder_xe5_win.isohttp://altd.embarcadero.com/download/radstudio/xe5/delphicbuilder_xe5_upd1_win.iso2. cnpack 助手工具http://www.cnpack.org/download/unstable/CnWizards_1.0.1.665_…

maven学习(4)-Maven 构建Web 项目

紧接着上一节(3)&#xff0c;现在maven新建web项目&#xff0c;user-web。模拟一个用户登录的需求&#xff1a; 工程结构&#xff1a; pom.xml: <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance&qu…

如何查看linux版本

1. 查看内核版本命令&#xff1a; 1) [rootq1test01 ~]# cat /proc/version Linux version 2.6.9-22.ELsmp (bhcompilecrowe.devel.redhat.com) (gcc version 3.4.4 20050721 (Red Hat 3.4.4-2)) #1 SMP Mon Sep 19 18:00:54 EDT 2005 2) [rootq1test01 ~]# uname -a …

存储过程由结构表生成表

结构表 CREATE TABLE JGTB5001( ZDM VARCHAR2(30 BYTE), HZM VARCHAR2(100 BYTE), LX VARCHAR2(50 BYTE), JD VARCHAR2(20 BYTE), WBKLX VARCHAR2(100 BYTE), FUNCTIONNAME VARCHAR2(50 BYTE), FUNCTIONPARAMETER VARCHAR2(50 BYTE)); 生成的TB表CREATE OR REPLACE PROCEDURE P…

好礼相送|CSDN云原生 Meetup 成都站报名热烈启动,12.18见!

伴随着容器、Kubernetes及微服务等技术热度的持续攀升&#xff0c;云原生正以不可撼动之势&#xff0c;剑指云计算的下一个十年。12月18日&#xff0c;CSDN将在成都举办第三场云原生线下Meetup。在这里&#xff0c;您可以了解各大领先企业的云原生落地实践&#xff0c;与众多云…

vue-music 音乐网站

在学习完vueJS,一直想做个项目来锻炼一下,选来选去&#xff0c;还是做个网易云音乐&#xff0c;其间遇到了很多坑,也逐渐接受了vue这种组件化的思想以及从Dom操作转换为用数据去驱动视图。并且在某部分基础组件上借鉴(搬运)了elementUI的源码(不过elementUI写的是真好) 技术栈 …

shell环境变量

shell环境变量 环境变量 还记得上一章里面﹐我曾经提到过﹕当我们登入系统的时候﹐首先就获得一 shell﹐而且它也占据一个行程&#xff08;进程&#xff09;﹐然后再输入的命令都属于这个 shell 的子程序&#xff08;子进程&#xff09;。如果您学习够细心﹐不难发现我们的 sh…

apache用户认证

先创建一个“用户认证”目录&#xff08;设为abc&#xff09;[rootLAMPLINUX ~]# cd /data/www[rootLAMPLINUX www]# mkdir abc进入abc目录[rootLAMPLINUX www]# cd abc拷贝一个文件&#xff08;作用&#xff1a;验证配置是否生效&#xff09;[rootLAMPLINUX abc]# cp /etc/pas…

20个经典函数细说 Pandas 中的数据读取与存储,强烈建议收藏

作者 | 俊欣来源 | 关于数据分析与可视化大家好&#xff0c;今天小编来为大家介绍几个Pandas读取数据以及保存数据的方法&#xff0c;毕竟我们很多时候需要读取各种形式的数据&#xff0c;以及将我们需要将所做的统计分析保存成特定的格式。我们大致会说到的方法有&#xff1a;…

fastlane自动打包--详细介绍

fastlane--Packaging 自动化打包&#xff0c;通过fastlane自动发布Fastlane安装不在这里详细罗列&#xff0c;参照一下链接流程 https://www.jianshu.com/p/0a113f754c09操作步骤 1.检查Fastlane是否正确安装。输入以下命令&#xff1a; fastlane --version 复制代码可以看到Fa…

【Big Data】HADOOP集群的配置(一)

Hadoop集群的配置&#xff08;一&#xff09; 摘要: hadoop集群配置系列文档&#xff0c;是笔者在实验室真机环境实验后整理而得。以便随后工作所需&#xff0c;做以知识整理&#xff0c;另则与博客园朋友分享实验成果&#xff0c;因为笔者在学习初期&#xff0c;也遇到不少问题…

C语言 条件编译详解

预处理过程扫描源代码&#xff0c;对其进行初步的转换&#xff0c;产生新的源代码提供给编译器。可见预处理过程先于编译器对源代码进行处理。在C 语言中&#xff0c;并没有任何内在的机制来完成如下一些功能&#xff1a;在编译时包含其他源文件、定义宏、根据条件决定编译时是…

凝聚406万开发者 飞桨十大发布提速产业智能化

12月12日&#xff0c;由深度学习技术及应用国家工程实验室主办的WAVE SUMMIT2021深度学习开发者峰会在上海召开。百度首席技术官、深度学习技术及应用国家工程实验室主任王海峰公布飞桨最新成绩单&#xff1a;凝聚406万开发者、创建47.6万模型、服务15.7万企事业单位&#xff0…

环境变量,cp,mv,查看文档命令

2019独角兽企业重金招聘Python工程师标准>>> 一、环境变量PATH echo $PATH 打印当前的环境变量 PATH$PATH:路径 自定义环境变量 which查找某个命令的绝对路径&#xff0c;也可以查看某个命令的别名&#xff0c;which查找的范围就在PATH下的几个目录下查找&#xff1…

Linux中errno使用

当linux中的C api函数发生异常时,一般会将errno变量(需include errno.h)赋一个整数值,不同的值表示不同的含义,可以通过查看该值推测出错的原因&#xff0c;在实际编程中用这一招解决了不少原本看来莫名其妙的问题。但是errno是一个数字&#xff0c;代表的具体含义还要到errno.…

工程师文化:BAT 为什么不喊老板

BAT员工之间不喊老板&#xff0c;也不喊真名&#xff0c;而是用同学、花名&#xff0c;这是虚情假意&#xff1f;还是弘扬武侠文化&#xff1f;还是另有隐情&#xff1f;为什么欧美公司不这么做&#xff1f;本文将带大家走进科学&#xff0c;探索真相。 BAT 的称呼方式 腾讯&am…

SVN常见问题

2019独角兽企业重金招聘Python工程师标准>>> 目录[隐藏] 1. 提示SVN证书过期&#xff1f; 2. 用户名密码校验失败&#xff1f; 3. SVN提交文件时提示文件冲突怎么办&#xff1f; 4. SVN提交文件时提示失败&#xff1f; 1. 提示SVN证书过期&#xff1f; 问题描述&…

2017海克斯康拉斯维加斯美国大会 精彩即将开始

海克斯康集团与遍及全球行业用户的故事已经证明&#xff0c;海克斯康先进的解决方案影响着世界各行各业的发展&#xff0c;并为他们带来了颠覆性的科技变革...... 通过海克斯康集团与遍及全球行业用户的故事&#xff0c;已经证明海克斯康先进的解决方案影响着世界各行各业的发展…

Linux环境编程--waitpid与fork与execlp

waitpidwaitpid(等待子进程中断或结束)表头文件#include<sys/types.h>#include<sys/wait.h>定义函数 pid_t waitpid(pid_t pid,int * status,int options);函数说明waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用 wait()时子进程已经结…

C# 批处理制作静默安装程序包

使用批处理WinRAR制作静默安装程序包 echo 安装完窗口会自动关闭&#xff01;&#xff01;&#xff01; echo off start /wait Lync.exe /Install /Silent start /wait vcredist_x86/vcredist_x86.exe /q /norestart start /wait DotNetFx40/dotNetFx40_Full_x86_x64.exe /q /…

程序员是复制粘贴的工具人?还是掌握“谜底”的魔术师?

作者 | David Heinemeier Hansson译者 | 弯月出品 | CSDN&#xff08;ID&#xff1a;CSDNnews&#xff09;编程世界在经历了“Imposter Syndrome(冒充者症候群/负担症候群&#xff09;”和“gatekeeping&#xff08;守门人理论&#xff09;”两方的激战之后&#xff0c;最终以“…

Josephus Problem的详细算法及其Python, Java语言的实现

笔者昨天看电视&#xff0c;偶尔看到一集讲述古罗马人与犹太人的战争——马萨达战争&#xff0c;深为震撼&#xff0c;有兴趣的同学可以移步&#xff1a;http://finance.ifeng.com/a/20170627/15491157_0.shtml . 这不仅让笔者想起以前在学数据结构时碰到的Josephus问题&a…

SlightPHP

SlightPHP是一个轻量级的php框架&#xff0c;支持php5&#xff0c;和php模块方式使用&#xff0c;和apc使用性能更高&#xff01;项目地址&#xff1a;http://code.google.com/p/slightphp/源码地址&#xff1a;http://slightphp.googlecode.com/svn/trunk/你有两种方法使用Sli…

bzoj1178

题目&#xff1a;http://www.lydsy.com/JudgeOnline/problem.php?id1178 看ppthttp://wenku.baidu.com/link?urldJv6LNme7syiLGM-TzbEEKXwx36JWEnI5HFrIlzfmzUXXg4HG8FDggj5WQS3EKL3k3p-sUYeJ268jCvN4t_kq2YPo3I4GXvaGulQjXrO3d7#include<cstdio> #include<cstdlib&…

编程能力差,学不好Python、AI、Java等技术,90%是输在了这点上!

据了解&#xff0c;超90%的人在学习Python、Java、AI等技术时&#xff0c;都是在网上随便找个入门的教程就开始学起来。然而多数人在看了不少教程后&#xff0c;还是很难独立完成项目&#xff0c;甚至反思自己为什么学了这么久编程能力还是这么差&#xff01;因为你在刚刚开始学…

cglib代理的使用

一、什么是CGLIB? 总的来说&#xff0c;无论是cglib、jdk动态代理又或者是aop面向切面编程&#xff0c;都运用到了一个最重要的设计模式--代理模式&#xff01;万变不离其终&#xff0c;学好代理模式&#xff0c;打遍天下无敌手&#xff01; cglib就是一个字节码生成和转换的库…

使用PHP+Sphinx建立高效的站内搜索引擎

1. 为什么要使用Sphinx假设你现在运营着一个论坛&#xff0c;论坛数据已经超过100W&#xff0c;很多用户都反映论坛搜索的速度非常慢&#xff0c;那么这时你就可以考虑使用Sphinx了&#xff08;当然其他的全文检索程序或方法也行&#xff09;。2. Sphinx是什么Sphinx由俄…

9个必知的 Python 操作文件/文件夹方法

作者 | 欣一来源 | Python爱好者集中营近几年随着Python的热度不断上涨&#xff0c;人们渐渐使用这门编程语言来进行一些自动化操作&#xff0c;以节省重复劳动带来的效率低下&#xff0c;那么必定会涉及到对文件系统的操作&#xff0c;包括文件的增、删、改、查等等&#xff0…

Get/POST方法提交的长度限制

&#xfeff;&#xfeff;1. Get方法长度限制 Http Get方法提交的数据大小长度并没有限制&#xff0c;HTTP协议规范没有对URL长度进行限制。这个限制是特定的浏览器及服务器对它的限制。 如&#xff1a;IE对URL长度的限制是2083字节(2K35)。 下面就是对各种浏览器和服务器的…

Bitmap上下合成图片

合成两张图片&#xff0c;上下叠加的效果&#xff1a; /*** 把两个位图覆盖合成为一个位图&#xff0c;以底层位图的长宽为基准** param backBitmap 在底部的位图* param frontBitmap 盖在上面的位图* return*/public static Bitmap mergeBitmap(Bitmap backBitmap, Bitmap fr…