2023-09-14
原文作者:王伟王胖胖 原文地址: https://blog.csdn.net/wangwei19871103/article/details/105563658

验证请求参数跟类方法匹配

我把Dog类方法的一些属性名改了:

202309142304141601.png
继续按方法的名字去掉前缀发送:

202309142304147922.png
结果可以:

202309142304157163.png
那我把方法名字改下,后面都加1

202309142304166834.png
结果这两个参数没绑定:

202309142304175575.png

结论

参数绑定跟方法的匹配,而且需要有set方法。比如setBirth,参数名字可以是Birth或者birth
CachedIntrospectionResults构造方法里会进行设置。

202309142304180646.png
判断可不可写的时候会获取:

202309142304193967.png
获取到之后会进行封装,封装成BeanPropertyHandler

202309142304204568.png
是否可读可写就是看有没有读方法和写方法:

202309142304219609.png

2023091423042264010.png
所以说这里属性是否可被绑定是跟属性set方法名字相关的,有兴趣的朋友可以调试下,里面还是比较深的。

绑定

参数名字修改完后就是这个样子,已经没有d.前缀了,其他的一些检查我就不说了,自己可以去看,我们接下去看怎么绑定到对象上的:

2023091423042359311.png

applyPropertyValues

2023091423042434812.png
就是遍历参数,然后设置:

2023091423042578813.png
最后到AbstractNestablePropertyAccessorprocessLocalProperty方法:
获取值,然后转换,最后赋值。

    private void processLocalProperty(PropertyTokenHolder tokens, PropertyValue pv) {
    		PropertyHandler ph = getLocalPropertyHandler(tokens.actualName);
    		...
    		Object oldValue = null;
    		try {
    			Object originalValue = pv.getValue();//获取参数值
    			Object valueToApply = originalValue;
    			if (!Boolean.FALSE.equals(pv.conversionNecessary)) {
    				if (pv.isConverted()) {
    					valueToApply = pv.getConvertedValue();
    				}
    				else {
    					if (isExtractOldValueForEditor() && ph.isReadable()) {
    						...
    							oldValue = ph.getValue();
    						...
    					//进行类型转换
    					valueToApply = convertForProperty(
    							tokens.canonicalName, oldValue, originalValue, ph.toTypeDescriptor());
    				}
    				pv.getOriginalPropertyValue().conversionNecessary = (valueToApply != originalValue);
    			}
    			ph.setValue(valueToApply);//设置值
    		}

最终还是调用反射set方法,传入参数:

2023091423042673714.png
看调用栈:

2023091423042745015.png
其实里面涉及了很多东西,太深入讲不完,又枯燥,现在知道是怎么绑定的就好了,简单的说就是 按照set方法名字进行请求参数的匹配,然后反射调用set方法传入参数来设置属性

好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。

阅读全文