Spring系列第39篇:强大的Spel表达式

算数运算表达式

SpEL支持加(+)、减(-)、乘(*)、除(/)、求余(%)、幂(^)运算。

| 类型 | 示例 |

| — | — |

| 加减乘除 | int result1 = parser.parseExpression(“1+2-3*4/2”).getValue(Integer.class);//-3 |

| 求余 | int result2 = parser.parseExpression(“4%3”).getValue(Integer.class);//1 |

| 幂运算 | int result3 = parser.parseExpression(“2^3”).getValue(Integer.class);//8 |

SpEL还提供求余(MOD)和除(DIV)而外两个运算符,与“%”和“/”等价,不区分大小写。

关系表达式

等于(==)、不等于(!=)、大于(>)、大于等于(>=)、小于(<)、小于等于(<=),区间(between)运算。

parser.parseExpression("1>2").getValue(boolean.class);将返回false;

parser.parseExpression("1 between {1, 2}").getValue(boolean.class);将返回true。

between运算符右边操作数必须是列表类型,且只能包含2个元素。第一个元素为开始,第二个元素为结束,区间运算是包含边界值的,即 xxx>=list.get(0) && xxx<=list.get(1)

SpEL同样提供了等价的“EQ” 、“NE”、 “GT”、“GE”、 “LT” 、“LE”来表示等于、不等于、大于、大于等于、小于、小于等于,不区分大小写。

@Test

public void test3() {

ExpressionParser parser = new SpelExpressionParser();

boolean v1 = parser.parseExpression(“1>2”).getValue(boolean.class);

boolean between1 = parser.parseExpression(“1 between {1,2}”).getValue(boolean.class);

System.out.println(“v1=” + v1);

System.out.println(“between1=” + between1);

}

输出

v1=false

between1=true

逻辑表达式

且(and或者&&)、或(or或者||)、非(!或NOT)。

@Test

public void test4() {

ExpressionParser parser = new SpelExpressionParser();

boolean result1 = parser.parseExpression(“2>1 and (!true or !false)”).getValue(boolean.class);

boolean result2 = parser.parseExpression(“2>1 && (!true || !false)”).getValue(boolean.class);

boolean result3 = parser.parseExpression(“2>1 and (NOT true or NOT false)”).getValue(boolean.class);

boolean result4 = parser.parseExpression(“2>1 && (NOT true || NOT false)”).getValue(boolean.class);

System.out.println(“result1=” + result1);

System.out.println(“result2=” + result2);

System.out.println(“result3=” + result3);

System.out.println(“result4=” + result4);

}

输出

result1=true

result2=true

result3=true

result4=false

字符串连接及截取表达式

使用“+”进行字符串连接,使用“‘String’[0] [index]”来截取一个字符,目前只支持截取一个,如“'Hello ’ + ‘World!’”得到“Hello World!”;而“‘Hello World!’[0]”将返回“H”。

三目运算

三目运算符 **“表达式1?表达式2:表达式3”**用于构造三目运算表达式,如“2>1?true:false”将返回true;

Elivis运算符

Elivis运算符**“表达式1?:表达式2”**从Groovy语言引入用于简化三目运算符的,当表达式1为非null时则返回表达式1,当表达式1为null时则返回表达式2,简化了三目运算符方式“表达式1? 表达式1:表达式2”,如“null?:false”将返回false,而“true?:false”将返回true;

正则表达式

使用“str matches regex,如“‘123’ matches ‘\d{3}’”将返回true;

括号优先级表达式

使用“(表达式)”构造,括号里的具有高优先级。

类相关表达式

类类型表达式

使用“T(Type)”来表示java.lang.Class实例,“Type”必须是类全限定名,“java.lang”包除外,即该包下的类可以不指定包名;使用类类型表达式还可以进行访问类静态方法及类静态字段。

具体使用方法如下:

@Test

public void testClassTypeExpression() {

ExpressionParser parser = new SpelExpressionParser();

//java.lang包类访问

Class result1 = parser.parseExpression(“T(String)”).getValue(Class.class);

System.out.println(result1);

//其他包类访问

String expression2 = “T(com.javacode2018.spel.SpelTest)”;

Class value = parser.parseExpression(expression2).getValue(Class.class);

System.out.println(value == SpelTest.class);

//类静态字段访问

int result3 = parser.parseExpression(“T(Integer).MAX_VALUE”).getValue(int.class);

System.out.println(result3 == Integer.MAX_VALUE);

//类静态方法调用

int result4 = parser.parseExpression(“T(Integer).parseInt(‘1’)”).getValue(int.class);

System.out.println(result4);

}

输出

class java.lang.String

true

true

1

对于java.lang包里的可以直接使用“T(String)”访问;其他包必须是类全限定名;可以进行静态字段访问如“T(Integer).MAX_VALUE”;也可以进行静态方法访问如“T(Integer).parseInt(‘1’)”。

类实例化

类实例化同样使用java关键字“new”,类名必须是全限定名,但java.lang包内的类型除外,如String、Integer。

@Test

public void testConstructorExpression() {

ExpressionParser parser = new SpelExpressionParser();

String result1 = parser.parseExpression(“new String(‘路人甲java’)”).getValue(String.class);

System.out.println(result1);

Date result2 = parser.parseExpression(“new java.util.Date()”).getValue(Date.class);

System.out.println(result2);

}

实例化完全跟Java内方式一样,运行输出

路人甲java

Tue Aug 03 20:22:43 CST 2020

instanceof表达式

SpEL支持instanceof运算符,跟Java内使用同义;如“‘haha’ instanceof T(String)”将返回true。

@Test

public void testInstanceOfExpression() {

ExpressionParser parser = new SpelExpressionParser();

Boolean value = parser.parseExpression(“‘路人甲’ instanceof T(String)”).getValue(Boolean.class);

System.out.println(value);

}

输出

true

变量定义及引用

变量定义通过EvaluationContext接口的setVariable(variableName, value)方法定义;在表达式中使用"#variableName"引用;除了引用自定义变量,SpE还允许引用根对象及当前上下文对象,使用"#root"引用根对象,使用"#this"引用当前上下文对象;

@Test

public void testVariableExpression() {

ExpressionParser parser = new SpelExpressionParser();

EvaluationContext context = new StandardEvaluationContext();

context.setVariable(“name”, “路人甲java”);

context.setVariable(“lesson”, “Spring系列”);

//获取name变量,lesson变量

String name = parser.parseExpression(“#name”).getValue(context, String.class);

System.out.println(name);

String lesson = parser.parseExpression(“#lesson”).getValue(context, String.class);

System.out.println(lesson);

//StandardEvaluationContext构造器传入root对象,可以通过#root来访问root对象

context = new StandardEvaluationContext(“我是root对象”);

String rootObj = parser.parseExpression(“#root”).getValue(context, String.class);

System.out.println(rootObj);

//#this用来访问当前上线文中的对象

String thisObj = parser.parseExpression(“#this”).getValue(context, String.class);

System.out.println(thisObj);

}

输出

路人甲java

Spring系列

我是root对象

我是root对象

使用“#variable”来引用在EvaluationContext定义的变量;除了可以引用自定义变量,还可以使用“#root”引用根对象,“#this”引用当前上下文对象,此处“#this”即根对象。

自定义函数

目前只支持类静态方法注册为自定义函数;SpEL使用StandardEvaluationContext的registerFunction方法进行注册自定义函数,其实完全可以使用setVariable代替,两者其实本质是一样的;

@Test

public void testFunctionExpression() throws SecurityException, NoSuchMethodException {

//定义2个函数,registerFunction和setVariable都可以,不过从语义上面来看用registerFunction更恰当

StandardEvaluationContext context = new StandardEvaluationContext();

Method parseInt = Integer.class.getDeclaredMethod(“parseInt”, String.class);

context.registerFunction(“parseInt1”, parseInt);

context.setVariable(“parseInt2”, parseInt);

ExpressionParser parser = new SpelExpressionParser();

System.out.println(parser.parseExpression(“#parseInt1(‘3’)”).getValue(context, int.class));

System.out.println(parser.parseExpression(“#parseInt2(‘3’)”).getValue(context, int.class));

String expression1 = “#parseInt1(‘3’) == #parseInt2(‘3’)”;

boolean result1 = parser.parseExpression(expression1).getValue(context, boolean.class);

System.out.println(result1);

}

此处可以看出“registerFunction”和“setVariable”都可以注册自定义函数,但是两个方法的含义不一样,推荐使用“registerFunction”方法注册自定义函数。

运行输出

3

3

true

表达式赋值

使用Expression#setValue方法可以给表达式赋值

@Test

public void testAssignExpression1() {

Object user = new Object() {

private String name;

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

@Override

public String toString() {

return “$classname{” +

“name='” + name + ‘’’ +

‘}’;

}

};

{

//user为root对象

ExpressionParser parser = new SpelExpressionParser();

EvaluationContext context = new StandardEvaluationContext(user);

parser.parseExpression(“#root.name”).setValue(context, “路人甲java”);

System.out.println(parser.parseExpression(“#root”).getValue(context, user.getClass()));

}

{

//user为变量

ExpressionParser parser = new SpelExpressionParser();

EvaluationContext context = new StandardEvaluationContext();

context.setVariable(“user”, user);

parser.parseExpression(“#user.name”).setValue(context, “路人甲java”);

System.out.println(parser.parseExpression(“#user”).getValue(context, user.getClass()));

}

}

运行输出

$classname{name=‘路人甲java’}

$classname{name=‘路人甲java’}

对象属性存取及安全导航表达式

对象属性获取非常简单,即使用如“a.property.property”这种点缀式获取,SpEL对于属性名首字母是不区分大小写的;SpEL还引入了Groovy语言中的安全导航运算符“(对象|属性)?.属性”,用来避免“?.”前边的表达式为null时抛出空指针异常,而是返回null;修改对象属性值则可以通过赋值表达式或Expression接口的setValue方法修改。

public static class Car {

private String name;

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

@Override

public String toString() {

return “Car{” +

“name='” + name + ‘’’ +

‘}’;

}

}

public static class User {

private Car car;

public Car getCar() {

return car;

}

public void setCar(Car car) {

this.car = car;

}

@Override

public String toString() {

return “User{” +

“car=” + car +

‘}’;

}

}

@Test

public void test5() {

User user = new User();

EvaluationContext context = new StandardEvaluationContext();

context.setVariable(“user”, user);

ExpressionParser parser = new SpelExpressionParser();

//使用.符号,访问user.car.name会报错,原因:user.car为空

try {

System.out.println(parser.parseExpression(“#user.car.name”).getValue(context, String.class));

} catch (EvaluationException | ParseException e) {

System.out.println(“出错了:” + e.getMessage());

}

//使用安全访问符号?.,可以规避null错误

System.out.println(parser.parseExpression(“#user?.car?.name”).getValue(context, String.class));

Car car = new Car();

car.setName(“保时捷”);

user.setCar(car);

System.out.println(parser.parseExpression(“#user?.car?.toString()”).getValue(context, String.class));

}

运行输出

出错了:EL1007E: Property or field ‘name’ cannot be found on null

null

Car{name=‘保时捷’}

对象方法调用

对象方法调用更简单,跟Java语法一样;如“‘haha’.substring(2,4)”将返回“ha”;而对于根对象可以直接调用方法;

Bean引用

SpEL支持使用“@”符号来引用Bean,在引用Bean时需要使用BeanResolver接口实现来查找Bean,Spring提供BeanFactoryResolver实现。

@Test

public void test6() {

DefaultListableBeanFactory factory = new DefaultListableBeanFactory();

User user = new User();

Car car = new Car();

car.setName(“保时捷”);

user.setCar(car);

factory.registerSingleton(“user”, user);

StandardEvaluationContext context = new StandardEvaluationContext();

context.setBeanResolver(new BeanFactoryResolver(factory));

ExpressionParser parser = new SpelExpressionParser();

User userBean = parser.parseExpression(“@user”).getValue(context, User.class);

System.out.println(userBean);

System.out.println(userBean == factory.getBean(“user”));

}

运行输出

User{car=Car{name=‘保时捷’}}

true

集合相关表达式

内联List

从Spring3.0.4开始支持内联List,使用{表达式,……}定义内联List,如“{1,2,3}”将返回一个整型的ArrayList,而“{}”将返回空的List,对于字面量表达式列表,SpEL会使用java.util.Collections.unmodifiableList方法将列表设置为不可修改。

@Test

public void test7() {

ExpressionParser parser = new SpelExpressionParser();

//将返回不可修改的空List

List result2 = parser.parseExpression(“{}”).getValue(List.class);

//对于字面量列表也将返回不可修改的List

List result1 = parser.parseExpression(“{1,2,3}”).getValue(List.class);

Assert.assertEquals(new Integer(1), result1.get(0));

try {

result1.set(0, 2);

} catch (Exception e) {

e.printStackTrace();

}

//对于列表中只要有一个不是字面量表达式,将只返回原始List,

//不会进行不可修改处理

String expression3 = “{{1+2,2+4},{3,4+4}}”;

List<List> result3 = parser.parseExpression(expression3).getValue(List.class);

result3.get(0).set(0, 1);

System.out.println(result3);

//声明二维数组并初始化

int[] result4 = parser.parseExpression(“new int[2]{1,2}”).getValue(int[].class);

System.out.println(result4[1]);

//定义一维数组并初始化

int[] result5 = parser.parseExpression(“new int[1]”).getValue(int[].class);

System.out.println(result5[0]);

}

输出

java.lang.UnsupportedOperationException

at java.util.Collections$UnmodifiableList.set(Collections.java:1311)

at com.javacode2018.spel.SpelTest.test7(SpelTest.java:315)

[[1, 6], [3, 8]]

2

0

内联数组

和Java 数组定义类似,只是在定义时进行多维数组初始化。

int[][][] result4 = parser.parseExpression(“new int[1][2][3]{{1}{2}{3}}”).getValue(int[][][].class);

集合,字典元素访问

SpEL目前支持所有集合类型和字典类型的元素访问,使用“集合[索引]”访问集合元素,使用“map[key]”访问字典元素;

//SpEL内联List访问

int result1 = parser.parseExpression(“{1,2,3}[0]”).getValue(int.class);

//SpEL目前支持所有集合类型的访问

Collection collection = new HashSet();

collection.add(1);

collection.add(2);

EvaluationContext context2 = new StandardEvaluationContext();

context2.setVariable(“collection”, collection);

int result2 = parser.parseExpression(“#collection[1]”).getValue(context2, int.class);

//SpEL对Map字典元素访问的支持

Map<String, Integer> map = new HashMap<String, Integer>();

map.put(“a”, 1);

EvaluationContext context3 = new StandardEvaluationContext();

context3.setVariable(“map”, map);

int result3 = parser.parseExpression(“#map[‘a’]”).getValue(context3, int.class);

列表,字典,数组元素修改

可以使用赋值表达式或Expression接口的setValue方法修改;

@Test

public void test8() {

ExpressionParser parser = new SpelExpressionParser();

//修改list元素值

List list = new ArrayList();

list.add(1);

list.add(2);

EvaluationContext context1 = new StandardEvaluationContext();

context1.setVariable(“collection”, list);

parser.parseExpression(“#collection[1]”).setValue(context1, 4);

int result1 = parser.parseExpression(“#collection[1]”).getValue(context1, int.class);

System.out.println(result1);

//修改map元素值

Map<String, Integer> map = new HashMap<String, Integer>();

map.put(“a”, 1);

EvaluationContext context2 = new StandardEvaluationContext();

context2.setVariable(“map”, map);

parser.parseExpression(“#map[‘a’]”).setValue(context2, 4);

Integer result2 = parser.parseExpression(“#map[‘a’]”).getValue(context2, int.class);

System.out.println(result2);

}

输出

4

4

集合投影

在SQL中投影指从表中选择出列,而在SpEL指根据集合中的元素中通过选择来构造另一个集合,该集合和原集合具有相同数量的元素;SpEL使用“(list|map).![投影表达式]”来进行投影运算:

@Test

public void test9() {

ExpressionParser parser = new SpelExpressionParser();

//1.测试集合或数组

List list = new ArrayList();

list.add(4);

list.add(5);

EvaluationContext context1 = new StandardEvaluationContext();

context1.setVariable(“list”, list);

Collection result1 = parser.parseExpression(“#list.![#this+1]”).getValue(context1, Collection.class);

result1.forEach(System.out::println);

System.out.println(“------------”);

//2.测试字典

Map<String, Integer> map = new HashMap<String, Integer>();

map.put(“a”, 1);

map.put(“b”, 2);

EvaluationContext context2 = new StandardEvaluationContext();

context2.setVariable(“map”, map);

List result2 = parser.parseExpression(“#map.![value+1]”).getValue(context2, List.class);

result2.forEach(System.out::println);

}

对于集合或数组使用如上表达式进行投影运算,其中投影表达式中“#this”代表每个集合或数组元素,可以使用比如“#this.property”来获取集合元素的属性,其中“#this”可以省略。

Map投影最终只能得到List结果,如上所示,对于投影表达式中的“#this”将是Map.Entry,所以可以使用“value”来获取值,使用“key”来获取键。

集合选择

在SQL中指使用select进行选择行数据,而在SpEL指根据原集合通过条件表达式选择出满足条件的元素并构造为新的集合,SpEL使用“(list|map).?[选择表达式]”,其中选择表达式结果必须是boolean类型,如果true则选择的元素将添加到新集合中,false将不添加到新集合中。

@Test

public void test10() {

ExpressionParser parser = new SpelExpressionParser();

//1.测试集合或数组

List list = new ArrayList();

list.add(1);

list.add(4);

list.add(5);

list.add(7);

EvaluationContext context1 = new StandardEvaluationContext();

context1.setVariable(“list”, list);

Collection result1 = parser.parseExpression(“#list.?[#this>4]”).getValue(context1, Collection.class);

result1.forEach(System.out::println);

System.out.println(“------------”);

}

输出

5

7

对于集合或数组选择,如“#collection.?[#this>4]”将选择出集合元素值大于4的所有元素。选择表达式必须返回布尔类型,使用“#this”表示当前元素。

//2.测试字典

Map<String, Integer> map = new HashMap<String, Integer>();

map.put(“a”, 1);

map.put(“b”, 2);

map.put(“c”, 3);

EvaluationContext context2 = new StandardEvaluationContext();

context2.setVariable(“map”, map);

Map<String, Integer> result2 = parser.parseExpression(“#map.?[key!=‘a’]”).getValue(context2, Map.class);

result2.forEach((key, value) -> {

System.out.println(key + “:” + value);

});

System.out.println(“------------”);

List result3 = parser.parseExpression(“#map.?[key!=‘a’].![value+1]”).getValue(context2, List.class);

result3.forEach(System.out::println);

输出

b:2

c:3


3

4

对于字典选择,如“#map.?[#this.key != ‘a’]”将选择键值不等于”a”的,其中选择表达式中“#this”是Map.Entry类型,而最终结果还是Map,这点和投影不同;集合选择和投影可以一起使用,如“#map.?[key != ‘a’].![value+1]”将首先选择键值不等于”a”的,然后在选出的Map中再进行“value+1”的投影。

表达式模板

模板表达式就是由字面量与一个或多个表达式块组成。每个表达式块由“前缀+表达式+后缀”形式组成,如“${1+2}”即表达式块。在前边我们已经介绍了使用ParserContext接口实现来定义表达式是否是模板及前缀和后缀定义。在此就不多介绍了,如“Error ${#v0} ${#v1}”表达式表示由字面量“Error ”、模板表达式“#v0”、模板表达式“#v1”组成,其中v0和v1表示自定义变量,需要在上下文定义。

解析表达式的时候需要指定模板,模板通过ParserContext接口来定义

public interface ParserContext {

//是否是模板

boolean isTemplate();

//模板表达式前缀

String getExpressionPrefix();

//模板表达式后缀

String getExpressionSuffix();

}

有个子类,我们直接可以拿来用:TemplateParserContext

@Test

public void test11() {

//创建解析器

SpelExpressionParser parser = new SpelExpressionParser();

//创建解析器上下文

ParserContext context = new TemplateParserContext(“%{”, “}”);

Expression expression = parser.parseExpression(“你好:%{#name},我们正在学习:%{#lesson}”, context);

//创建表达式计算上下文

EvaluationContext evaluationContext = new StandardEvaluationContext();

evaluationContext.setVariable(“name”, “路人甲java”);

evaluationContext.setVariable(“lesson”, “spring高手系列!”);

//获取值

String value = expression.getValue(evaluationContext, String.class);

System.out.println(value);

}

运行输出

你好:路人甲java,我们正在学习:spring高手系列!

在Bean定义中使用spel表达式


xml风格的配置

SpEL支持在Bean定义时注入,默认使用“#{SpEL表达式}”表示,其中“#root”根对象默认可以认为是ApplicationContext,只有ApplicationContext实现默认支持SpEL,获取根对象属性其实是获取容器中的Bean。

如:

                             **自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

总结

本文从基础到高级再到实战,由浅入深,把MySQL讲的清清楚楚,明明白白,这应该是我目前为止看到过最好的有关MySQL的学习笔记了,我相信如果你把这份笔记认真看完后,无论是工作中碰到的问题还是被面试官问到的问题都能迎刃而解!

MySQL50道高频面试题整理:

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
L表达式}”表示,其中“#root”根对象默认可以认为是ApplicationContext,只有ApplicationContext实现默认支持SpEL,获取根对象属性其实是获取容器中的Bean。

如:

                             **自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。[外链图片转存中…(img-A0K63ct9-1712114110663)]

[外链图片转存中…(img-YnUZ33to-1712114110663)]

[外链图片转存中…(img-cuj1zsJ3-1712114110663)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

总结

本文从基础到高级再到实战,由浅入深,把MySQL讲的清清楚楚,明明白白,这应该是我目前为止看到过最好的有关MySQL的学习笔记了,我相信如果你把这份笔记认真看完后,无论是工作中碰到的问题还是被面试官问到的问题都能迎刃而解!

MySQL50道高频面试题整理:

[外链图片转存中…(img-yzT3XHSU-1712114110664)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

  • 11
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值