Spring學習總結(四)——表達式語言 Spring Expression Language

来源:http://www.cnblogs.com/best/archive/2016/08/08/5748105.html
-Advertisement-
Play Games

SpEL簡介與功能特性 Spring表達式語言(簡稱SpEL)是一個支持查詢併在運行時操縱一個對象圖的功能強大的表達式語言。SpEL語言的語法類似於統一EL,但提供了更多的功能,最主要的是顯式方法調用和基本字元串模板函數。 同很多可用的Java 表達式語言相比,例如OGNL,MVEL和JBoss E ...


SpEL簡介與功能特性

Spring表達式語言(簡稱SpEL)是一個支持查詢併在運行時操縱一個對象圖的功能強大的表達式語言。SpEL語言的語法類似於統一EL,但提供了更多的功能,最主要的是顯式方法調用和基本字元串模板函數。

同很多可用的Java 表達式語言相比,例如OGNL,MVEL和JBoss EL,SpEL的誕生是為了給Spring社區提供一個可以給Spring目錄中所有產品提供單一良好支持的表達式語言。其語言特性由Spring目錄中的項目需求驅動,包括基於eclipse的SpringSource套件中的代碼補全工具需求。也就是說,SpEL是一個基於技術中立的API允許需要時與其他表達式語言集成。

SpEL作為Spring目錄中表達式求值的基礎,它並不是直接依賴於Spring而是可以被獨立使用。為了能夠自包含,本章中的許多示例把SpEL作為一個獨立的表達式語言來使用。這就需要創建一些如解析器的引導基礎組件類。大多數Spring用戶只需要為求值編寫表達式字元串而不需要關心這些基礎組件

SpEL功能特性:

  • ž 字元表達式
  • ž 布爾和關係操作符
  • ž 正則表達式
  • ž 類表達式
  • ž 訪問properties,arrays,lists,maps
  • ž 方法調用
  • ž 關係操作符
  • ž 賦值
  • ž 調用構造器
  • ž 三元操作符
  • ž 變數
  • ž 用戶自定義函數
  • ž 集合投影
  • ž 集合選擇
  • ž 模板表達式

一、為什麼需要Spring表達式語言

1.1、新建一個Maven Web項目,添加依賴,pom.xml如下所示:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.zhangguo</groupId>
    <artifactId>Spring053</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>Spring053</name>
    <url>http://maven.apache.org</url>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <spring.version>4.3.0.RELEASE</spring.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <scope>test</scope>
            <version>4.10</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.8.9</version>
        </dependency>
        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
            <version>3.2.4</version>
        </dependency>
    </dependencies>
</project>

表達式語言所依賴的包如下所示:

 2.2、為了IOC,定義了用戶類User.java與Order.java,如下所示:

用戶類:

package com.zhangguo.Spring053.spel01;

/**
 * 訂單類
 *
 */
public class Order {
    /**
     * 訂單名稱
     */
    private String orderName;
    /*
     * 用戶姓名
     */
    private String userName;
    /**
     * 用戶對象
     */
    private User customer;

    public String getOrderName() {
        return orderName;
    }

    public void setOrderName(String orderName) {
        this.orderName = orderName;
    }

    public User getCustomer() {
        return customer;
    }

    public void setCustomer(User customer) {
        this.customer = customer;
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }
    
    @Override
    public String toString() {
        return "訂單名:"+this.getOrderName()+",姓名:"+this.getUserName()+",編號:"+this.getCustomer().getId();
    }
}
View Code

訂單類:

package com.zhangguo.Spring053.spel01;

/**
 * 訂單類
 *
 */
public class Order {
    /**
     * 訂單名稱
     */
    private String orderName;
    /*
     * 用戶姓名
     */
    private String userName;
    /**
     * 用戶對象
     */
    private User customer;

    public String getOrderName() {
        return orderName;
    }

    public void setOrderName(String orderName) {
        this.orderName = orderName;
    }

    public User getCustomer() {
        return customer;
    }

    public void setCustomer(User customer) {
        this.customer = customer;
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }
}
View Code

2.3、編寫容器初始化的配置文件spel01.xml,內容如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">

    <bean id="gyl" class="com.zhangguo.Spring053.spel01.User" p:id="9527">
        <property name="name" value="郭永樂">
        </property>
    </bean>
    
    <bean id="order001" class="com.zhangguo.Spring053.spel01.Order">
        <property name="customer" ref="gyl"></property>
        <property name="name" value="#{gyl.name}"></property>
        <property name="orderName" value='#{"Apples".toUpperCase()}'></property>
    </bean>

</beans>

在配置文件中,出現了#{}形式的表達式,我們就稱為Spel表達式。#{gyl.name}作用是找到名稱為gyl的bean取出中間的name值;#{"Apples".toUpperCase()}把字元串Apples轉換成大寫並輸出。

2.4、取出bean測試

package com.zhangguo.Spring053.spel01;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Test {

    public static void main(String[] args) {
        ApplicationContext ctx=new ClassPathXmlApplicationContext("spel01.xml");
        Order order=ctx.getBean("order001",Order.class);
        System.out.println(order);
    }
}

2.5、運行結果

二、SpEL表達式Hello World!

Spring表達式語言(SpEL)從3.X開始支持,它是一種能夠支持運行時查詢和操作對象圖的強大的表達式,其表達式語法類似於統一表達式語言。
SpEL支持如下表達式:
基本表達式:字面量表達式、關係,邏輯與算數運算表達式、字元串連接及截取表達式、三目運算、正則表達式、括弧優先順序表達式;
類相關表達式:類類型表達式、類實例化、instanceof表達式、變數定義及引用、賦值表達式、自定義函數、對象屬性存取及安全導航表達式、對象方法調用、Bean引用;
集合相關表達式:內聯List、內聯數組、集合,字典訪問、列表,字典,數組修改、集合投影、集合選擇;不支持多維內聯數組初始化;不支持內聯字典定義;
其他表達式:模板表達式。

從一個Hello World!的示例開始:

package com.zhangguo.Spring053.spel02;

import org.springframework.expression.Expression;
import org.springframework.expression.ExpressionParser;
import org.springframework.expression.spel.standard.SpelExpressionParser;

public class Test {

    public static void main(String[] args) {
        //創建SpEL表達式的解析器
        ExpressionParser parser=new SpelExpressionParser();
        //解析表達式'Hello '+' World!'
        Expression exp=parser.parseExpression("'Hello '+' World!'");
        //取出解析結果
        String result=exp.getValue().toString();
        //輸出結果
        System.out.println(result);
    }
}

運行結果:

Hello World!

從示例中可以看出java成功的將一個字元解析出了結果,如果我們把要解析的內容設置成1+1則會解析出2來,這裡的1+1就是一個SpEL表達式,該表達式在java中只是一個String,通過解析可以得到字元串本身的意見,有點類型編譯程式的感覺。

三、SpEL表達式

3.1、文字表達式

支持的文字表達的類型是字元串,日期,數值(整型,實型,和十六進位),布爾和空。字元串是由單引號分隔。使用反斜杠字元轉移把一個單引號字元本身放在字元串中。

        ExpressionParser ep= new SpelExpressionParser();
        System.out.println(ep.parseExpression("'HelloWorld'").getValue());
        System.out.println(ep.parseExpression("0xffffff").getValue());
        System.out.println(ep.parseExpression("1.234345e+3").getValue());
        System.out.println(ep.parseExpression("new java.util.Date()").getValue());

運行結果:

HelloWorld
16777215
1234.345
Fri Jul 01 14:50:59 CST 2016

3.2、SPEL語言特性

3.2.1、屬性

        //創建SpEL表達式的解析器
        ExpressionParser parser=new SpelExpressionParser();
        User user=new User(9527,"周星馳");
        //解析表達式需要的上下文,解析時有一個預設的上下文
        EvaluationContext ctx = new StandardEvaluationContext();
        //在上下文中設置變數,變數名為user,內容為user對象
        ctx.setVariable("user", user);
        //從用戶對象中獲得id並+1900,獲得解析後的值在ctx上下文中
        int id = (Integer) parser.parseExpression("#user.getId() + 1900").getValue(ctx);
        System.out.println(id);

運行結果:

11427

User類在前面已定義,這裡增加了一個有參構造方法。上面的示例是調用方法,其實可以這樣:引用對象屬性,只需使用一個句點來表示一個嵌套的屬性值,如下代碼所示:

int id = (Integer) parser.parseExpression("#user.id + 1900").getValue(ctx);

運行結果:

11427

要註意的是此時#user後不再是一個方法而是.id,直接訪問屬性,在java中這樣做是不行的,便SpEL中允許

3.2.2、數組

        String[] students=new String[]{"tom","jack","rose","mark","lucy"};
        ctx.setVariable("students", students);
        String student = parser.parseExpression("#students[3]").getValue(ctx,
                String.class);
        System.out.println(student);

結果:mark

3.2.3、列表

        List numbers = (List) parser.parseExpression("{1,2,3,4,5}").getValue();
        System.out.println(numbers.get(2)+"");
        List listOfLists = (List) parser.parseExpression("{{'a','b'},{'x','y'}}").getValue();
        System.out.println(((List)listOfLists.get(1)).get(1));

結果:3 y

3.2.4、索引器、與字典

        User user1=new User(9001,"鄒任飛");
        User user2=new User(9002,"練汶峰");
        List<User> users=new ArrayList<User>();
        users.add(user1);
        users.add(user2);
        ctx.setVariable("users", users);
        String name = parser.parseExpression("#users[1].name").getValue(ctx,String.class);
        System.out.println(name);

結果:練汶峰

在java中訪問集合中的對象通過get(索引)而在SpEL中我們可以直接像數組那樣訪問對象。如果是一個字典可以這樣:#users["tom"].id

3.2.5、方法

方法調用使用典型的Java編程語法。 你可能 還在文字調用方法。 也支持可變參數。

        String c = parser.parseExpression("'abcdef'.substring(2, 3)").getValue(String.class);
        System.out.println(c);

結果:c

3.2.6、操作符

關係操作符:使用標準的操作符號支持關係操作符:等於,不等於,小於,小於等於,大於,大於等於。
邏輯操作符:支持的邏輯操作符包括and,or和not(!),不支持&&和||。
算術操作符:加法運算符可以用於數字,字元串和日期。減法可用於數字和日期。乘法和除法僅可以用於。其他支持的數學運算包括取模(%)和指數冪(^)。使用標準的運算符優先順序。

關係運算符:

        //true
        boolean trueValue1 = parser.parseExpression("2 == 2").getValue(Boolean.class);
        //false
        boolean falseValue1 = parser.parseExpression("2 < -5.0").getValue(Boolean.class);
        //true
        boolean trueValue2 = parser.parseExpression("'black' < 'block'").getValue(Boolean.class);
        //false,字元xyz是否為int類型
        boolean falseValue2 = parser.parseExpression("'xyz' instanceof T(int)").getValue(Boolean.class);
        //true,正則是否匹配
        boolean trueValue3 =parser.parseExpression("'5.00' matches '^-?\\d+(\\.\\d{2})?$'").getValue(Boolean.class);
        //false
        boolean falseValue3=parser.parseExpression("'5.0067' matches '^-?\\d+(\\.\\d{2})?$'").getValue(Boolean.class);

邏輯運算:

        // -- AND 與運算 --
        //false 
        boolean falseValue4 = parser.parseExpression("true and false").getValue(Boolean.class);
        //true,isMember方法用於測試是否為某個對象的成員
        String expression = "isMember('Nikola Tesla') and isMember('Mihajlo Pupin')";
        boolean trueValue4 = parser.parseExpression(expression).getValue(Boolean.class);
        // -- OR 或運算--
        //true
        boolean trueValue5 = parser.parseExpression("true or false").getValue(Boolean.class);
        //true
        String expression1 = "isMember('Nikola Tesla') or isMember('Albert Einstein')";
        boolean trueValue6 = parser.parseExpression(expression).getValue( Boolean.class);
        //false
        boolean falseValue5 = parser.parseExpression("!true").getValue(Boolean.class);
        //false
        String expression2 = "isMember('Nikola Tesla') and !isMember('Mihajlo Pupin')";
        boolean falseValue6 = parser.parseExpression(expression).getValue(Boolean.class);

算術運算:

// Addition
int two = parser.parseExpression("1 + 1").getValue(Integer.class); // 2
String testString =
parser.parseExpression("'test' + ' ' + 'string'").getValue(String.class); // 'test string'
// Subtraction
int four = parser.parseExpression("1 - -3").getValue(Integer.class); // 4
double d = parser.parseExpression("1000.00 - 1e4").getValue(Double.class); // -9000
// Multiplication
int six = parser.parseExpression("-2 * -3").getValue(Integer.class); // 6
double twentyFour = parser.parseExpression("2.0 * 3e0 * 4").getValue(Double.class); // 24.0
// Division
int minusTwo = parser.parseExpression("6 / -3").getValue(Integer.class); // -2
double one = parser.parseExpression("8.0 / 4e0 / 2").getValue(Double.class); // 1.0
// Modulus
int three = parser.parseExpression("7 % 4").getValue(Integer.class); // 3
int one = parser.parseExpression("8 / 5 % 2").getValue(Integer.class); // 1
// Operator precedence
int minusTwentyOne = parser.parseExpression("1+2-3*8").getValue(Integer.class); // -21

3.2.7、表達式支持定義bean、基於XML的配置

在xml配置中可以自由的使用SpEL,如下所示:

<bean id="gyl" class="com.zhangguo.Spring053.spel01.User" p:id="9527">
        <property name="name" value="郭永樂">
        </property>
    </bean>

    <bean id="order001" class="com.zhangguo.Spring053.spel01.Order">
        <property name="customer" ref="gyl"></property>
        <property name="userName" value="#{gyl.name}"></property>
        <property name="orderName" value='#{"Apples".toUpperCase()}'></property>
    </bean>

    <bean id="numberGuess" class="org.spring.samples.NumberGuess">
        <!--調用靜態方法random() -->
        <property name="randomNumber" value="#{ T(java.lang.Math).random() * 100.0 }" />
    </bean>

    <bean id="taxCalculator" class="org.spring.samples.TaxCalculator">
        <property name="defaultLocale" value="#{ systemProperties['user.region'] }" />
    </bean>

    <bean id="numberGuess" class="org.spring.samples.NumberGuess">
        <property name="randomNumber" value="#{ T(java.lang.Math).random() * 100.0 }" />
    </bean>
    <bean id="shapeGuess" class="org.spring.samples.ShapeGuess">
        <property name="initialShapeSeed" value="#{ numberGuess.randomNumber }" />
    </bean>

3.2.8、表達式支持定義bean、基於註解的配置

用戶類User.java:

package com.zhangguo.Spring053.spel03;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

/**
 * 用戶類
 */
@Component("user1")
public class User {
    /**
     * 編號
     */
    @Value("#{9527+100}")
    private int id;
    /**
     * 姓名
     */
    @Value("#{'Hello'.toUpperCase()}")
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

訂單類:Order.java

package com.zhangguo.Spring053.spel03;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Repository;

/**
 * 訂單類
 *
 */
@Repository("Order03")
public class Order {
    /**
     * 訂單名稱
     */
    @Value("#{'Apple訂單'}")
    private String orderName;
    /*
     * 用戶姓名
     */
    @Value("#{user1.name}")
    private String userName;
    /**
     * 用戶對象
     */
    @Value("#{user1}")
    private User customer;

    
    public String getOrderName() {
        return orderName;
    }

    public void setOrderName(String orderName) {
        this.orderName = orderName;
    }

    public User getCustomer() {
        return customer;
    }

    public void setCustomer(User customer) {
        this.customer = customer;
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }
    
    @Override
    public String toString() {
        return "訂單名:"+this.getOrderName()+",姓名:"+this.getUserName()+",編號:"+this.getCustomer().getId();
    }
}

測試代碼Test.java

package com.zhangguo.Spring053.spel03;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.zhangguo.Spring053.spel03.User;

public class Test {

    public static void main(String[] args) {
        ApplicationContext ctx=new ClassPathXmlApplicationContext("spel03.xml");
        User user1=ctx.getBean("user1",User.class);
        System.out.println(user1.getId()+","+user1.getName());
        
        Order Order03=ctx.getBean("Order03",Order.class);
        System.out.println(Order03);
    }
}

運行結果:

9627,HELLO
訂單名:Apple訂單,姓名:HELLO,編號:9627

3.2.9、操作符

        ExpressionParser ep = new SpelExpressionParser();
        // 關係操作符
        System.out.println(ep.parseExpression("5>2").getValue());
        System.out.println(ep.parseExpression("2 between {1,9}").getValue());
        // 邏輯運算符
        System.out.println(ep.parseExpression("(5>2) and (2==1)").getValue());
        // 算術操作符
        System.out.println(ep.parseExpression("100-2^2").getValue());

結果:

true
true
false
96

3.2.10、變數與賦值

變數:變數可以在表達式中使用語法#’變數名’引用
ExpressionParser ep= new SpelExpressionParser();
//創建上下文變數
EvaluationContext ctx = new StandardEvaluationContext();
ctx.setVariable(“name”, “Hello”);
System.out.println(ep.parseExpression("#name").getValue(ctx));

輸出:Hello
賦值:屬性設置是通過使用賦值運算符。這通常是在調用setValue中執行但也可以在調用getValue內,也可通過”#varName=value”的形式給變數賦值。
System.out.println(ep.parseExpression("#name='Ryo'").getValue(ctx));

輸出:Ryo

四、示例下載

點擊下載


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 網站優化必做的事情之一,百度ping,主動推送給百度 文章添加時調用百度推送方法 common類庫增加百度ping方法 ...
  • 先上一段最近項目中的代碼,此代碼可以放到自己項目中的dbContext中 EF6中可以覆寫SaveChangesAsync(非同步)或者SaveChanges來實現記錄變化的跟蹤,這其中包括新增、修改、和刪除,dbContext中的屬性ChangeTracker可以跟蹤屬性的變化,即查找實體修改記錄: ...
  • 按照網上的教程寫了一個cURL的小例子,在apache環境下執行,一點反應也沒有,放在IIS環境里就ok的,感覺問題一定出在動態連接庫上,因為配置文件里的php_curl.dll已經打開了,而且在iis上ok; 網上找了一些解決方案: 設置了【環境變數】:phpext,PHPRC;無效 把php_c ...
  • 博客園上傳代碼時拷貝vs裡面的代碼不能直接粘貼,否則空格會不符合要求 去掉空格代碼 ...
  • 實例運行結果如下 實例運行結果如下 實例運行結果如下 ...
  • 詳細的見 https://github.com/linux-wang/DRF_tutorial/blob/master/README.md DRF中有一個serializer的概念,實現的功能是將各種Django Queryset和model instance轉換成Python原生格式,這裡就省去了 ...
  • 實力運行效果如下 ...
  • String類和StringBuffer類主要用來處理字元串,這兩個類提供了很多字元串的使用處理方法。String類是不可變類,表示對象所包含的字元串類不能改變。StringBuffer類是可變類,其對象所包含的字元串內容可以被添加或修改。 關於這兩個類處理字元串的常用方法請參考:http://ww ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...