【開源】使用Angular9和TypeScript開發RPG游戲(20200410版)

来源:https://www.cnblogs.com/TextEditor/archive/2020/04/10/12672284.html
-Advertisement-
Play Games

源代碼地址 通過對於鬥羅大陸小說的游戲化過程,熟悉Angular的結構以及使用TypeScript的面向對象開發方法。 "Github項目源代碼地址" 線上體驗網址(推薦使用移動設備訪問) http://datavisualization.club:8888/ 極簡游戲攻略 除了劇情對話之外,本游戲 ...


源代碼地址

通過對於鬥羅大陸小說的游戲化過程,熟悉Angular的結構以及使用TypeScript的面向對象開發方法。
Github項目源代碼地址

線上體驗網址(推薦使用移動設備訪問)

http://datavisualization.club:8888/

極簡游戲攻略

除了劇情對話之外,本游戲暫時只有一個分叉選擇

  • 【趙無極試煉】是正確的分支,可以通往史萊克的正常劇情,這個分支的話,史萊克七怪將挑戰趙無極老師等人
  • 【昆圖庫塔卡提考特蘇瓦西拉松試煉】是番外篇的分支,可以進入番外分支

如果你選擇了【趙無極試煉】分支,則戰鬥會非常幸苦。如果你選擇了【昆圖庫塔卡提考特蘇瓦西拉松試煉】分支,則你會發現敵我由於等級屬性一致,無法分出勝負。

  • 提示1.星斗大森林的二層寶箱有一個道具:佛怒唐蓮,群體傷害99999(核彈級別殺傷力啊)
  • 提示2.你的背包裡面有一個道具:觀音淚,群體傷害99,屠龍(LV1)是沒有問題的。
  • 提示99.地圖去過的地方,可以瞬移的

RPG系統構造

ver0.03 2020/04/10

人物

唐三數據結構JSON版

和其他RPG游戲類似,游戲裡面的人物角色大致有這樣的一些屬性:生命值,魔法值(魂力),攻擊力,防禦力,速度。RPG游戲中的角色隨著等級的提高,這些屬性都會提升,屬性提升的快慢則取決於資質,同時,由於在實際戰鬥中,會出現各種增益和光環效果,這些值都是動態變化的,所以這裡將這些屬性都設置了Base和Real兩套數據。

Base屬性是指人物的初始屬性,是一種固有屬性,在整個游戲開始的時候就固定下來的。然後每個人物根據不同的資質,有一個成長值,例如SSR的角色,成長值可以是1.5,普通角色是1。這個成長值關係到每提升一個等級,角色屬性的增加值,代碼大致如下:

    /**經過增益之後的生命最大值 */
    get RealMaxHP(): number {
        var R = this.BaseMaxHP + (this.LV - 1) * this.MaxHPUpPerLv * this.GrowthFactor;
        ...
        ...
        ...
        return Math.round(R);
    }

這裡的 MaxHPUpPerLv 表示每個等級的最大生命值提升數值,GrowthFactor則表示成長值。

註意:這裡使用了TypeScript的get屬性,也就是只讀/計算屬性來處理Real系的屬性,這些屬性都是實時計算出來的!

在小說裡面,經常可以看到3成功力的角色,為了表示這種情況,代碼裡面還設定了一個Factor變數,通過這個變數可以設定整體的縮放比例。這個值預設為1,表示不縮放。

    /**經過增益之後的生命最大值 */
    get RealMaxHP(): number {
        var R = this.BaseMaxHP + (this.LV - 1) * this.MaxHPUpPerLv * this.GrowthFactor;
        R = R * this.Factor;
        ...
        ...
        ...
        return Math.round(R);
    }

由於乘法計算會出現小數點,這裡使用了Math.round對結果進行取整。

技能

技能是一個游戲的戰鬥核心,所有技能本質上都是為了改變角色狀態。如果要具體細分大致可以分為

  • 攻擊類:對於指定角色產生傷害
  • 回覆類:對於指定角色,回覆生命值和魔法值
  • 狀態改變類:這裡其實包含了Buffer和狀態變化兩種情況,Buffer類大多是被動技能,游戲中只要某個角色在戰場上就獲得,並且效果是持續性的。狀態變化則一般必須主動施放技能才行,而且持續時間也是有限制的。

同時技能設計的時候,還需要設定使用的方向,既這個技能是對於我方使用,還是敵方使用,還是無差別使用。另外這個技能的對象是某個對象,還是群體。

/**技能類型 */
export enum enmSkillType {
    /**攻擊 */
    Attact,
    /**治療 */
    Heal,
    /**光環和狀態  */
    Buffer
}

/**技能範圍 */
export enum enmRange {
    Self,       //自己
    PickOne,    //選擇一個人
    RandomOne,  //隨機選擇一個人
    FrontAll,   //前排所有人
    BackAll,    //後排所有人
    EveryOne,   //戰場所有人
}

/**技能方向 */
export enum enmDirect {
    MyTeam,     //本方
    Enemy,      //敵方
    All,        //全體
}

一般使用枚舉來編寫這樣相對固定,項目較少的列表

技能的設計,這裡使用了OOP的繼承來實現,技能的基類定義了一些共通的屬性和抽象方法。設計的時候還考慮到以下幾種特殊情況

  • 每一種具體技能必須要實現一個執行(施放)方法:Excute,這裡使用抽象函數,來強制子類型必須要實現這個方法
  • 對於複雜技能,需要有一個自定義的執行方法:CustomeExcute,同時通過返回值來告訴系統是不是該技能有自定義執行方法。則跳過固有的Excute方法。
  • 對於有些技能可能要同時實現兩種效果,這裡增加了AddtionSkill變數
/** 技能 */
export abstract class SkillInfo {
    Name: string;
    SkillType: enmSkillType;
    Range: enmRange;
    Direct: enmDirect;
    Description: string;
    /**冷卻回合數 */
    ColdDownTurn: number = 0;
    /**實時冷卻剩餘數 */
    CurrentColdDown = 0;
    /**是否能使用 */
    IsAvalible(fs: FightStatus): string {
        let c = fs.currentActionCharater;
        if (c.MP < this.MpUsage) return "MP不足";
        if (this.CurrentColdDown !== 0) return "冷卻中:" + this.CurrentColdDown;
        if (this.Combine !== undefined) {
            //武魂融合技
            let EveryOneCanAction = true;
            this.Combine.forEach(
                name => {
                    if (name !== c.Name) {
                        if (fs.TurnList.find(x => x.Name === name) === undefined) EveryOneCanAction = false;
                    }
                }
            );
            if (!EveryOneCanAction) return "融合者已行動";
        }
        return "";
    };
    /**效果隨著等級變化 */
    EffectWithLevel = false;
    MpUsage: number = 5;
    /**武魂融合技的融合者列表 */
    Combine: string[] = [];
    abstract Excute(c: Character, fs: FightStatus): void;
    /**自定義執行方法 */
    CustomeExcute(c: Character, fs: FightStatus): boolean {
        return false;
    }
    //攻擊並中毒這樣的兩個效果疊加的技能
    AddtionSkill: SkillInfo = undefined;
}

export class AttactSkillInfo extends SkillInfo {
    SkillType = enmSkillType.Attact;
    Harm: number;
    IgnoreceDefence: boolean;
    Excute(c: Character, fs: FightStatus) {
        //如果自定義方法被執行,則跳過後續代碼
        if (this.CustomeExcute(c, fs)) return;
        let factor = 1 + fs.currentActionCharater.LV / 100;
        c.HP -= Math.round(this.Harm * factor);
        if (c.HP <= 0) c.HP = 0;
        if (this.AddtionSkill !== undefined) this.AddtionSkill.Excute(c, fs);
    }
}

undefined來檢測是否擁有對象

Buffer技能

Buffer,可以叫做狀態增益,本系統的Buffer如下所示:該結構標明瞭Buffer的作用,來源,剩餘回合數,已經對於狀態的影響。

其中,狀態有常規的攻防增益,中毒,也有一些特殊的,例如施法之後產生的Flag型狀態:浴火鳳凰,幽冥影分身,飛行等就屬於這種特殊狀態。

/**狀態 */
export enum characterStatus {
    /**通用 */
    魂技,
    /**增益 */
    攻擊增益,
    防禦增益,
    速度增益,
    生命增益,
    魂力增益,

    /**每回合失去生命值 */
    中毒,
    /**無法使用技能 */
    禁言,
    /**無法物理和技能攻擊 */
    暈眩,
    /**無法普通攻擊,可以使用技能 */
    束縛,
    /**物理攻擊免疫 */
    物免,
    /**技能攻擊免疫 */
    魔免,
    /**全部免疫 */
    無敵,
    //特色特殊狀態:戰鬥開始的時候將被清除掉
    /**馬紅俊 */
    浴火鳳凰,
    /**朱竹清 */
    幽冥影分身,
    /**香腸效果 */
    飛行
}

/**Buffer */
export class Buffer {
    //Value表示絕對值,Percent表示百分比

    MaxHPValue: number = undefined;
    MaxHPFactor: number = undefined;

    HPValue: number = undefined;
    HPFactor: number = undefined;

    MaxMPValue: number = undefined;
    MaxMPFactor: number = undefined;

    MPValue: number = undefined;
    MPFactor: number = undefined;

    SpeedValue: number = undefined;
    SpeedFactor: number = undefined;

    AttactValue: number = undefined;
    AttactFactor: number = undefined;

    DefenceValue: number = undefined;
    DefenceFactor: number = undefined;
    /**來源 */
    Source: string;
    /**持續回合數 */
    Turns: number = 999;    //預設999回合
    /**狀態 */
    Status: characterStatus[] = [characterStatus.魂技];
}

在技能裡面有一類是Buffer技能,這個時候需要將Buffer放入角色的BufferList中,註意,由於技能描述中的Buffer是對於Skill的描述,是一個類,不能直接放入到人物BufferList中。而應該將Buffer的副本放入人物BufferList中去。

/**增益和減弱 */
export class BufferStatusSkillInfo extends SkillInfo {
    SkillType = enmSkillType.Buffer;
    Buffer: Buffer = new Buffer();
    /**Buffer強度是否和施法者等級掛鉤? */

    Excute(c: character, fs: FightStatus) {
        if (this.CustomeExcute(c, fs)) return;
        //增加Buffer來源信息,相同的不疊加
        if (c.BufferList.find(x => x.Source === this.Name) !== undefined) return;
        //增幅強度和等級關聯:如果是和施法者相關,必須使用currentActionCharater的信息
        if (this.BufferFactorByLV) {
            let factor = fs.currentActionCharater.LV / 100;
            //以下不使用 1 + factor 是因為RealTimeAct()計算使用了 R += R * element.AttactFactor; 
            if (this.Buffer.AttactFactor !== undefined) this.Buffer.AttactFactor = factor;
            if (this.Buffer.DefenceFactor !== undefined) this.Buffer.DefenceFactor = factor;
            if (this.Buffer.MaxHPFactor !== undefined) this.Buffer.MaxHPFactor = factor;
            if (this.Buffer.MaxMPFactor !== undefined) this.Buffer.MaxMPFactor = factor;
            if (this.Buffer.SpeedFactor !== undefined) this.Buffer.SpeedFactor = factor;
        }
        //從技能使用點開始就起效的屬性變化的調整:由於使用了get自動屬性功能,Real系的都會自動計算
        let MaxHpBefore = c.RealMaxHP;
        let MaxMpBefore = c.RealMaxMP;
        this.Buffer.Source = this.Name;
        //這裡必須使用副本
        c.BufferList.push(JSON.parse(JSON.stringify(this.Buffer)));
        let MaxHpAfter = c.RealMaxHP;
        let MaxMpAfter = c.RealMaxMP;
        //魂力和生命的等比縮放
        if (MaxHpAfter !== MaxHpBefore) c.HP = Math.round(c.HP * (MaxHpAfter / MaxHpBefore))
        if (MaxMpAfter !== MaxMpBefore) c.MP = Math.round(c.MP * (MaxMpAfter / MaxMpBefore))
        //生命值和魂力的Buffer,還需要對於HP和MP進行修正
        if (c.HP > c.RealMaxHP) c.HP = c.RealMaxHP;
        if (c.MP > c.RealMaxMP) c.MP = c.RealMaxMP;
        if (fs.IsDebugMode) {
            console.log("技能對象:" + c.Name);
            c.BufferList.forEach(element => {
                console.log("回合數:" + element.Turns + "\t狀態" + element.Status.toString() + "\t來源" + element.Source);
            });
        }
        if (this.AddtionSkill !== undefined) this.AddtionSkill.Excute(c, fs);
    }
}

具體到鬥羅大陸,其技能可能來自於魂骨(類似於極品裝備的概念)和魂環,或者角色自身融合技,設計的時候,暫時考慮技能獨立體系獨立存在,然後分配給魂骨魂環,魂骨魂環分配給人物。用這樣的方式將人物和技能串聯起來。

    public static 唐三(): Character {
        let 唐三 = new Character("唐三");
        唐三.LV = 29;
        唐三.GrowthFactor = 1.5;
        唐三.Bones = [
            BoneCreator.外附魂骨八蛛矛(),
            BoneCreator.天青牛蟒右臂骨(),
            BoneCreator.泰坦巨猿左臂骨(),
            BoneCreator.深海魔鯨王的軀幹骨(),
            BoneCreator.精神凝聚之智慧頭骨(),
            BoneCreator.藍銀皇右腿骨(),
            BoneCreator.邪魔虎鯨王左腿骨()
        ]
        唐三.TeamPosition = enmTeamPosition.控制系;
        唐三.Description = "唐三前世為巴蜀唐門外門子弟,來到鬥羅大陸後與伙伴們一起在異界大陸重新建立了唐門。"
        唐三.Soul = "藍銀皇";
        唐三.Circles = CircleCreator.唐三();
        唐三.SecondSoul = "昊天錘";
        唐三.Fields = [FieldCreator.藍銀領域(), FieldCreator.海神領域(),
        FieldCreator.殺神領域(), FieldCreator.修羅領域()];
        return 唐三;
    }

    public static 邪魔虎鯨王左腿骨(): Bone {
        let e = new Bone();
        e.Name = "邪魔虎鯨王左腿骨";
        e.Position = BonePosition.左腿骨;
        e.FirstSkill = BoneSkillCreator.虎鯨碎牙斬();
        e.SecondSkill = BoneSkillCreator.虎鯨邪魔斧();
        return e;
    }

    //邪魔虎鯨王左腿骨
    public static 虎鯨邪魔斧(): SkillInfo {
        let s = new AttactSkillInfo();
        s.Name = "虎鯨邪魔斧";
        s.Description = "完全作用於攻擊,凝全身功力於左腿,經魂骨增幅,化為薄如蟬翼的戰斧利刃,直線型單體攻擊";
        s.Direct = enmDirect.Enemy;
        s.Range = enmRange.PickOne;
        s.Harm = 5000;
        return s;
    }

    public static 虎鯨碎牙斬(): SkillInfo {
        let s = new AttactSkillInfo();
        s.Name = "虎鯨碎牙斬";
        s.Description = "群攻技能";
        s.Direct = enmDirect.Enemy;
        s.Range = enmRange.EveryOne;
        s.Harm = 2000;
        return s;
    }

劇情

每個場景包含了名稱,標題,對白(戰鬥)列表,背景,下一個場景名稱和分支的信息。

public static lineIdx: number = 0;    //臺詞位置
export interface SceneInfo {
    Name: string;
    Title: string;
    Lines: string[];
    Background: string;
    NextScene?: string;
    Branch?: [string, string][]
}

export const Scene0001: SceneInfo = {
    Name: "Scene0001",
    Title: "引子 穿越的唐家三少",
    Background: "唐門",
    Lines: [
        "唐門長老@玄天寶錄,你竟然連玄天寶錄中本門最高內功也學了?",
        "唐門唐三@赤裸而來,赤裸而去,佛怒唐蓮算是唐三最後留給本門的禮物。",
        "唐門唐三@現在,除了我這個人以外,我再沒有帶走唐門任何東西,秘籍都在我房間門內第一塊磚下。唐三現在就將一切都還給唐門。",
        "唐門唐三@哈哈哈哈哈哈哈……。",
        "唐門長老@等一下。",
        "唐門唐三@(雲霧很濃,帶著陣陣濕氣,帶走了陽光,也帶走了那將一生貢獻給了唐門和暗器的唐三。)",
    ],
    Branch: [
        ["趙無極試煉", "Scene0011"],
        ["達拉崩巴試煉", "Scene0012"]
    ]
};

每次對話發生的時候,lineIdx這個臺詞位置的指針都會下移,指向下一句臺詞或者開啟戰鬥。這裡使用 FightPrefix表示進入戰鬥。對話列表則使用@符號將角色和臺詞進行區分。

export const Scene0011: SceneInfo = {
    Name: "Scene0011",
    Title: "史萊克學院",
    Background: "史萊克學院",
    Lines: [
        "小舞@史萊克學院的趙無極老師及其厲害,小心對付啊。",
        FightPrefix + "Battle0001",
        "唐三@終於通過史萊克學院的入學測試了!奧力給!",
    ]
};

道具系統

可以將道具看作一種特殊的技能,只是這種技能是可以購買的。當然特殊的劇情道具則不屬於這個範疇,設計起來比較複雜,需要配合場景的通過條件來使用。

import { SkillInfo } from './SkillInfo';

/** 道具 */
export class ToolInfo {
    /** 名字 */
    Name: string;
    /** 圖標 */
    Icon: string;
    /** 價格 */
    Price: number;
    /** 道具和技能可以合併 */
    Func: SkillInfo;
    /**道具類型 */
    ToolType: enmToolType = enmToolType.StoreItem;
}

export class HiddenWeapon extends ToolInfo {
    ToolType = enmToolType.HiddenWeapon;
};

export enum enmToolType {
    /**暗器 */
    HiddenWeapon,
    /**可購入的一般道具 */
    StoreItem,
    /**劇情道具 */
    Spacial
}

public static 佛怒唐蓮(): ToolInfo {
    let t = new ToolInfo();
    t.ToolType = enmToolType.HiddenWeapon;
    t.Name = "佛怒唐蓮";
    t.Icon = ResourceMgr.icon_attact;
    t.Func = ToolSkillCreator.佛怒唐蓮();
    t.Price = 99999;
    return t;
}

戰鬥流程

ver0.02 2020/03/30

回合開始

每一個回合開始的時候,首先對上一個回合進行一次清算。

  • 狀態回合數的遞減
  • 中毒狀態的傷害計算
    BufferTurnDown() {
        this.BufferList.forEach(element => {
            if (element.Status.find(x => x === characterStatus.中毒) !== undefined) {
                //中毒狀態,如果存在HP傷害部分,則這裡處理,由於使用了get自動屬性功能,Real系的都會自動計算
                if (element.HPFactor !== undefined) this.HP += this.HP * element.HPFactor;
                if (element.HPValue !== undefined) this.HP += element.HPValue;
            }
            element.Turns -= 1;
        });
        this.BufferList = this.BufferList.filter(x => x.Turns > 0);
    }

極端情況下,敵我雙方都可能被束縛,無法行動,所以先做一下判斷是否有可以行動的角色。

按照出手速度,將所有角色放在一個數組裡面,然後決定第一個出手的人,如果是我方人員,等待用戶界面的指令輸入,如果是敵方的話,則使用AI進行行動。無論是AI還是用戶界面的指令,一旦完成,則執行ActionDone方法,進行勝負判定,切換當前的行動角色。

    /**當前角色動作完成 */
    ActionDone() {
        //勝負統計
        let MyTeamLive = this.MyTeam.find(x => x !== undefined && x.HP > 0);
        if (MyTeamLive === undefined) {
            console.log("團滅");
            this.MyTeam.forEach(element => { this.InitRole(element) });
            this.ResultEvent.emit(0);
            return;
        }

        let EnemyTeamLive = this.Enemy.find(x => x !== undefined && x.HP > 0);
        if (EnemyTeamLive === undefined) {
            console.log("勝利");
            //這裡需要還原MyTeam的隊列
            this.MyTeam = this.info.MyTeam.map(x => this.GetRoleByName(x));
            this.MyTeam.forEach(element => {
                if (element !== undefined) {
                    element.Exp += this.Exp;
                    this.InitRole(element)
                }
            });
            this.ResultEvent.emit(this.Exp);
            return;
        }

        //氣絕者去除
        this.MyTeam = this.MyTeam.map(x => (x !== undefined && x.HP > 0) ? x : undefined);
        this.Enemy = this.Enemy.map(x => (x !== undefined && x.HP > 0) ? x : undefined);
        this.TurnList = this.TurnList.map(x => (x !== undefined && x.HP > 0) ? x : undefined);
        this.TurnList = this.TurnList.filter(x => x !== undefined);

        if (this.TurnList.length == 0) {
            console.log("回合結束");
            this.NewTurn();
        } else {
            let Role = this.TurnList.pop();
            let block = Role.StatusList.find(x => x === characterStatus.束縛 || x === characterStatus.暈眩);

            if (Role === undefined || block !== undefined) {
                console.log(Role.Name + ":角色已經氣絕,或者角色被束縛");
                this.ActionDone();
            } else {
                console.log("當前角色:" + Role.Name + "[" + Role.IsMyTeam + "]");
                this.currentActionCharater = Role;
                if (!Role.IsMyTeam) {
                    //AI For Enemy
                    this.EnemyAction.emit(RPGCore.EnemyAI(Role, this));
                    this.ActionDone();
                }
            }
        }
    }

這裡使用了@Output()的EventEmitter<>向外部發送消息戰鬥結束。由於敵方AI運行速度極快,所以這裡沒有發送消息給用戶界面指示我方可以行動了。

    ngOnInit(): void {
        this.ge.InitFightStatus();
        this.Message = this.ge.fightStatus.currentActionCharater.Name + "的行動";
        this.ge.fightStatus.ResultEvent.subscribe((x) => {
            if (x === 0) {
                this.FightResultTitle = "團滅了......魂力不足"
                this.ge.gamestatus.lineIdx--;
            } else {
                this.FightResultTitle = "勝利了......奧力給"
                this.ge.gamestatus.lineIdx++;
            }
            this.FightEnd = true;
            console.log("jump to scene");
            setTimeout(() => { this.router.navigateByUrl("scene"); }, 3000);
        }, null, null);
    }

EventEmitter在用戶界面使用subscribe進行訂閱


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

-Advertisement-
Play Games
更多相關文章
  • html代碼 <div class="layui-row" style="margin-top:0.5rem"> <div style="width:90%;margin:0rem auto 0.7rem auto;font-size:0.28rem"> <div style="background ...
  • 今天有群友提出這樣一個問題: ajax請求成功後,然後彈窗提示用戶,接著刷新頁面。結果彈窗一閃而逝,頁面就刷新了,提示的信息用戶都來不及看到。 這樣的問題我之前也用到過,我的解決方法如下: 重點代碼如下: setTimeout(function () { location.href = "www.b ...
  • 通過定位實現二級菜單: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <tit ...
  • 安裝Vuefity的時候,碰到一些坑,經過一番折騰,終於成功,記錄正確的姿勢如下: 創建一個Vue項目: vue init webpack-simple vular-admin 進入項目目錄: cd vular-admin 選擇:Webpack 安裝方式 npm install npm instal ...
  • 一、什麼是Web前端開發? Web前端簡單理解為搭建網頁,也就是通過瀏覽器的方式打開某個網頁,像平時我們打開的一些網頁,比如京東,網易,蘇寧,愛奇藝等,只要是以網頁的形式給大家展示的,都離不開web前端移動開發。 1、更深層次的理解就是,web前端具備三種形式:結構(Html),表現(Css),行為 ...
  • 寫在前面 寫【高併發專題】有一段時間了,一些讀者朋友留言說,併發編程很難,學習了很多的知識,但是在實際工作中卻無從下手。對於一個線上產生的併發問題,又不知產生這個問題的原因究竟是什麼。對於併發編程,感覺上似乎是掌握了,但是真正用起來卻不是那麼回事! 其實,造成這種現象的本質原因就是沒有透徹的理解併發 ...
  • 面向對象(OOP)的理解 喜歡程式的朋友們,大家應該都聽過一句話“萬物皆對象”,感覺老牛X了。 面向對象的程式設計,它是圍繞真實世界來設計程式的。 面向對象三要素:封裝、繼承、多態。 我們可以定義一個類,用來表示小轎車,並且定義小轎車的一些屬性跟行為。然而現實中,有很多種類的車,有公交車、貨車、出租 ...
  • 本文介紹了高德地圖中POI深度信息接入在平臺化過程中的一些思考和實踐,從最開始的單體應用,隨著業務發展面臨挑戰,從業務角度提出解決問題的思路和方案,進而轉化成技術設計並落地實現的過程。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...