Due to the needs of the work project, song playback is needed. With reference to various materials, I wrote an interface imitating NetEase Cloud Music playback, which can fully realize the music playback function. The front end simply uses vue components and audio tags to implement the player interface, while the back end calls NetEase Cloud's API to obtain the corresponding song information. Without further ado, let's get to the code Song playback interface (musicPlay.vue) <template> <div class="main-page"> <audio :src="songInfo.url" id="music" autoplay="autoplay"></audio> <div class="background-flitter" :style="`background-image: url(${songInfo.cover})`" ></div> <div class="play-mini"> <div class="progress-bar" @click="handleProgressClick" ref="track"> <div class="progress-box" :style="{ width: audioProgressPercent }" ></div> </div> <div class="songInfo"> <img class="poster" :src="songInfo.cover" alt="" /> <!-- Song title, singer name--> <div class="info"> <p style="font-weight: 600; color: #fff;">{{ songInfo.name }}</p> <p style="font-size: 14px; color: #fff">{{ songInfo.artistsName }}</p> </div> </div> <div class="controls"> <!-- Previous song hint--> <Tooltip content="Previous song" theme="light" :delay="1500"> <a href="javascript:;"> <Icon type="md-skip-backward" @click="skipBack" size="26" color="white" /> </a> </Tooltip> <!-- Playback Pause --> <Tooltip content="Play Pause" theme="light" :delay="1500"> <img @click="playMusic" class="status" v-show="!playing" src="@/assets/play_icon.png" alt="" /> <img class="status" @click="playMusic" v-show="playing" src="@/assets/play-02.png" alt="" /> </Tooltip> <!-- Next song hint--> <Tooltip content="Next song" theme="light" :delay="1500"> <a href="javascript:;"> <Icon type="md-skip-forward" @click="skipForward" size="26" color="white" /> </a> </Tooltip> </div> <div class="right-bottom"> <div class="text-div"></div> <!-- Volume --> <a href="javascript:;"> <Icon :type="volumeType" size="26" color="white" @click="volumeChange" /> </a> <div class="text-div"></div> <Slider style="width: 80px; z-index: 99999" @on-input="changeVolum" v-model="volume" ></Slider> <div class="text-div"></div> <div class="text-div"></div> <div class="text-div"></div> <!-- Song playback type --> <Tooltip :content="showPlayType" theme="light" :delay="1500"> <a href="javascript:;"> <Icon :custom="playStatus" @click="playTypeChange" size="26" color="white" /> </a> </Tooltip> <div class="text-div"></div> <div class="text-div"></div> <div class="playlist"> <!-- Playlist --> <Tooltip content="List" theme="light" :delay="1500"> <a href="javascript:;"> <Icon custom="iconfont icon-bofangqi-xuanji" @click="drawer = true" size="36" color="white" /> </a> </Tooltip> </div> </div> </div> <div class="song-cover-lyric"> <div class="disc-continer"> <div class="poster" ref="rotate"> <img :src="songInfo.cover" alt="" /> </div> <div class="song-name">{{ songInfo.name }}</div> <div class="song-artistsName">{{ songInfo.artistsName }}</div> </div> <div class="lyric"> <mscroll ref="lyric" :color="color" :colorLight="colorLight" :lineHeight="lineHeight" :paddingTop="paddingTop" :fontSize="fontSize" :lyricIndex="lyricIndex" :lyricsList="lyricInfo" ></mscroll> </div> </div> <Drawer title="Playlist" placement="left" width="320" :closable="false" v-model="drawer" > <div class="list-container"> <div class="songInfo" v-for="(item, index) in songList" :key="index" @click="PlayListMusic(index)" > <img :src="item.cover" alt="" /> <div class="info"> <div class="name">{{ item.name }}</div> <div class="singer">{{ item.artistsName }}</div> </div> </div> </div> </Drawer> </div> </template> <script> import { showMessage } from "../../publicMethod/publicMethods"; import Mscroll from "./lyricScroll.vue"; import commonJs from "../../utils/timeConversion"; import axios from "axios"; export default { data() { return { volumeNum: 80, //Stored volume volumeStatus: true, //Volume icon changes volumeType: "md-volume-up", //Volume icon playStatus: "iconfont icon-xunhuanbofang", //Play type icon playing: false, drawer: false, playIndex: 0, songInfo: {}, songList: [], volume: 80, // volume lyricInfo: [], playType: 1, // Play type: 1-list loop, 2-random play, 3-single loopshowPlayType: "list loop", audioProgress: 0, thumbTranslateX: 0, lyricIndex: 0, color: "#fff", //Default color of lyrics colorLight: "#40ce8f", //Highlight color of lyrics fontSize: "16px", //Font size of lyrics lineHeight: "42", //Height of each paragraph paddingTop: "300px", //Center the highlighted lyrics part}; }, created() {}, components: Mscroll, }, computed: { audioProgressPercent() { return `${this.audioProgress * 100}%`; }, }, mounted() { const music = document.getElementById("music"); if (this.$route.query.play) { this.ClickPlay(); } else { this.GetSongList(); } if (this.playing) { // Playing, click to pause this.playing = false; this.$refs.rotate.style.animationPlayState = "paused"; music.pause(); } else { // Paused, click to play this.playing = true; this.$refs.rotate.style.animationPlayState = "running"; music.play(); } }, methods: { //Volume changes volumeChange() { if (this.volumeStatus) { this.volumeStatus = false; this.volumeNum = this.volume; this.volume = 0; this.volumeType = "md-volume-off"; } else { this.volumeStatus = true; this.volume = this.volumeNum; this.volumeType = "md-volume-up"; } }, //Song playback type changes playTypeChange() { if (this.playType == 1) { this.playType = 2; this.showPlayType = "Random Play"; this.playStatus = "iconfont icon-suijibofang"; return; } if (this.playType == 2) { this.playType = 3; this.showPlayType = "Single loop"; this.playStatus = "iconfont icon-danquxunhuan"; return; } if (this.playType == 3) { this.playType = 1; this.showPlayType = "List loop"; this.playStatus = "iconfont icon-xunhuanbofang"; return; } }, ClickPlay() { this.audioInit(); this.getMusicList(this.songInfo.id); this.$refs.rotate.style.animationPlayState = "running"; this.playing = true; setTimeout(() => { music.play(); }, 100); }, GetSongList() { axios.get("/musicController/getMusicList").then(this.GetSongListInfo); }, GetSongListInfo(res) { let myList; if (res.code == "0000") { myList = res.data; } else { console.log("No data found"); myList = [ { artistsName: "The Landlord's Cat", cover: "https://p3.music.126.net/KkrcSwKbRsd8GuaOHILlxA==/109951166077317301.jpg", id: 1857630559, name: "New Boy", url: "https://music.163.com/song/media/outer/url?id=1857630559.mp3", Lyrics: : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : : }, ]; } this.songList = myList; this.songInfo = this.songList[0]; this.getMusicList(this.songInfo.id); //Get the URL lyrics information of the song being played through the id of the song being played this.audioInit(); }, audioInit() { let that = this; let progressL = this.$refs.track.offsetWidth; // Total length of progress bar music.addEventListener("timeupdate", () => { // Current playback time let compareTime = music.currentTime; for (let i = 0; i < that.lyricInfo.length; i++) { if (compareTime > parseInt(that.lyricInfo[i].time)) { const index = that.lyricInfo[i].index; if (i === parseInt(index)) { that.lyricIndex = i; } } } that.currentTime = commonJs.TimeToString(music.currentTime); that.audioProgress = music.currentTime / music.duration; that.thumbTranslateX = (that.audioProgress * progressL).toFixed(3); }); music.addEventListener("ended", () => { switch (parseInt(that.playType)) { case 1: // List loop that.playIndex = that.playIndex + 1 >= that.songList.length ? 0 : that.playIndex + 1; break; case 2: // Play randomly that.playIndex = Math.floor(Math.random() * that.songList.length); break; case 3: // single loop break; } that.songInfo = that.songList[that.playIndex]; this.getMusicList(that.songInfo.id); setTimeout(() => { this.$refs.rotate.style.animationPlayState = "running"; music.play(); }, 200); }); }, //Play and pause playMusic() { if (this.playing) { // Playing, click to pause this.playing = false; this.$refs.rotate.style.animationPlayState = "paused"; music.pause(); } else { // Paused, click to play this.playing = true; this.$refs.rotate.style.animationPlayState = "running"; music.play(); } }, PlayListMusic(index) { this.playIndex = index; this.songInfo = this.songList[this.playIndex]; this.getMusicList(this.songInfo.id); this.playing = true; this.drawer = false; setTimeout(() => { this.$refs.rotate.style.animationPlayState = "running"; music.play(); }, 200); }, //Click the progress bar handleProgressClick(event) { let progressL = this.$refs.track.offsetWidth; // Total length of progress bar let clickX = event.offsetX; let time = (clickX / progressL).toFixed(2); this.setProgress(time); }, setProgress(x) { music.currentTime = music.duration * x; }, // Previous song skipBack() { this.skipFn("skipBack"); }, // Next song skipForward() { this.skipFn("skipForward"); }, // Upper and lower header encapsulation skipFn(type) { switch (parseInt(this.playType)) { case 2: // Randomly play this.playIndex = Math.floor(Math.random() * this.songList.length); break; default: if (type == "skipBack") { this.playIndex - 1 >= 0 ? this.playIndex-- : 0; } else { this.playIndex = this.playIndex + 1 >= this.songList.length ? this.songList.length - 1 : this.playIndex + 1; } break; } this.songInfo = this.songList[this.playIndex]; this.getMusicList(this.songInfo.id); this.playing = true; setTimeout(() => { this.$refs.rotate.style.animationPlayState = "running"; music.play(); }, 200); }, //Adjust the volume changeVolum(c) { music.volume = c / 100; if (music.volume == 0) { this.volumeType = "md-volume-off"; } else { this.volumeType = "md-volume-up"; } }, //Get the URL information of the song to be played getMusicList(id) { let that = this; axios.get("/musicController/getMusicURLInfo/" + id).then(function (res) { switch (res.code) { case "0000": that.songInfo.url = res.data.url; that.GetLyricList(res.data.lyric); break; case "1111": showMessage("warning", res.message); break; } }); }, GetLyricList(lrc) { let lyricsObjArr = []; const regNewLine = /\n/; const lineArr = lrc.split(regNewLine); // Array of lyrics for each line const regTime = /\[\d{2}:\d{2}.\d{2,3}\]/; lineArr.forEach((item) => { if (item === "") return; const obj = {}; const time = item.match(regTime); obj.lyric = item.split("]")[1].trim() === "" ? "" : item.split("]")[1].trim(); obj.time = time ? commonJs.TimeToSeconds(time[0].slice(1, time[0].length - 1)) : 0; obj.uid = Math.random().toString().slice(-6); if (obj.lyric === "") { console.log("There are no lyrics in this line"); } else { lyricsObjArr.push(obj); } }); this.lyricInfo = lyricsObjArr.map((item, index) => { item.index = index; return { ...item, }; }); }, }, }; </script> <style lang="less" scoped> .main-page { width: 100%; height: 100%; position: absolute; background: rgba(15, 15, 15, 0.4); overflow: hidden; .background-flitter { position: absolute; z-index: 0; background-repeat: no-repeat; width: 100%; height: 100%; top: 0; left: 0; background-size: cover; background-position: 50%; filter: blur(8px); // margin: -20px; opacity: 0.7; overflow: hidden; box-sizing: border-box; } .play-mini { position: absolute; bottom: 0; left: 0; width: 100%; height: 72px; // background: #fff; display: flex; align-items: center; padding: 6px 0; box-sizing: border-box; z-index: 10; .songInfo { min-width: 360px; max-width: 480px; position: relative; padding: 0 18px; box-sizing: border-box; display: flex; .poster { width: 52px; height: 52px; border-radius: 5px; margin-top: 4px; margin-right: 20px; } .info { min-width: 280px; height: 100%; line-height: 30px; font-size: 16px; } } .controls { width: 280px; height: 100%; display: flex; align-items: center; img { width: 40px; height: 40px; cursor: pointer; } .status { width: 40px; height: 40px; margin-left: 36px; margin-right: 36px; cursor: pointer; } } .progress-bar { position: absolute; z-index: 10; top: -5px; width: 100%; height: 5px; background: rgba(255, 255, 255, 0.5); cursor: pointer; .progress-box { height: 100%; background: #40ce8f; position: relative; } } .right-bottom { position: relative; width: 420px; height: 100%; display: flex; align-items: center; .text-div { color: #fff; height: 100%; line-height: 60px; margin-left: 5px; margin-right: 5px; } .playlist { position: absolute; right: 0px; } a { color: #333; } } } .song-cover-lyrics { position: relative; width: 100%; height: 100%; padding-bottom: 72px; box-sizing: border-box; display: flex; overflow: hidden; .disc-continer { width: 50%; height: 100%; position: relative; .poster { position: relative; width: 280px; height: 280px; border-radius: 50%; background: rgba(255, 255, 255, 0.3); left: 50%; top: 100px; margin-left: -140px; box-shadow: 0 0 0 12px rgba(255, 255, 255, 0.4); animation: animations1 12s linear infinite forwards; animation-play-state: paused; overflow: hidden; margin-bottom: 160px; img { width: 100%; height: 100%; } } .song-name { width: 100%; height: 40px; text-align: center; font-size: 32px; font-weight: 600; color: #fff; line-height: 40px; } .song-artistsName { width: 100%; height: 40px; text-align: center; font-size: 28px; font-weight: 600; color: #fff; line-height: 40px; margin-top: 24px; } @keyframes animations1 { from { transform: rotate(0deg); } to { transform: rotate(360deg); } } } .lyric { width: 50%; height: 600px; position: relative; overflow: hidden; } } } </style> <style lang="less"> .ivu-drawer-body { .list-container { width: 100%; height: 100%; overflow:auto; position: relative; .songInfo { width: 100%; height: 42px; display: flex; align-items: center; margin-bottom: 12px; cursor: pointer; img { width: 36px; height: 36px; border-radius: 5px; margin-right: 12px; } .info { position: relative; width: 240px; height: 36px; line-height: 18px; .name { width: 100%; height: 18px; font-size: 14px; font-weight: 600; text-overflow: ellipsis; overflow: hidden; white-space: nowrap; line-height: 18px; } .singer { width: 100%; height: 18px; font-size: 12px; text-overflow: ellipsis; overflow: hidden; white-space: nowrap; line-height: 18px; } } } } } </style> Lyrics part (lyricScroll.vue) <template> <!--Lyrics--> <div ref="musicLyric" class="music-lyric" :style="{ 'padding-top': paddingTop }" > <div class="music-lyric-items" :style="lyricTop"> <template v-if="lyricsList.length > 0"> <p v-for="(item, index) in lyricsList" :key="index" :data-index="index" ref="lyric" :style="{ color: lyricIndex === index ? colorLight : color, 'font-size': fontSize, }" > {{ item.lyric }} </p> </template> <p style="color: #fff" v-else>The lyrics are loading. . . . . </p> </div> </div> </template> <script> const COMPONENT_NAME = "scroll"; export default { name: COMPONENT_NAME, props: { // Lyrics list lyricsList: { type: Array, default: () => [], }, // Current lyrics index lyricIndex: { type: Number, default: 0, }, // Lyrics default color color: { type: String, default: "#fff", }, // Lyrics highlight colorcolorLight: { type: String, default: "#40ce8f", }, fontSize: { type: String, default: "16px", }, lineHeight: { type: String, default: "42", }, paddingTop: { type: String, default: "300px", }, }, data() { return { top: 0, // Lyrics centered // Lyrics list example lyricListDemo: [ { index: 0, lyric: "Composition: CMJ", time: 0, }, { index: 1, lyric: "Lyrics: Tao Jiu", time: 0.29, }, { index: 2, lyric: "Listen to the falling leaves in late autumn", time: 0.89, }, { index: 3, lyrics: "You listen to it sighing goodbye", time: 5.14, }, { index: 4, lyric: "I am left alone to appreciate it", time: 9.39, }, { index: 5, lyrics: "Sea, Mountain, Wind and Moon", time: 13.14, }, ], }; }, mounted() {}, watch: lyricIndex(newVal, oldVal) {}, }, computed: { lyricTop() { return `transform :translate3d(0, ${(0 - this.lineHeight) * (this.lyricIndex - this.top)}px, 0);color: ${this.color};line-height: ${ this.lineHeight }px`; }, }, methods: {}, }; </script> <style lang="less" scoped> /*Lyrics part*/ .music-lyrics { padding-top: 300px; box-sizing: border-box; overflow: hidden; text-align: center; mask-image: linear-gradient( to bottom, rgba(255, 255, 255, 0) 0, rgba(255, 255, 255, 0.6) 5%, rgba(255, 255, 255, 1) 15%, rgba(255, 255, 255, 1) 85%, rgba(255, 255, 255, 0.6) 95%, rgba(255, 255, 255, 0) 100% ); .music-lyric-items { text-align: center; font-size: 16px; color: #fff; transform: translate3d(0, 0, 0); transition: transform 0.6s ease-out; text-overflow: ellipsis; overflow: hidden; white-space: nowrap; .on { color: #40ce8f; } } } </style> Time conversion JS (timeConversion.js) /** * @author lyq * @time November 21, 2021 21:08:48 * * Convert seconds to hours, minutes and seconds* @name TimeToString * @example Convert seconds to hours, minutes and seconds* @param {String} seconds seconds*/ const TimeToString = (seconds) => { let param = parseInt(seconds); let hh = "", mm = "", ss = ""; if (param >= 0 && param < 60) { param < 10 ? (ss = "0" + param) : (ss = param); return "00:" + ss; } else if (param >= 60 && param < 3600) { mm = parseInt(param / 60); mm < 10 ? (mm = "0" + mm) : mm; param - parseInt(mm * 60) < 10 ? (ss = "0" + String(param - parseInt(mm * 60))) : (ss = param - parseInt(mm * 60)); return mm + ":" + ss; } } The basic front-end code is here, and the following is the interface logic layer code of the back-end /** * @author : [LiuYanQiang] * @version : [v1.0] * @className : MusicServiceImpl * @description: [Description of the function of this class] * @createTime : [2021/11/16 14:28] * @updateUser : [LiuYanQiang] * @updateTime : [2021/11/16 14:28] * @updateRemark: [Description of this modification] */ @Service @Slf4j public class MusicServiceImpl implements MusicService { @Autowired private Environment environment; /* * @version V1.0 * Title: getMusicList * @author LiuYanQiang * @description Get information about 20 random songs on the hot song list* @createTime 2021/11/16 14:32 * @param [] * @return java.util.Map*/ @Override public List<Map<String, Object>> getMusicList() { JSONArray resultObject = null; //Concatenate the URLs and send the corresponding request StringBuffer url = new StringBuffer(); url.append("https://api.vvhan.com/api/rand.music?type=all&sort=Hot Song List"); //Get the return value of the interface String body = HttpUtils.sendGet(url.toString()); resultObject = JSONObject.parseArray(body); List<Map<String, Object>> list = new ArrayList<>(); for (int i = 0; i < resultObject.size(); i++) { HashMap<String, Object> map = new HashMap<String, Object>(); JSONObject jsonObject = resultObject.getJSONObject(i); map.put("cover", JSONObject.parseObject(jsonObject.get("album").toString()).getString("picUrl")); map.put("artistsName", JSONObject.parseArray(jsonObject.get("artists").toString()).getJSONObject(0).getString("name")); map.put("name", jsonObject.getString("name")); map.put("id", jsonObject.getString("id")); list.add(map); } Random random = new Random(); int num = random.nextInt(179) % (179 - 0 + 1) + 0; list = list.subList(num, num + 20); return list; } /* * @version V1.0 * Title: getMusicURLInfo * @author LiuYanQiang * @description Get the URL information of the music player * @createTime 2021/11/19 9:22 * @param [Id——Music Id] * @return java.util.List<java.util.Map<java.lang.String,java.lang.Object>>*/ @Override public Map<String, Object> getMusicURLInfo(String Id) { JSONObject resultObject = null; //Concatenate the URLs and send the corresponding request StringBuffer url = new StringBuffer(); url.append("https://api.vvhan.com/api/music?id=" + Id + "&type=song&media=netease"); //Get the return value of the interface String body = HttpUtils.sendGet(url.toString()); resultObject = JSONObject.parseObject(body); HashMap<String, Object> map = new HashMap<String, Object>(); //Judge whether the music URL given by the third party is valid. If it is invalid, replace it with the official URL if(this.isValid(resultObject.get("mp3url").toString())){ map.put("id", resultObject.get("song_id").toString()); map.put("name", resultObject.get("name")); map.put("artistsName", resultObject.get("author")); map.put("cover", resultObject.get("cover")); map.put("url", resultObject.get("mp3url")); map.put("lyric", this.getMusicLyricById(Id) != null ? this.getMusicLyricById(Id) : null); } else{ map.put("id", Id); map.put("url", "https://music.163.com/song/media/outer/url?id="+Id+".mp3"); map.put("lyric", this.getMusicLyricById(Id) != null ? this.getMusicLyricById(Id) : null); } return map; } /* * @version V1.0 * Title: isValid * @author LiuYanQiang * @description Determine whether the link is valid* @createTime 2021/11/20 10:23 * @param [strLink——input link] * @return boolean * */ public boolean isValid(String strLink) { URL url; try { url = new URL(strLink); HttpURLConnection connt = (HttpURLConnection) url.openConnection(); connt.setRequestMethod("HEAD"); String strMessage = connt.getResponseMessage(); if (strMessage.compareTo("Not Found") == 0) { return false; } connt.disconnect(); } catch (Exception e) { return false; } return true; } /* * @version V1.0 * Title: getRandomFiveMusic * @author LiuYanQiang * @description Randomly play 5 songs. Do not call frequently, otherwise the NetEase Cloud API callback will be abnormal. * @createTime 2021/11/19 9:08 * @param [] * @return java.util.List<java.util.Map<java.lang.String,java.lang.Object>>*/ @Override public List<Map<String, Object>> getRandomFiveMusic() { List<Map<String, Object>> list = new ArrayList<>(); for (int i = 0; i < 5; i++) { JSONObject resultObject = null; //Concatenate the URLs and send the corresponding request StringBuffer url = new StringBuffer(); url.append("https://api.vvhan.com/api/rand.music?type=json&sort=Hot Song List"); //Get the return value of the interface String body = HttpUtils.sendGet(url.toString()); resultObject = JSONObject.parseObject(body); JSONObject info = JSONObject.parseObject(resultObject.get("info").toString()); HashMap<String, Object> map = new HashMap<String, Object>(); map.put("id", info.get("id").toString()); map.put("name", info.get("name")); map.put("artistsName", info.get("auther")); map.put("cover", info.get("picUrl")); map.put("url", info.get("mp3url")); map.put("lyric", this.getMusicLyricById(info.get("id").toString()) != null ? this.getMusicLyricById(info.get("id").toString()) : null); list.add(map); log.info("Call successful" + i + "times"); } return list; } /* * @version V1.0 * Title: getMusicLyricById * @author LiuYanQiang * @description Get lyrics information* @createTime 2021/11/16 19:23 * @param [Id——Music Id] * @return java.lang.String*/ @Override public String getMusicLyricById(String Id) { try { JSONObject resultObject = null; //Concatenate the URLs and send the corresponding request StringBuffer url = new StringBuffer(); url.append("https://music.163.com/api/song/media?id=" + Id); //Get the return value of the interface String body = HttpUtils.sendGet(url.toString()); resultObject = JSONObject.parseObject(body); if (resultObject.get("lyric").toString() != null) { return resultObject.get("lyric").toString(); } else { return null; } } catch (Exception e) { e.printStackTrace(); } return null; } } Summarize This is the end of this article about the simple implementation of Vue imitating NetEase Cloud Music Player interface. For more relevant Vue NetEase Cloud Music Player interface content, please search 123WORDPRESS.COM's previous articles or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: The website is grayed out. Compatible code including images supports all browsers
>>: 503 service unavailable error solution explanation
This article introduces MySQL string interception...
Implement Nginx load balancing based on Docker ne...
What is a covering index? Creating an index that ...
Table of contents 1. Three modes of binlog 1.Stat...
Table of contents 1. Open the project directory o...
1 Installation Download the corresponding unzippe...
Table of contents 1. Preparation 2. Writing comma...
constraint Constraints ensure data integrity and ...
This article example shares the specific code of ...
Table of contents What is a headless browser? Why...
Isolation Level: Isolation is more complicated th...
I installed node to the D drive, and I also neede...
This article is based on MySQL 8.0 This article i...
This article shares the encapsulation code of Jav...
If the field storing the name uses the GBK charac...