编码踩坑——记一次fastjson引发的空指针问题、引用标识$ref

news2024/11/24 21:07:43

本篇介绍在使用fastjson过程中遇到的一个问题,从而引申出使用fastjson时的注意事项——(1)尽量避免在实体中定义 get 开头的方法;(2)避免较深的实体字段层级;(3)避免实体字段之间的引用关系;

问题描述

直接现象:某个预期不为null的字段在调用get方法时抛出了NPE;该字段是通过对存于数据库的某个放JSONString的字段,执行 JSON.parseObject方法 反序列化成Java对象;

根据字段来源,找到原始的数据库存放JSONString的字段,发现字段中多了一个属性值,并且在JSON字符串中发现了带有$符号的KV,将该JSON字符格式化,如下:

{
    "allTicketList": [
        {
            "ticketAmount": 5000,
            "ticketId": 119
        }
    ],
    "strategy": "A",
    "loginStrategy": {
        "loginActName": "loginStrategyA",
        "rangeVOList": [
            {
                "days": 7,
                "ticketsInRange": [
                    {
                        "$ref": "$.allTicketList[0]"
                    }
                ]
            }
        ]
    }
}

上图中,allTicketList非预期中的字段,且ticketsInRange的属性值中出现了"$ref": "$.allTicketList[0]"这类变量符号;

实体类的定义如下:

@Data
class StrategyVO {
    private String strategy;
    private LoginStrategyVO loginStrategy;

    public List<TicketVO> getAllTicketList() {
        final List<TicketVO> allTickets = this.getLoginStrategy().getRangeVOList()
                .stream()
                .map(LoginRangeVO::getTicketsInRange)
                .flatMap(Collection::stream)
                .collect(Collectors.toList());
        return allTickets;
    }
}

@Data
class LoginStrategyVO {
    private String loginActName;
    private List<LoginRangeVO> rangeVOList;
}

@Data
class LoginRangeVO {
    private Integer days;
    private List<TicketVO> ticketsInRange;
}

@Data
class TicketVO {
    private Long ticketId;
    private Long ticketAmount;
}

可见,类StrategyVO的属性中存在多层嵌套,且根据领域模型的思想,在该类中定义了getAllTicketList方法(获取属性中的所有TicketVO对象);

原因初步分析

根据上述的条件,可知:

(1)观察到getAllTicketList方法与JSON字符串中多出来的allTicketList字段存在某种关联,即如果StrategyVO类有allTicketList这个字段,那么它的getter方法就是getAllTicketList();

(2)观察getAllTicketList方法的逻辑,其值实际上就是StrategyVO对象的属性中的引用,例如 getAllTicketList()[0] -> StrategyVO#loginStrategy#rangeVOList[0]#ticketsInRange[0];

现在存在两个疑问

1. fastjson为什么通过getAllTicketList方法推断StrategyVO对象有一个allTicketList属性?

2. 为什么当对象中存在相同引用的属性值时,除第一个引用,后面的会被类似"$ref": "$.allTicketList[0]"的KV替代?

问题1:FastJson如何判断有对象有哪些属性?

为了解决这个问题,需要debug一下JSON的源码,这里选择 JSON.toJsonString方法;以下是debug栈,这里只截图出关键代码;

1. com.alibaba.fastjson.serializer.JSONSerializer#write(java.lang.Object)

2. com.alibaba.fastjson.serializer.SerializeConfig#getObjectWriter(java.lang.Class<?>, boolean)

getObjectWriter方法是流程中的比较重要的方法之一,根据Java对象打印出其JSON字符串就是靠这里返回的 ObjectSerializer对象调用write方法获得的;

3. com.alibaba.fastjson.serializer.SerializeConfig#createJavaBeanSerializer(java.lang.Class<?>)

4. com.alibaba.fastjson.util.TypeUtils#buildBeanInfo(java.lang.Class<?>, java.util.Map<java.lang.String,java.lang.String>, com.alibaba.fastjson.PropertyNamingStrategy, boolean)

这里的FieldInfo类中的属性包括:字段名、类型、顺序、取值方法等,打印json时就是靠每个属性对应的这个类的对象来获取其json值,重点看一下这里的 computeGetters方法

5. com.alibaba.fastjson.util.TypeUtils#computeGetters(java.lang.Class<?>, com.alibaba.fastjson.annotation.JSONType, java.util.Map<java.lang.String,java.lang.String>, java.util.Map<java.lang.String,java.lang.reflect.Field>, boolean, com.alibaba.fastjson.PropertyNamingStrategy)

这里省略的蓝色部分包含了对fastson注解JSONField的解析:

走到这里可以看到,实际是通过"get"前缀匹配方法名的:

6. com.alibaba.fastjson.util.TypeUtils#getPropertyNameByMethodName

至此,就可以知道,FastJson并不是根据对象的field来确定json属性的,而是根据get为前缀的方法名来计算json属性名的

因此,在我们上述的例子中,由于getAllTicketList()方法的前缀是"get",因此走一遍FastJson的解析json属性的逻辑,"误算"出来了一个allTicketList属性

问题2:关于FastJson中的"$ref"符号?

(1)什么时候会出现"$ref"符号?

先说结论,JSON字符串中出现$ref: "$.list[2]"的原因是因为循环引用或内存对象重复

(2)FastJson中定义的重复/循环引用?

简单说,重复引用就是一个集合/对象中的多个元素/属性同时引用同一对象;循环引用就是集合/对象中的多个元素/属性存在相互引用导致循环;

(3)为什么要用$ref引用而不是直接使用真实的变量值?

重复引用的情况比较好理解,使用$ref替换真实的对象值,可以很大程度减小JSON字符串的空间,反序列化时只需要根据解析规则找到对应的重复变量即可;

存在循环引用时,会触发StackOverflowError异常;例如,变量A和B互相持有彼此的引用,当序列化引擎解析A时,它发现这个对象持有一个B的引用,转而去解析B;解析B时,发现他又持有A的引用,又转回解析A;如此产生StackOverflowError异常;为了避免StackOverflowError异常,fastjson会对引用进行检测,如果检测到存在重复/循环引用的情况,FastJson默认会以"引用标识"代替同一对象,从而避免继续循环解析变量的值导致StackOverflowError

(4)关闭FastJson引用检测的方法?

出现上述"$ref"符号的原因,正是FastJson开启了引用检测;查看FastJson的API,关闭FastJson的引用检测有局部关闭和全局关闭两种方式:

(1)局部关闭;就是在调用fastjson的地方指明使用的序列化器的特性,如下:

JSON.toJSONString(object, SerializerFeature.DisableCircularReferenceDetect);

(2)全局关闭;修改全局配置中的默认序列化器的特性,如下:

public class FastJsonHttpMessageConverterEx extends FastJsonHttpMessageConverter{
    public FastJsonHttpMessageConverterEx(){
        //在这里配置fastjson特性(全局设置的)
        FastJsonConfig fastJsonConfig = new FastJsonConfig();
        fastJsonConfig.setDateFormat("yyyy-MM-dd HH:mm:ss");    //自定义时间格式
        fastJsonConfig.setSerializerFeatures(SerializerFeature.WriteMapNullValue);  //正常转换null值
        fastJsonConfig.setSerializerFeatures(SerializerFeature.DisableCircularReferenceDetect);     //关闭循环引用
        this.setFastJsonConfig(fastJsonConfig);
    }

    @Override
    protected boolean supports(Class<?> clazz) {
        return super.supports(clazz);
    }
}

(5)关闭FastJson的“循环引用检测”特性并不能解决问题!

全局关闭这种方式是不推荐的,因为你无法预知在调用JSON.toJSONString方法的哪些地方会出现问题,如果一定要使用优先使用局部关闭的方式;

但是,尽管FastJson提供了SerializerFeature.DisableCircularReferenceDetect这个序列化选项,用来关闭引用检测;关闭引用检测后,重复引用对象时就不会被$ref代替,但是在循环引用时也会导致StackOverflowError异常

避免重复/循环引用的正确姿势

(1)在编码时,使用新对象为集合或对象赋值,而非使用同一对象;

不要在多处引用同一个对象,这可以说是一种java编码规范;不仅有重复引用问题,这个对象既然被多个对象引用,那么它可能在多个业务场景都有可能被修改,因此需要时刻注意;建议使用深度clone,如Spring中自带的BeanUtil或一些二方包工具,做对象的深拷贝而非仅复制引用,创建新的对象,把原对象的属性值复制给新对象;

关于循环引用,这种逻辑本身就不常见,可能在特殊的设计模式中需要维护两个对象之间的关联;自己编码时需要尽量避免,可以转换成第三个对象持有这2个对象的引用;

(2)尽量不要关闭FastJson的引用检测来避免显示$ref;

引用检测是FastJson提供的一种避免运行时异常的优良机制,如果为了避免在重复引用时显示$ref而关闭它,会有很大可能导致循环引用时发生StackOverflowError异常;这也是FastJson默认开启引用检测的原因;

真正要解决的应该是如何避免重复引用/循环引用;

(3)合理的使用 @JSONField(serialize=false) 来关闭对部分字段的序列化

如果前端用不到实体中的某个属性,建议在该属性的get方法上加上注解属性@JSONField(serialize=false),这样该属性就不会被序列化出来;这个可以解决重复引用问题,并且本文出现的问题就是这么解决的;

(4)避免在实体中定义以 "get" 开头的public方法;

通过上述的FastJson源码分析,可知FastJson是通过getter来解析对象中的属性的;当自己定义了一个以 "get" 开头的方法,如上面提到的StrategyVO#getAllTicketList方法,那么FastJson就会"误算"出一个StrategyVO#allTicketList属性;

FastJson引发问题的原因解释及解决方法

条件1:StrategyVO类中定义了 "get" 开头的方法StrategyVO#getAllTicketList;

导致的结果就是对象转成JSON字符串时多了个"预料之外的allTicketList属性";

条件2:getAllTicketList方法中的逻辑是返回了子属性中的引用,即触发了"FastJson重复引用"的条件;

导致的结果就是在JSON中打出了allTicketList属性的值后,对于StrategyVO的子属性,直接使用了$ref引用符——"$ref": "$.allTicketList[0]";

条件3:"allTicketList属性"的顺序在StrategyVO#loginStrategy#rangeVOList#ticketsInRange前面;

FastJson默认按照属性名排序,如果StrategyVO#loginStrategy的属性值先解析,那么打出来JSON字符串中,"allTicketList属性"的属性值应该是带"$ref"的引用;

条件4:环境中有2个版本的代码,版本A代码中StrategyVO类有getAllTicketList方法,而版本B代码中StrategyVO类没有getAllTicketList方法;

尽管触发"FastJson重复引用"检测时,JSON字符串会出现"$ref"符号,这样的JSON字符串其实是可以被FastJson正确的解析恢复成对象的;但是这段JSON字符串如果作用到上面描述的代码B中,由于没有getAllTicketList方法,因此没有"多余的allTicketList属性",从而B代码中FastJson反序列化时会出错;

解决方案

(1)如果都部署版本A代码,以上问题不会出现;

(2)如果allTicketList属性后于StrategyVO#loginStrategy解析,如使用 @JSONField(ordinal = 1) 注解属性,以上问题不会出现;

(3)如果不将getAllTicketList计入JSON序列化的getter,如使 @JSONField(serialize=false) 注解属性,以上问题不会出现;

最优的解决方案是修改实体中的getAllTicketList方法名,让其不要以 "get" 开头;此外,尽管领域模型的思想是把跟实体领域相关的能力方法封装到实体本身,但是在定义方法的时候最好自觉地加上 @JSONField(serialize=false) 注解属性以避免上述问题;

测试代码

1. 实体类定义,包含:目标类StrategyVO、属性类LoginStrategyVO、属性类LoginRangeVO、属性类TicketVO,以及与目标类具有相同成员属性的StrategyVOClone类(不带getter方法);

@Data
class StrategyVO {

    private String strategy;
    private LoginStrategyVO loginStrategy;

    // 额外的getter方法
    public List<TicketVO> getAllTicketList() {
        final List<TicketVO> allTickets = this.getLoginStrategy().getRangeVOList()
                .stream()
                .map(LoginRangeVO::getTicketsInRange)
                .flatMap(Collection::stream)
                .collect(Collectors.toList());
        return allTickets;
    }
}

@Data
class LoginStrategyVO {
    private String loginActName;
    private List<LoginRangeVO> rangeVOList;
}

@Data
class LoginRangeVO {
    private Integer days;
    private List<TicketVO> ticketsInRange;
}

@Data
class TicketVO {
    private Long ticketId;
    private Long ticketAmount;
}

/**
 * 与StrategyVO属性值一致,但不带getAllTicketList方法
 */
@Data
class StrategyVOClone {
    private String strategy;
    private LoginStrategyVO loginStrategy;
}

2. 构建目标类对象的测试方法buildStrategyVO;

    /**
     * 构造StrategyVO测试数据
     */
    private static StrategyVO buildStrategyVO() {
        final StrategyVO strategyVO = new StrategyVO();
        strategyVO.setStrategy("A");

        final LoginStrategyVO loginStrategyVO = new LoginStrategyVO();
        loginStrategyVO.setLoginActName("loginStrategyA");

        final LoginRangeVO loginRangeVO = new LoginRangeVO();
        loginRangeVO.setDays(7);

        final TicketVO ticketVO = new TicketVO();
        ticketVO.setTicketId(119L);
        ticketVO.setTicketAmount(5000L);

        loginRangeVO.setTicketsInRange(Lists.newArrayList(ticketVO));
        loginStrategyVO.setRangeVOList(Lists.newArrayList(loginRangeVO));
        strategyVO.setLoginStrategy(loginStrategyVO);

        return strategyVO;
    }

3. 测试方法;

(1)测试getter导致JSON字符串出现额外的字段;以及使用SerializerFeature.DisableCircularReferenceDetect临时关闭引用检测从而不会出现"$ref";

    /**
     * 由于getter方法StrategyVO#getAllTicketList(),导致json中有预期之外的字段allTicketList
     * 临时关闭引用检测时不会出现"$ref";
     */
    @Test
    public void testGetter() {
        final StrategyVO strategyVO = buildStrategyVO();
        System.out.println(JSON.toJSONString(strategyVO));
        System.out.println("临时关闭引用检测:" + JSON.toJSONString(strategyVO, SerializerFeature.DisableCircularReferenceDetect));
    }

输出:

{"allTicketList":[{"ticketAmount":5000,"ticketId":119}],"loginStrategy":{"loginActName":"loginStrategyA","rangeVOList":[{"days":7,"ticketsInRange":[{"$ref":"$.allTicketList[0]"}]}]},"strategy":"A"}
临时关闭引用检测:{"allTicketList":[{"ticketAmount":5000,"ticketId":119}],"loginStrategy":{"loginActName":"loginStrategyA","rangeVOList":[{"days":7,"ticketsInRange":[{"ticketAmount":5000,"ticketId":119}]}]},"strategy":"A"}

(2)测试重复引用导致的"$ref";

    /**
     * 重复引用导致的"$ref";
     */
    @Test
    public void test$valueSameObject() {
        final StrategyVO strategyVO = buildStrategyVO();
        final ArrayList<StrategyVO> list = Lists.newArrayList();
        // 添加重复元素
        list.add(strategyVO);
        list.add(strategyVO);

        final String listJson = JSON.toJSONString(list);
        System.out.println(listJson);
    }

输出:

[{"allTicketList":[{"ticketAmount":5000,"ticketId":119}],"loginStrategy":{"loginActName":"loginStrategyA","rangeVOList":[{"days":7,"ticketsInRange":[{"$ref":"$[0].allTicketList[0]"}]}]},"strategy":"A"},{"$ref":"$[0]"}]

(3)测试带有"$ref"的JSONString,反序列化时使用原实体类型,是可以被fastjson正确解析并反序列化成对象的;

    /**
     * 带有"$ref"的JSONString,反序列化时使用原实体类型,是可以被fastjson正确解析并反序列化成对象的
     */
    @Test
    public void test$valueSameRefReverse() {
        final StrategyVO strategyVO = buildStrategyVO();
        final ArrayList<StrategyVO> list = Lists.newArrayList();
        // 添加重复元素
        list.add(strategyVO);
        list.add(strategyVO);

        System.out.println("序列化之前: " + list);
        // 带有"$ref"的JSONString
        final String jsonStrWith$Ref = JSON.toJSONString(list);
        System.out.println("序列化JSON字符串: " + jsonStrWith$Ref);
        // 使用JSON.parseArray(String, Class<T>)
        final List<StrategyVO> strategyVOS = JSON.parseArray(jsonStrWith$Ref, StrategyVO.class);
        System.out.println("反序列化后: " + strategyVOS);
    }

输出:

序列化之前: [StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])])), StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])]))]
序列化JSON字符串: [{"allTicketList":[{"ticketAmount":5000,"ticketId":119}],"loginStrategy":{"loginActName":"loginStrategyA","rangeVOList":[{"days":7,"ticketsInRange":[{"$ref":"$[0].allTicketList[0]"}]}]},"strategy":"A"},{"$ref":"$[0]"}]
反序列化后: [StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])])), StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])]))]

(4)测试带有"$ref"的JSONString,反序列化时如果使用不带getAllTicketList方法的实体接收,则会丢失字段,也就是本次出现的异常复现;

    /**
     * 带有"$ref"的JSONString,反序列化时如果使用不带getAllTicketList方法的实体接收,则会丢失字段
     */
    @Test
    public void test$valueSameRefReverse2() {
        final StrategyVO strategyVO = buildStrategyVO();
        System.out.println("strategyVO: " + strategyVO);
        String jsonString = JSON.toJSONString(strategyVO);
        System.out.println("jsonString: " + jsonString);
        final StrategyVO strategyVOFrJson = JSON.parseObject(jsonString, StrategyVO.class);
        System.out.println("strategyVOFrJson: " + strategyVOFrJson);
        final StrategyVOClone strategyVOClone = JSON.parseObject(jsonString, StrategyVOClone.class);
        System.out.println("strategyVOCloneFrJson(丢失字段): " + strategyVOClone);
    }

输出:

strategyVO: StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])]))
jsonString: {"allTicketList":[{"ticketAmount":5000,"ticketId":119}],"loginStrategy":{"loginActName":"loginStrategyA","rangeVOList":[{"days":7,"ticketsInRange":[{"$ref":"$.allTicketList[0]"}]}]},"strategy":"A"}
strategyVOFrJson: StrategyVO(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[TicketVO(ticketId=119, ticketAmount=5000)])]))
strategyVOCloneFrJson(丢失字段): StrategyVOClone(strategy=A, loginStrategy=LoginStrategyVO(loginActName=loginStrategyA, rangeVOList=[LoginRangeVO(days=7, ticketsInRange=[null])]))

(5)循环引用时,使用SerializerFeature.DisableCircularReferenceDetect临时关闭引用检测从而导致的StackOverflowError;

    /**
     * 循环引用时,使用SerializerFeature.DisableCircularReferenceDetect临时关闭引用检测从而导致的StackOverflowError
     */
    @Test
    public void testDisableCircularReference() {
        // map1引用了map2,而map2又引用map1,导致循环引用
        Map<String, Object> map1 = new HashMap<>();
        Map<String, Object> map2 = new HashMap<>();
        map1.put("map", map2);
        map2.put("map", map1);
        System.out.println("map1Json:" + JSON.toJSONString(map1) + " map1Json:" + JSON.toJSONString(map1));
        System.out.println("临时关闭引用检测 map1Json:" + JSON.toJSONString(map1, SerializerFeature.DisableCircularReferenceDetect) +
                "临时关闭引用检测 map2Json:" + JSON.toJSONString(map2, SerializerFeature.DisableCircularReferenceDetect));
    }

输出:

map1Json:{"map":{"map":{"$ref":".."}}} map1Json:{"map":{"map":{"$ref":".."}}}

java.lang.StackOverflowError
	at com.alibaba.fastjson.serializer.SerializeWriter.writeStringWithDoubleQuote(SerializeWriter.java:840)
	at com.alibaba.fastjson.serializer.SerializeWriter.writeFieldName(SerializeWriter.java:2384)
	at com.alibaba.fastjson.serializer.MapSerializer.write(MapSerializer.java:226)
	at com.alibaba.fastjson.serializer.MapSerializer.write(MapSerializer.java:44)
    ...

小结

本篇通过一个日常开发中遇到的fastjson序列化相关的问题,通过源码分析,知道了fastjson计算对象的属性值是基于getter方法而非成员变量列表;并且在序列化过程中getter方法会被执行,因此理论上存在反序列化攻击漏洞;

为了避免StackOverflowError异常,fastjson会对引用进行检测,如果检测到存在重复/循环引用的情况,FastJson默认会以"引用标识$ref"代替同一对象,从而避免继续循环解析变量的值导致StackOverflowError;

反序列化时,尽管触发"FastJson重复引用"检测时,JSON字符串会出现"$ref"符号,这样的JSON字符串如果使用对象所在的类解析,是可以被FastJson正确的解析恢复成原对象的;

建议:尽量避免触发触发"FastJson重复引用"检测从而导致JSON字符串中出现预期之外的"引用标识$ref";

本文参考:

使用fastjson时出现$ref: "$.list[2]"的解决办法(重复引用) - 简书

【Fastjson】Fastjson反序列化由浅入深 - Aur0ra* - 博客园

fastjson反序列过程中getter 和 setter方法 - TT0TT - 博客园

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

企业信息化改革怎么做?

前几天遇到一位朋友提问&#xff0c;他说他们公司目前需要进行信息化改革&#xff0c;问我有哪些好用的信息化管理系统推荐&#xff1f;并附上了几点要求&#xff1a; 扩展性强&#xff08;公司管理结构变化很快&#xff0c;套装软件扩展升级太麻烦&#xff0c;并不适合&#…

怎样建立自己网站?【建立网站】

怎样建立自己网站&#xff1f;以前建立网站大部分情况下都是需要懂编程&#xff0c;又或者是懂一点代码然后去找源码模板做二次开发。不过随着技术的发展&#xff0c;建立自己网站的方式也在变多&#xff0c;例如目前比较流行的自助搭建网站。那么怎样建立自己网站呢&#xff1…

全卷积神经网络图像去噪研究-含Matlab代码

⭕⭕ 目 录 ⭕⭕✳️ 一、引言✳️ 二、网络结构✳️ 三、实验结果✳️ 3.1 数据集与网络训练✳️ 3.2 卷积神经网络去噪实验✳️ 3.3 基于BM3D的对比实验✳️ 四、参考文献✳️ 五、Matlab代码获取✳️ 一、引言 图像去噪在底层视觉中的重要性可以从多方面体现出来。首先&…

批量导入Npm包依赖到Nexus私服(批量上传脚本)

背景 批量导入是在以下几点情况下产生的需求&#xff1a; 已有Nexus系统&#xff0c;在测试构建环境中Nexus系统为离线环境不能配置外网代理自动下载项目代码工程所在工作电脑有条件联网现需要将项目代码在测试构建环境中连接Nexus私服进行编译构建 基于上面几点情况&#x…

HTML+CSS美食静态网站设计【海鲜网站】web结课作业的源码 web网页设计实例作业

&#x1f380; 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业…

Spring Security(3)

您好&#xff0c;我是湘王&#xff0c;这是我的CSDN博客&#xff0c;欢迎您来&#xff0c;欢迎您再来&#xff5e; 前面运行写好的代码之所以没有任何显示&#xff0c;是因为还没有对Spring Security进行配置&#xff0c;当然啥也不显示了。这就好比你坐在车上&#xff0c;却不…

智慧灾备解决方案-最新全套文件

智慧灾备解决方案-最新全套文件一、建设背景二、建设思路三、建设方案四、获取 - 智慧灾备全套最新解决方案合集一、建设背景 随着“云物移大智”各种技术的飞速发展&#xff0c;信息系统种类越来越多&#xff0c;数据保护也备受关注&#xff0c;传统数据保护往往仅覆盖数据库…

(CVPR 2019) 3D-SIS: 3D Semantic Instance Segmentation of RGB-D Scans

图 1&#xff1a;3D-SIS对RGB-D扫描数据执行3D实例分割&#xff0c;学习将2D RGB输入特征与3D扫描几何特征联合融合。结合能够在测试时对全3D扫描进行推理的全卷积方法&#xff0c;我们实现了对目标边界框、类标签和实例掩码的准确推理。 Abstract 我们介绍了3D-SIS&#xff0…

编译原理实验--实验三 预测分析法判断算术表达式的正确性--Python实现

目录 一、实验目的和要求 二、实验内容 三、实验环境 四、实验步骤 1、语法分析所依据的文法&#xff1b; 2、给出消除左递归及提取左公因子的LL(1)文法&#xff1b; 3、预测分析表 4、关键代码 五、实验结果与分析 一、实验目的和要求 理解自顶向下语法分析方法&#…

线程是什么?线程的相关概念以及基本的使用方法说明【内附可执行源码注释完整】

文章目录❓线程是什么&#x1f680;为什么要在程序中使用线程&#x1f34e;线程的优点、缺点&#x1f382;线程的应用场合&#x1f330;线程的基本使用⭐创建线程⭐线程的终止⭐等待指定线程结束⭐线程程序的编译命令&#x1f3e0;线程使用案例❓线程是什么 首先我们要知道进程…

3.1 首页功能的开发-跳转到首页

第三章. 业务功能开发 项目结构如下&#xff1a; 3.1 首页功能的开发 用户访问项目首页&#xff0c;首先进入登录页面。 用户登录流程图&#xff1a; 我们先写跳转到登录界面&#xff0c;在请求准发到index中如下图所示&#xff1a; 由于配置了视图解析器&#xff0c;查看ind…

GaussDB修改表空间占用磁盘使用率阈值

GaussDB数据库表空间占用磁盘使用率达到了你所设定的值后数据库会进入只读模式 执行非只读SQL时报错如下&#xff1a; ERROR: cannot execute %s in a read-only transaction. 或者运行中部分非只读SQL&#xff08;insert、update、create table as、create index、alter tab…

MCE | 促炎症的 PCD——细胞焦亡

程序性细胞死亡 (Programmed cell death, PCD) 是多细胞生物中&#xff0c;由基因调控的细胞自杀过程&#xff0c;对多细胞生物的发育、体内稳态和完整性至关重要。PCD 的研究涉及多个领域&#xff0c;如免疫、神经系统发育、癌症、感染等。常见的 PCD 有细胞凋亡 (Apoptosis)、…

贪吃蛇-第12届蓝桥杯Scratch选拔赛真题精选

[导读]&#xff1a;超平老师计划推出Scratch蓝桥杯真题解析100讲&#xff0c;这是超平老师解读Scratch蓝桥真题系列的第83讲。 蓝桥杯选拔赛每一届都要举行4~5次&#xff0c;和省赛、国赛相比&#xff0c;题目要简单不少&#xff0c;再加上篇幅有限&#xff0c;因此我精挑细选…

通信基石Socket结合OOP实现程序间的通信

学习目录前言一.Socket是用来干什么的二.如何用代码建立通信连接三.Java实现TCP发收消息四.UDP的Socket编程前言 先分享一下最近看到的几篇面经&#xff1a; 看了一下关于Socket的知识点好像面试十分高频&#xff0c;它作为通信的基石许多组件&#xff0c;框架都是在他的基础…

oracle数据库id字段自增长

mysql数据库中建表的时候可以设置id字段自增长&#xff0c;oracle数据库中要实现id字段自增长需要借助于sequence&#xff08;序列&#xff09;和trigger&#xff08;触发器&#xff09;。 下面通过一个简单的示范说明。 --1.新建一个账户表 account create table accout( …

机械转码日记【25】多态

目录 前言 1.多态的概念 2. 多态的定义及实现 2.1多态的构成条件 2.2 虚函数 2.3虚函数的重写 2.4虚函数重写的两个例外 2.4 C11 override 和 final 2.5 重载、覆盖(重写)、隐藏(重定义)的对比 3.抽象类 3.1概念 3.2 接口继承和实现继承 4.多态的原理 4.1虚函…

个人微信api

我们是一家专业服务企业数字化微信管理服务的技术服务团队&#xff0c;服务于需求SCRM、机器人、营销系统、社群小助手等具有研发能力的企业,同时我们也接收因使用Xp方案、ipad方案、PC方案导致被批量封号的企业&#xff0c;我们合作伙伴目前包含&#xff1a;金融服务行业Top10…

无协同资源创新打法,这几个品牌在双11「品牌嘉年华」实现品效双收

抖音双11好物节圆满收官&#xff0c;每年双11&#xff0c;我们关注的不仅仅是不断刷新的成绩&#xff0c;也是在更多元的场景、更丰厚的资源和更强劲的平台资源助力之下&#xff0c;商家在营销动作上带来了怎样的“惊喜”。 在「内容场景」&#xff0c;双11期间&#xff0c;抖…

VScode设置pretty-printer无效

文章目录VScode设置pretty-printer无效问题解决尝试正式解决参考VScode设置pretty-printer无效 问题 win10系统下&#xff0c;VScode中即使在launch.json中进行了如下设置&#xff0c;还是无效 {"description": "为 gdb 启用整齐打印","text"…