Spring boot實現(xiàn)RESTful接口

移動端開發(fā)經(jīng)常接觸到后端API,使用Java實現(xiàn)一個RESTful Web API昵宇。

安裝Visual Sutdio Code

安裝好VS Code后,需要安裝以下插件:

  1. Extension Pack for Java形葬。
  2. Spring boot Extension Pack竖幔。

新建項目

  1. 使用快捷鍵(Ctrl+Shift+P)命令窗口,輸入Spring選擇創(chuàng)建Gradle(Maven) 項目砸脊【咂或者,左側(cè)資源管理器中凌埂,點擊“創(chuàng)建Java項目”驱显,輸入Spring,選擇Spring Boot,創(chuàng)建Gradle(Maven) 項目埃疫。
  2. 選擇Spring boot版本伏恐,2.6.3,任意選栓霜。
  3. 選擇開發(fā)語言翠桦,Java,也可以選擇其他語言胳蛮。
  4. 輸入項目包名com.example.api秤掌,自己取。
  5. 輸入項目名demo鹰霍,自己取闻鉴。
  6. 選擇打包方式War,也可選另一種Jar茂洒。
  7. 指定JDK版本孟岛。
  8. 選擇依賴庫,Spring Web督勺。

回車渠羞,到這里已經(jīng)自動生成了一個項目,打開項目智哀,build.gradle文件中可以修改配置次询。

實現(xiàn)一個文件上傳的Controller

文件保存和主機(jī)文件系統(tǒng)有關(guān)系,本示例是Mac系統(tǒng)瓷叫,Linux系統(tǒng)也可用屯吊。

package com.example.test.demo.controllers;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;

import com.example.test.demo.bean.BaseResponse;
import com.example.test.demo.utils.FileUtils;

import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

@RestController
@RequestMapping(path = "/upload")
public class UploadController {

    static String rootDir = "/Users/wacdgit/files";
    @PostMapping(path = "/text", consumes = MediaType.TEXT_PLAIN_VALUE, produces = MediaType.APPLICATION_JSON_VALUE)
    public BaseResponse uploadText(@RequestBody String text) {
        if (text == null) {
            return new BaseResponse(-1, "Text is null");
        }
        File dir = new File(rootDir);
        if (!dir.exists()) {
            dir.mkdirs();
        }
        FileUtils.saveText(text, new File(rootDir, "temp.md"));
        return new BaseResponse(0, "Success");
    }

    @PostMapping(path = "/form", consumes = MediaType.MULTIPART_FORM_DATA_VALUE, produces = MediaType.APPLICATION_JSON_VALUE)
    public BaseResponse uploadForm(@RequestParam("file") MultipartFile multipartFile) {
        InputStream ins = null;
        try {
            ins = multipartFile.getInputStream();
            File dir = new File(rootDir);
            if (!dir.exists()) {
                dir.mkdirs();
            }
            File file = new File(rootDir, multipartFile.getOriginalFilename());
            FileUtils.saveFile(ins, file);
        } catch (Exception e1) {
            e1.printStackTrace();
            return new BaseResponse(-1, e1.getMessage());
        } finally {
            if (ins != null) {
                try {
                    ins.close();
                } catch (IOException e2) {
                    e2.printStackTrace();
                    return new BaseResponse(-2, e2.getMessage());
                }
            }
        }
        return new BaseResponse(0, "Success");
    }
}

響應(yīng)父類

package com.example.test.demo.bean;

import java.beans.JavaBean;

@JavaBean
public class BaseResponse {
    private int error;
    private String msg;

    public BaseResponse() {
    }

    public BaseResponse(int error, String msg) {
        this.error = error;
        this.msg = msg;
    }

    public int getError() {
        return error;
    }

    public void setError(int error) {
        this.error = error;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }

}

兩個RESTful API接口分別為upload/textupload/formVS Code運(yùn)行項目后摹菠,可以以內(nèi)容方式或者表單方式上傳文件盒卸。

例如:
url1 = "http://localhost:8080/upload/text"
url2 = "http://localhost:8080/upload/form"

部署

前面已經(jīng)可以在開發(fā)環(huán)境運(yùn)行,如果需要部署到主機(jī)或者服務(wù)器上正在運(yùn)行的Tomcat容器次氨,則需要生成WarJar包蔽介。

生成資源包:

./gradlew build

運(yùn)行上面指令后,在項目的build/libs目錄下生成的資源包煮寡,拷貝出來放置Tomcat下的webapps目錄虹蓄,即可自動運(yùn)行。

注意:
部署后的url需要加上war資源包的名稱幸撕,即項目名薇组。
url1 = "http://localhost:8080/demo/upload/text"
url2 = "http://localhost:8080/demo/upload/form"

數(shù)據(jù)庫

添加數(shù)據(jù)庫處理能力,模擬實現(xiàn)用戶注冊杈帐,登錄体箕。

1. 數(shù)據(jù)庫安裝和設(shè)置

安裝Mysql

Mac: brew install mysql
Linux: apt-get install mysql 或 yun install mysql

創(chuàng)建demodb數(shù)據(jù)庫

登錄root用戶专钉,默認(rèn)空密碼:mysql -u root -p
創(chuàng)建數(shù)據(jù)庫:create database demodb;

新建數(shù)據(jù)庫訪問用戶test挑童,并設(shè)置權(quán)限

創(chuàng)建用戶:create user 'test'@'localhost' identified by '123456';
授權(quán)'demo'數(shù)據(jù)庫的所有操作:grant all privileges on demodb.* to 'test'@'localhost';

localhost表示限制為本機(jī)訪問累铅,也可指定IP地址,或%替代表示允許所有地址訪問站叼。

新建用戶表user

也可以不做這一步娃兽,JPA框架可以自動生成表。

USE demodb;

CREATE TABLE IF NOT EXISTS user(
   id INT UNSIGNED AUTO_INCREMENT,
   name VARCHAR(32) NOT NULL,
   password VARCHAR(20) NOT NULL,
   PRIMARY KEY(id)
)ENGINE=InnoDB DEFAULT CHARSET=utf8;

DESC user;

數(shù)據(jù)庫名demodb尽楔,用戶名test和密碼123456一會需要用到投储。

2. 項目配置

首先,需要在build.gradle文件中添加兩個依賴:

    //Java持久層接口阔馋,ORM框架玛荞,類似移動端Room,GreenDao呕寝,Realm
    implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
    //mysql 數(shù)據(jù)庫驅(qū)動
    implementation 'mysql:mysql-connector-java'

其次勋眯,在resources文件夾下的application.properties文件中添加如下設(shè)置:

#驅(qū)動
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
#鏈接的數(shù)據(jù)庫地址,端口和名稱
spring.datasource.url=jdbc:mysql://localhost:3306/demodb
#登錄用戶名
spring.datasource.username=test
#登錄密碼
spring.datasource.password=123456
#第一次啟動根據(jù)實體建立表結(jié)構(gòu)下梢,之后啟動會根據(jù)實體的改變更新表結(jié)構(gòu)
spring.jpa.hibernate.ddl-auto=update
#打印Sql
spring.jpa.show-sql=true
#禁用
spring.jpa.open-in-view=false

3. 登錄注冊接口實現(xiàn)

User對象客蹋,即數(shù)據(jù)庫user表。

package com.example.test.demo.dao;

import java.beans.JavaBean;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;

@Entity
@Table(name = "user")
@JavaBean
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;
    @Column(name = "name", length = 32, unique = true, nullable = false)
    private String name;
    @Column(name = "password", length = 20, nullable = false)
    private String pwd;
    
    //查詢必須有無參構(gòu)造
    public User() {

    }

    public User(String name, String pwd) {
        this.name = name;
        this.pwd = pwd;
    }

    public String getName() {
        return name;
    }

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

    public String getPwd() {
        return pwd;
    }

    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

}

UserDao孽江,數(shù)據(jù)庫操作接口讶坯。

package com.example.test.demo.dao;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.stereotype.Repository;

@Repository
public interface UserDao extends JpaRepository<User, Integer> {

    @Query(value = "select * from user where name=?", nativeQuery = true)
    User findByName(String name); 

    @Query(value = "select * from user where name=?1 and password=?2", nativeQuery = true)
    User findByNameAndPwd(String name, String pwd);
}

UserController控制器實現(xiàn)。

package com.example.test.demo.controllers;

import com.example.test.demo.bean.BaseResponse;
import com.example.test.demo.dao.User;
import com.example.test.demo.dao.UserDao;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping(path = "/user")
public class UserController {

    @Autowired
    UserDao userDao;

    /**
     * POST register a user
     * 
     * @param name
     * @param pwd
     * @return
     */

    @PostMapping(path = "/register", consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE, produces = MediaType.APPLICATION_JSON_VALUE)
    public BaseResponse register(String name, String pwd) {
        if (name == null || name.isEmpty() || name.isBlank()) {
            return new BaseResponse(-1, "The name can not be empty");
        }
        if (pwd == null || pwd.isEmpty() || pwd.isBlank()) {
            return new BaseResponse(-2, "The pwd can not be empty");
        }
        User user = userDao.findByName(name);
        if (user != null) {
            return new BaseResponse(-3, "Exist user");
        } else {
            userDao.save(new User (name, pwd));
            return new BaseResponse(0, "Success");
        }
    }

    /**
     * GET request岗屏,鏈接中帶用戶名和密碼
     * 
     * @param name
     * @param pwd
     * @return
     */
    @GetMapping(path = "/login")
    public BaseResponse login1(@RequestParam(value = "name") String name, @RequestParam(value = "pwd") String pwd) {
        return login(name, pwd);
    }

    @PostMapping(path = "/login", consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE, produces = MediaType.APPLICATION_JSON_VALUE)
    public BaseResponse login2(String name, String pwd) {
        return login(name, pwd);
    }

    /**
     * POST request and response data format: Json
     * User 需要@JavaBean注解
     * 
     * @param user
     * @return
     */
    @PostMapping(path = "/login")
    public BaseResponse login3(@RequestBody User user) {
        return login(user.getName(), user.getPwd());
    }

    /**
     * POST request and response data format: Json
     * User 可以不使用@JavaBean注解
     * 
     * @param user
     * @return
     */
    @PostMapping(path = "/login", consumes = MediaType.APPLICATION_JSON_VALUE, produces = MediaType.APPLICATION_JSON_VALUE)
    public BaseResponse login4(@RequestBody User user) {
        return login(user.getName(), user.getPwd());
    }

    private BaseResponse login(String name, String pwd) {
        if (name == null || name.isEmpty() || name.isBlank()) {
            return new BaseResponse(-1, "The name can not be empty");
        }
        if (pwd == null || pwd.isEmpty() || pwd.isBlank()) {
            return new BaseResponse(-2, "The pwd can not be empty");
        }
        User user = userDao.findByNameAndPwd(name, pwd);
        if (user == null) {
            return new BaseResponse(-3, "The user is not exist");
        } else {
            return new BaseResponse(0, "Success");
        }
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末辆琅,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子这刷,更是在濱河造成了極大的恐慌涎跨,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件隅很,死亡現(xiàn)場離奇詭異,居然都是意外死亡率碾,警方通過查閱死者的電腦和手機(jī)叔营,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進(jìn)店門蟹但,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事沙郭。” “怎么了夜矗?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵浪南,是天一觀的道長虐先。 經(jīng)常有香客問我,道長埠对,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任固灵,我火速辦了婚禮熄诡,結(jié)果婚禮上袜茧,老公的妹妹穿的比我還像新娘。我一直安慰自己圈澈,他們只是感情好惫周,可當(dāng)我...
    茶點故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著康栈,像睡著了一般递递。 火紅的嫁衣襯著肌膚如雪喷橙。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天登舞,我揣著相機(jī)與錄音贰逾,去河邊找鬼。 笑死菠秒,一個胖子當(dāng)著我的面吹牛疙剑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播践叠,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼言缤,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了禁灼?” 一聲冷哼從身側(cè)響起管挟,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎弄捕,沒想到半個月后僻孝,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡守谓,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年穿铆,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片斋荞。...
    茶點故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡荞雏,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出譬猫,到底是詐尸還是另有隱情讯檐,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布染服,位于F島的核電站别洪,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏柳刮。R本人自食惡果不足惜挖垛,卻給世界環(huán)境...
    茶點故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望秉颗。 院中可真熱鬧痢毒,春花似錦、人聲如沸蚕甥。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽菇怀。三九已至凭舶,卻和暖如春晌块,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背帅霜。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工匆背, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人身冀。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓钝尸,卻偏偏與公主長得像,于是被迫代替她去往敵國和親搂根。 傳聞我的和親對象是個殘疾皇子珍促,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內(nèi)容