- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试实现在图像中查找图像的各种方法。我从严格匹配开始。让我们用 image
来表示我们要搜索的图像,用 bigImage
来表示我们要在中搜索的图像。
/** Finds image in bigImage by exact pixel match (all pixels must be exactly the same color).
*
* @param image the smaller image you want to find
* @param bigImage the big image you're searching in
* @return Rect object describing the location where the small image was found. Returns null if nothing was found.
*/
public static Rect findByExactMatch(BufferedImage image, BufferedImage bigImage) {
//I marked these final so that I don't accidentally change them later
final int iw = image.getWidth();
final int ih = image.getHeight();
final int bw = bigImage.getWidth();
final int bh = bigImage.getHeight();
//Loop from 0 to big image width/height MINUS the small image width/height
//The MINUS there is, because once you are at the end, the small image overlaps to undefined area
for(int rect_x=0, mrx=bw-iw; rect_x<mrx; rect_x++) {
for(int rect_y=0, mry=bh-ih; rect_y<mry; rect_y++) {
//This is where pixel looping begins
int x = 0;
int y = 0;
for (; x < iw; x++) {
for (; y < ih; y++) {
//Get RGB returns 0x00rrggbb
if(image.getRGB(x, y)!=
bigImage.getRGB(x+rect_x, y+rect_y)) {
//If the color does not match, break back to the rectangular search
//WITHOUT -1 THE VALUE OVERFLOWS ON NEXT ITERATION (damnit, debuged this like an idiot!!!)
y = x = Integer.MAX_VALUE-1;
break;
}
}
}
//This statement asks if the loop ended normally
// - otherwise, the x and y are MAX_INT and greater than iw
if(x==iw) {
return Rect.byWidthHeight(rect_x, rect_y, iw, ih);
}
}
}
//Nothing found - return null
return null;
}
正如你所看到的,函数非常简单,大部分代码都是注释。前两个循环移动我们正在比较的位置的帧,而内部循环则比较当前偏移处的小图像和大图像。
用法如下:
public static void main(String[] args)
{
//The small image to search for
BufferedImage thing = loadFromPath("thing.png");
//The big image to search in
BufferedImage screenshot = loadFromPath("screenshot.png");
if(thing!=null && screenshot!=null) {
Rect pos = autoclick.ScreenWatcher.findByExactMatch(thing, screenshot);
if(pos!=null) {
System.out.println("Found object: "+pos);
//Draw rectangle on discovered position
Graphics2D graph = screenshot.createGraphics();
graph.setColor(Color.RED);
graph.drawRect(pos.top, pos.left, pos.width, pos.height);
graph.dispose();
//Save the file for review
try {
ImageIO.write(screenshot, "png", new File("output.png"));
} catch (IOException ex) {
Logger.getLogger("wtf goes here?").log(Level.SEVERE, null, ex);
}
}
}
}
不幸的是,它似乎不起作用(红框显示程序识别的位置):
然而,当我向我的 friend 提示并想向他表明邪恶代码不起作用时,它突然起作用了:
需要注意的重要一点是,如果图像的第一列不是白色,它就可以工作。或者在我看来是这样。
我创建了一个测试项目,其中包含所有必需的可用文件:https://gist.github.com/Darker/f08b2fbf1795af9ebbe2 。默认情况下,它期望“thing.png”作为image
,“screenshot.png”作为“bigImage”。
最佳答案
您的内部匹配循环(x 和 y)编码错误;仔细看看。您在循环之前初始化两个循环变量,这意味着当第一列中没有不匹配时,y 已递增到 ih 的值,从而在 x 的所有后续传递中绕过 y 循环-循环。
解决此问题的一种方法是将 x 和 y 初始化移动到各自的 for 语句中(它们所属的位置),并在 for(rect_y...)
语句上放置一个标签。如果像素不匹配,请使用 Continueecty
中止 x 和 y 循环,而不是中断。这避免了操作 x/y 以及之后的人工检查的需要。
for (rect_x ...) {
label: for (rect_y...) {
for (x = 0; ...) {
for (y = 0; ....) {
if (mismatch) {
continue label;
}
}
}
return "match";
}
}
return null;
为了让事情变得不那么复杂,您还可以将内部两个循环重构为单独的方法,并在发生不匹配时简单地使用 return
来中断循环。
关于java - 为什么我在图像函数中查找图像匹配不完全匹配?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28175816/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!