每天挤一点,实现自己的selector函数(4)——表达式提取函数的实现

在(3)中说到对子表达式求值的步骤如下:

1、 输入子表达式字符串,如"div:last-child","ul","li.class",

2、 对字符串预处理,除去两端空格,除去二元运算符两边多余的空格,

3、 获得第一项非二元组表达式组,上述为:"div:last-child"

4、 将非二元组分解为表达式,上述为"div",":last-child"

5、 求出符合非二元组的元素集合ret

6、将剩余的字符串构造为二元组集合binExpr,即为二元运算符加非二元表达式组,也就是符合语法规则 binExpr = expr1 + op + expr2,因为expr1前面我们已经求过值了,所以现在就是根据op和expr2来过滤掉不符合条件的元素,而expr2可以重复利用第4步骤进行分解存储。

6、遍历二元组集合binExpr,根据运算符和expr2筛选符合条件的元素

7、返回ret

并且已经给出了XX.selector的实现,其中步骤1,2需要的函数都已实现,现在需要实现步骤3,4所需要的关键函数XX.getExprGroup,XX.getUnaryExprObj,同时考虑到扩展性,额外定义了三个对象表XX.UNAOP,XX.BINOP,XX.PSEOP,XX.ATTOP分别用来保存各个运算符的功能函数,在表达式对象中filter以及calculate方法都会委托到这些表中,大体框架如下:

/*一元运算符功能函数表*/
XX.UNAOP = {
	//ID
	'#':{
		calculate: function(){},
		filter: function(){}
	}, 
	//Class
	'.': {
		calculate: function(){},
		filter: function(){}
	},
	//Tag
	'':{
		calculate: function(){},
		filter: function(){}
	}
};
/*二元运算符功能函数表,二元运算符没有calculate,无法单独求出值*/
XX.BINOP = {
	'>' : {
		filter : function() {}
	},

	'+' : {
		filter : function() {}
	},

	' ' : {
		filter : function() {}
	}
};

//属性运算功能函数表
XX.ATTOP = {
	'':{
		filter:function(){},
		calculate:function(){}
	},
	'^=':{
		filter:function(){},
		calculate:function(){}
	}
	//待添加

}; 
//伪类运算功能函数表
XX.PSEOP = {
	'last-child':{
		filter:function(){},
		calculate:function(){}
	},
	'first-child':{
		filter:function(){},
		calculate:function(){}
	}
	//待添加
};

下面给出 XX.getExprGroup的实现,利用到了上面的一些表来进行操作符判断:主要思路已经在注释中:

/*顺序获得一组一元表达式字符串
 *参数selector:输入的字符串表达式
 * start: 开始位置
 * 返回值:表达式组字符串
 * */
XX.getExprGroup = function(selector, start){
	//str存储一元表达式字符串
	var str = '', len, curChar;
	start = start || 0;
	//逐字符扫描
	for( len = selector.length; start < len; ++start) {
		curChar = selector.charAt(start);
		//若遇到二元运算符,则一组一元表达式已经获得,直接返回,如"#id.class div [type]"
		if(XX.BINOP[curChar]) {//二元运算
			return str;
		} else if(curChar === '[') {// 属性运算
			//则需要截取整个"[type]"出来拼接
			var end = selector.indexOf(']', start);
			str += selector.substring(start, end + 1);
			start = end;
		} else { //否则直接拼接
			str += curChar;
		}
	}
	
	return str;
};

对于XX.getUnaryExprObj函数,这里就直接返回构造后的表达式对象。这里对表达式组进行了逆向扫描,主要考虑到,在表达式组由两个以上的表达式组成时,最后一个表达式不可能是标签运算,这样先计算得出的元素集合更小,方便后续过滤。

/*
 * 逆向扫描,获取一个表达式对象
 * 参数selector:选择符字符串
 * 参数curIndex:开始的搜索的索引位置
 * 返回值:无表达式则为null,否则返回一个对象,{expr:表达式对象, index:运算符的位置}
 * */

XX.getExprObj = function(selector, curIndex) {
	var str = '', 
		curChar = selector.charAt(curIndex),
		end,
		expr = null;
	
	for(; curIndex > -1; --curIndex) {
		curChar = selector.charAt(curIndex);
		if(']' === curChar) {//属性运算符
			end = selector.lastIndexOf('[', curIndex);
			expr = new XX.AttExpr(selector.substring(end+1, curIndex));
			return {expr:expr, index: end};
		}
		
		if(XX.UNAOP[curChar]){ //一元运算符
			return {expr: new XX.unaryExpr(curChar, str), index: curIndex};
		}
		
		if(':' === curChar ){ //伪类运算符
			return {expr: new XX.PseExpr(str), index: curIndex};
		}
		//标识符拼接
		str = curChar + str;
	}
	//边界情况,对应如:"div#id",扫描到div时,循环已经退出,这里需要判断是否第一个表达式为标签
	if(str != ''){ 
		return { expr : new XX.unaryExpr('', str), index : curIndex};
	}
	return null;
};




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
import cv2 as cv import numpy as np def fill_color_demo(image): copyImage = image.copy() h,w = image.shape[:2] mask = np.zeros([h+2,w+2],np.uint8) cv.floodFill(copyImage,mask,(30,30),(0,255,255),(100,100,100),(50,50,50),cv.FLOODFILL_FIXED_RANGE) cv.imshow("fill_color",copyImage) def fill_binary_demo(): image = np.zeros([400,400,3],np.uint8) image[100:300,100:300,:] =255 cv.imshow("fill_binary",image) mask = np.ones([402,402,1],np.uint8) mask[101:301,101:301] =0 cv.floodFill(image,mask,(200,200),(0,0,255),cv.FLOODFILL_MASK_ONLY) cv.imshow("filled_binary",image) src = cv.imread("E:/opencv/picture/me.jpg") cv.imshow("initial_window",src) face = src[50:350,100:300] gray = cv.cvtColor(face,cv.COLOR_BGR2GRAY) back_face = cv.cvtColor(gray,cv.COLOR_GRAY2BGR) src[50:350,100:300] = back_face cv.imshow("face_window",src) fill_color_demo(src) fill_binary_demo() cv.waitKey(0) cv.destroyAllWindows() 知识点: 1. 选取ROI感兴趣区域 src[ : , : ] = 像素矩阵 然后显示src图像就会出现ROI区域的像素值是像素矩阵的值。 容易混淆的。 h,w= src.shape[:2] ,获取图片的高度(x),获取图片的宽度(y) 2. 彩色图像的泛洪填充 def fill_color_demo(image): copyImage = image.copy() #复制图像 h,w = image.shape[:2] #读取图像的高,宽 mask = np.zeros([h+2,w+2],np.uint8) #新建一个mask矩阵,+2是官方要求必须是8位字节的 cv.floodFill(copyImage,mask,(30,30),(0,255,255)(100,100,100),(50,50,50),cv.FLOODFILL_FIXED_RANGE) cv.imshow("fill_color",copyImage) 关于floodFill(InputOutputArray image, Point seedPoint, Scalar newVal, Rect* rect=0, Scalar loDiff=Scalar(), Scalar upDiff=Scalar(), int flags=4 ) 通俗解释:floodFill( 1.操作的图像, 2.掩模mask, 3.漫水填充算法的起始位置,4.填充的颜色, 5.填充颜色的低值, 6.填充颜色的高值 ,7.填充的方法) 关于参数5:读取起始位置的像素值,并由起始位置的像素值-参数五的值即可得到漫水目标区域的下限值 关于参数6:读取起始位置的像素值,并由起始位置的像素值+参数六的值即可得到漫水目标区域的上限值 FLOODFILL_FIXED_RANGE -(既可以用于彩色也可用于灰度图像) 如果设置为这个标识符的话,就会考虑当前像素与种子像素之间的差,否则就考虑当前像素与其相邻像素的差。也就是说,这个范围是浮动的。 FLOODFILL_MASK_ONLY - (只用于灰度图像)如果设置为这个标识符的话,函数不会去填充改变原始图像 (也就是忽略第三个参数newVal), 而是去填充掩模图像(mask) 注意:mask表示操作掩膜,单通道,8位,长宽都比输入图像大2个像素点的图像。漫水填充不会填充掩膜mask的非零像素区域。(这个大小只能是size.height+2,size.width+2)! 如果是对灰度图像进行漫水操作,填充的颜色值new_value就会被忽略掉(但是也需要设置编写否则语法过不了关),flag部分写cv.FLOODFILL_MASK_ONLY。 3. 灰度图像的泛洪填充 4. image = np.zeros([400,400,3],np.uint8) image[100:300,100:300,:] =255 #建立一个图像矩阵,并将[100:300,100:300]像素值设置为255,即那一块是白色 cv.imshow("fill_binary",image) mask = np.ones([402,402,1],np.uint8) #建立mask矩阵,官方要求+2,必须是8位字节的 mask[101:301,101:301] =0#设置mask[101:301,101:301]位置的像素值为0 cv.floodFill(image,mask,(200,200),(0,0,255),cv.FLOODFILL_MASK_ONLY) #cv.FLOODFILL_MASK_ONLY:mask的指定的位置为零时才填充,不为零不填充。 #如果设置为cv.FLOODFILL_MASK_ONLY的话,函数不会去填充改变原始图像 (也就是忽略第三个漫水填充算法的起始位置), 而是去填充掩模图像mask cv.imshow("filled_binary",image) 知识盲区解答: 对ROI进行直接赋值像素值怎么做? 答:image[100:400,100:200] = np.zeros([300,100,3],np.uint8)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值