21.1 Python 使用PEfile分析PE文件

来源:https://www.cnblogs.com/LyShark/archive/2023/10/19/17775685.html
-Advertisement-
Play Games

PeFile模塊是`Python`中一個強大的攜帶型第三方`PE`格式分析工具,用於解析和處理`Windows`可執行文件。該模塊提供了一系列的API介面,使得用戶可以通過`Python`腳本來讀取和分析PE文件的結構,包括文件頭、節表、導入表、導出表、資源表、重定位表等等。此外,PEfile模塊還... ...


PeFile模塊是Python中一個強大的攜帶型第三方PE格式分析工具,用於解析和處理Windows可執行文件。該模塊提供了一系列的API介面,使得用戶可以通過Python腳本來讀取和分析PE文件的結構,包括文件頭、節表、導入表、導出表、資源表、重定位表等等。此外,PEfile模塊還可以幫助用戶進行一些惡意代碼分析,比如提取樣本中的字元串、獲取函數列表、重構導入表、反混淆等等。PEfile模塊是Python中處理PE文件的重要工具之一,廣泛應用於二進位分析、安全研究和軟體逆向工程等領域。

由於該模塊為第三方模塊,在使用之前讀者需要在命令行下執行pip install pefile命令安裝第三方庫,當安裝成功後即可正常使用,如下所示則是該模塊的基本使用方法,讀者可自行學習理解。

21.1.1 打開並載入PE文件

如下這段代碼封裝並實現了OpenPeFile函數,可用於打開一個PE文件,在其內部首先判斷了可執行文件是否被壓縮如果被壓縮則會通過zipfile模塊將壓縮包讀入記憶體並調用C2BIP3函數將數據集轉換為2位元組,接著再執行pefile.PE()函數,該函數可用於將可執行文件載入,至此讀者可在主函數內通過pe.dump_dict()的方式輸出該PE文件的所有參數,由於輸出的是字典,讀者可以使用字典與列表的方式靈活的提取出該程式的所有參數信息。

import sys
import zipfile
import pefile

# 如果是Python3則轉換為2位元組
def C2BIP3(string):
    if sys.version_info[0] > 2:
        return bytes([ord(x) for x in string])
    else:
        return string

# 打開文件
def OpenPeFile(filename):

    # 判斷是否是ZIP壓縮包
    if filename.lower().endswith('.zip'):
        try:
            oZipfile = zipfile.ZipFile(filename, 'r')
            file = oZipfile.open(oZipfile.infolist()[0], 'r', C2BIP3('infected'))
        except Exception:
            print(sys.exc_info()[1])
            sys.exit()
        oPE = pefile.PE(data=file.read())
        file.close()
        oZipfile.close()

    # 如果是空則
    elif filename == '':
        oPE = False
        return oPE
    # 否則直接打開文件
    else:
        oPE = pefile.PE(filename)
    return oPE

if __name__ == "__main__":
    pe = OpenPeFile("d://lyshark.exe")
    print(pe.FILE_HEADER.dump())
    print(pe.dump_dict())

21.1.2 解析PE頭部數據

如下代碼實現瞭解析PE結構中頭部基本數據,在GetHeader函數內,我們首先通過pe.FILE_HEADER.Machine成員判斷當前讀入的文件的位數信息,通過pe.FILE_HEADER.Characteristics可判斷PE文件的類型,通常為EXE可執行文件或DLL動態鏈接庫文件,通過AddressOfEntryPoint加上ImageBase則可獲取到程式的實際裝載地址,壓縮數據的計算可通過hashlib模塊對PE文件位元組數據進行計算摘要獲取,最後是附加數據,通過get_overlay_data_start_offset則可獲取到,並依次迴圈即可輸出所有附加數據。

import hashlib
import pefile

# 計算得到數據長度,自動使用推薦大小
def NumberOfBytesHumanRepresentation(value):
    if value <= 1024:
        return '%s bytes' % value
    elif value < 1024 * 1024:
        return '%.1f KB' % (float(value) / 1024.0)
    elif value < 1024 * 1024 * 1024:
        return '%.1f MB' % (float(value) / 1024.0 / 1024.0)
    else:
        return '%.1f GB' % (float(value) / 1024.0 / 1024.0 / 1024.0)

# 獲取PE頭部基本信息
def GetHeader(pe):
    raw = pe.write()

    # 掃描基本信息
    print("-" * 50)
    print("程式基本信息")
    print("-" * 50)
    if (hex(pe.FILE_HEADER.Machine) == "0x14c"):
        print("程式位數: {}".format("x86"))
    if (hex(pe.FILE_HEADER.Machine) == "0x8664"):
        print("程式位數: {}".format("x64"))

    if (hex(pe.FILE_HEADER.Characteristics) == "0x102"):
        print("程式類型: Executable")
    elif (hex(pe.FILE_HEADER.Characteristics) == "0x2102"):
        print("程式類型: Dynamic link library")

    if pe.OPTIONAL_HEADER.AddressOfEntryPoint:
        oep = pe.OPTIONAL_HEADER.AddressOfEntryPoint + pe.OPTIONAL_HEADER.ImageBase
        print("實際入口: {}".format(hex(oep)))

    print("映像基址: {}".format(hex(pe.OPTIONAL_HEADER.ImageBase)))
    print("虛擬入口: {}".format(hex(pe.OPTIONAL_HEADER.AddressOfEntryPoint)))
    print("映像大小: {}".format(hex(pe.OPTIONAL_HEADER.SizeOfImage)))
    print("區段對齊: {}".format(hex(pe.OPTIONAL_HEADER.SectionAlignment)))
    print("文件對齊: {}".format(hex(pe.OPTIONAL_HEADER.FileAlignment)))
    print("區塊數量: {}".format(int(pe.FILE_HEADER.NumberOfSections + 1)))
    print('熵值比例: %f (Min=0.0, Max=8.0)' % pe.sections[0].entropy_H(raw))

    # 計算壓縮數據
    print("-" * 50)
    print("計算壓縮數據")
    print("-" * 50)
    print('MD5     : %s' % hashlib.md5(raw).hexdigest())
    print('SHA-1   : %s' % hashlib.sha1(raw).hexdigest())
    print('SHA-256 : %s' % hashlib.sha256(raw).hexdigest())
    print('SHA-512 : %s' % hashlib.sha512(raw).hexdigest())

    # 掃描文件末尾是否存在附加數據
    print("-" * 50)
    print("掃描附加數據")
    print("-" * 50)
    overlayOffset = pe.get_overlay_data_start_offset()
    if overlayOffset != None:
        print("起始文件位置: 0x%08x"%overlayOffset)
        overlaySize = len(raw[overlayOffset:])
        print("長度: 0x%08x %s %.2f%%"%(overlaySize, NumberOfBytesHumanRepresentation(overlaySize), float(overlaySize) / float(len(raw)) * 100.0))
        print("MD5: %s" %hashlib.md5(raw[overlayOffset:]).hexdigest())
        print("SHA-256: %s" %hashlib.sha256(raw[overlayOffset:]).hexdigest())

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")
    GetHeader(pe)

21.1.3 解析節表數據

運用PEFile模塊解析節表也很容易,如下代碼中分別實現了兩個功能函數,函數ScanSection()用於輸出當前文件的所有節表數據,其中通過pe.FILE_HEADER.NumberOfSections得到節表數量,並通過迴圈的方式依次解析pe.sections中的每一個節中元素,函數CheckSection()則可用於計算PE文件節大小以及節MD5值,完整代碼如下所示;

import hashlib
import pefile

# 計算得到數據長度,自動使用推薦大小
def NumberOfBytesHumanRepresentation(value):
    if value <= 1024:
        return '%s bytes' % value
    elif value < 1024 * 1024:
        return '%.1f KB' % (float(value) / 1024.0)
    elif value < 1024 * 1024 * 1024:
        return '%.1f MB' % (float(value) / 1024.0 / 1024.0)
    else:
        return '%.1f GB' % (float(value) / 1024.0 / 1024.0 / 1024.0)

# 輸出所有的節
def ScanSection(pe):
    print("-" * 100)
    print("{:10s}{:10s}{:10s}{:10s}{:10s}{:10s}{:10s}{:10s}".
          format("序號","節區名稱","虛擬偏移","虛擬大小","實際偏移","實際大小","節區屬性","熵值"))
    print("-" * 100)
    section_count = int(pe.FILE_HEADER.NumberOfSections + 1)

    for count,item in zip(range(1,section_count),pe.sections):
        print("%d\t\t\t%-10s\t0x%.8X\t0x%.8X\t0x%.8X\t0x%.8X\t0x%.8X\t%f"
              %(count,(item.Name).decode("utf-8"),item.VirtualAddress,item.Misc_VirtualSize,item.PointerToRawData,item.SizeOfRawData,item.Characteristics,item.get_entropy()))
    print("-" * 100)

# 計算所有節的MD5
def CheckSection(pe):
    print("-" * 100)
    print("序號\t\t節名稱\t\t文件偏移\t\t大小\t\tMD5\t\t\t\t\t\t\t\t\t\t節大小")
    print("-" * 100)

    # 讀取PE文件到記憶體
    image_data = pe.get_memory_mapped_image()

    section_count = int(pe.FILE_HEADER.NumberOfSections + 1)
    for count,item in zip(range(1,section_count),pe.sections):

        section_data = image_data[item.PointerToRawData: item.PointerToRawData + item.SizeOfRawData - 1]
        data_size = NumberOfBytesHumanRepresentation(len(section_data))
        hash_value = hashlib.md5(section_data).hexdigest()
        print("{}\t{:10s}\t{:10X}\t{:10X}\t{:30s}\t{}".format(count,(item.Name).decode("utf-8"),item.PointerToRawData,item.SizeOfRawData,hash_value,data_size))
    print("-" * 100)

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")
    ScanSection(pe)
    CheckSection(pe)

21.1.4 節區RVA與FOA互轉

此處計算節偏移地址,相信讀者能理解,在之前的文章中我們詳細的介紹了PE文件如何進行RVAFOA以及VA之間的轉換的,如果是在平時的惡意代碼分析中需要快速實現轉換那麼使用Python將是一個不錯的選擇,如下代碼中RVAToFOA可將一個RVA相對地址轉換為FOA文件偏移,FOAToRVA則可實現將一個FOA文件偏移轉換為RVA先對地址,當然PeFile模塊內也提供了get_rva_from_offset實現從FOA轉RVA,get_offset_from_rva則是從RVA到FOA,讀者可自行選擇不同的轉換方式。

import pefile

# 將RVA轉換為FOA的函數
def RVAToFOA(pe,rva):
    for item in pe.sections:
        Section_Start = item.VirtualAddress
        Section_Ends = item.VirtualAddress + item.SizeOfRawData
        if rva >= Section_Start and rva < Section_Ends:
            return rva - item.VirtualAddress + item.PointerToRawData
    return -1

# 將FOA文件偏移轉換為RVA相對地址
def FOAToRVA(pe,foa):
    ImageBase = pe.OPTIONAL_HEADER.ImageBase
    NumberOfSectionsCount = pe.FILE_HEADER.NumberOfSections

    for index in range(0,NumberOfSectionsCount):
        PointerRawStart = pe.sections[index].PointerToRawData
        PointerRawEnds = pe.sections[index].PointerToRawData + pe.sections[index].SizeOfRawData

        if foa >= PointerRawStart and foa <= PointerRawEnds:
            rva = pe.sections[index].VirtualAddress + (foa - pe.sections[index].PointerToRawData)
            return rva
    return -1

# 內部功能實現FOA->RVA互轉
def inside(pe):
    # 從FOA獲取RVA 傳入十進位
    rva = pe.get_rva_from_offset(3952)
    print("對應記憶體RVA: {}".format(hex(rva)))

    # 從RVA獲取FOA 傳入十進位
    foa = pe.get_offset_from_rva(rva)
    print("對應文件FOA: {}".format(foa))

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")
    ref = RVAToFOA(pe,4128)
    print("RVA轉FOA => 輸出十進位: {}".format(ref))

    ref = FOAToRVA(pe,1056)
    print("FOA轉RVA => 輸出十進位: {}".format(ref))

21.1.5 解析數據為Hex格式

如下代碼片段實現了對PE文件的各種十六進位操作功能,封裝cDump()類,該類內由多個類函數可以使用,其中HexDump()可用於將讀入的PE文件以16進位方式輸出,HexAsciiDump()則可用於輸出十六進位以及所對應的ASCII格式,GetSectionHex()用於找到PE文件的.text節,並將此節內的數據讀入到記憶體中,這段代碼可以很好的實現對PE文件的十六進位輸出與解析,讀者可在實際開發中使用。

import pefile
from io import StringIO
import sys
import re

dumplinelength = 16

def CIC(expression):
    if callable(expression):
        return expression()
    else:
        return expression

def IFF(expression, valueTrue, valueFalse):
    if expression:
        return CIC(valueTrue)
    else:
        return CIC(valueFalse)

class cDump():
    def __init__(self, data, prefix='', offset=0, dumplinelength=16):
        self.data = data
        self.prefix = prefix
        self.offset = offset
        self.dumplinelength = dumplinelength

    # 輸出指定位置的十六進位格式
    def HexDump(self):
        oDumpStream = self.cDumpStream(self.prefix)
        hexDump = ''
        for i, b in enumerate(self.data):
            if i % self.dumplinelength == 0 and hexDump != '':
                oDumpStream.Addline(hexDump)
                hexDump = ''
            hexDump += IFF(hexDump == '', '', ' ') + '%02X' % self.C2IIP2(b)
        oDumpStream.Addline(hexDump)
        return oDumpStream.Content()

    def CombineHexAscii(self, hexDump, asciiDump):
        if hexDump == '':
            return ''
        countSpaces = 3 * (self.dumplinelength - len(asciiDump))
        if len(asciiDump) <= self.dumplinelength / 2:
            countSpaces += 1
        return hexDump + '  ' + (' ' * countSpaces) + asciiDump

    # 輸出指定位置的十六進位格式以及ASCII字元串
    def HexAsciiDump(self):
        oDumpStream = self.cDumpStream(self.prefix)
        hexDump = ''
        asciiDump = ''
        for i, b in enumerate(self.data):
            b = self.C2IIP2(b)
            if i % self.dumplinelength == 0:
                if hexDump != '':
                    oDumpStream.Addline(self.CombineHexAscii(hexDump, asciiDump))
                hexDump = '%08X:' % (i + self.offset)
                asciiDump = ''
            if i % self.dumplinelength == self.dumplinelength / 2:
                hexDump += ' '
            hexDump += ' %02X' % b
            asciiDump += IFF(b >= 32 and b <= 128, chr(b), '.')
        oDumpStream.Addline(self.CombineHexAscii(hexDump, asciiDump))
        return oDumpStream.Content()

    class cDumpStream():
        def __init__(self, prefix=''):
            self.oStringIO = StringIO()
            self.prefix = prefix

        def Addline(self, line):
            if line != '':
                self.oStringIO.write(self.prefix + line + '\n')

        def Content(self):
            return self.oStringIO.getvalue()

    @staticmethod
    def C2IIP2(data):
        if sys.version_info[0] > 2:
            return data
        else:
            return ord(data)

# 只輸出十六進位數據
def HexDump(data):
    return cDump(data, dumplinelength=dumplinelength).HexDump()

# 輸出十六進位與ASCII字元串
def HexAsciiDump(data):
    return cDump(data, dumplinelength=dumplinelength).HexAsciiDump()

# 找到指定節並讀取hex數據
def GetSectionHex(pe):
    ImageBase = pe.OPTIONAL_HEADER.ImageBase
    for item in pe.sections:
        # 判斷是否是.text節
        if str(item.Name.decode('UTF-8').strip(b'\x00'.decode())) == ".text":
            # print("虛擬地址: 0x%.8X 虛擬大小: 0x%.8X" %(item.VirtualAddress,item.Misc_VirtualSize))
            VirtualAddress = item.VirtualAddress
            VirtualSize = item.Misc_VirtualSize
            ActualOffset = item.PointerToRawData

            StartVA = hex(ImageBase + VirtualAddress)
            StopVA = hex(ImageBase + VirtualAddress + VirtualSize)
            print("[+] 代碼段起始地址: {} 結束: {} 實際偏移:{} 長度: {}".format(StartVA, StopVA, ActualOffset, VirtualSize))

            # 獲取到.text節區間內的數據
            hex_code = pe.write()[ActualOffset: VirtualSize]
            return hex_code
        else:
            print("程式中不存在.text節")
            return 0
    return 0

REGEX_STANDARD = '[\x09\x20-\x7E]'

def ExtractStringsASCII(data):
    regex = REGEX_STANDARD + '{%d,}'
    return re.findall(regex % 4, data)

def ExtractStringsUNICODE(data):
    regex = '((' + REGEX_STANDARD + '\x00){%d,})'
    return [foundunicodestring.replace('\x00', '') for foundunicodestring, dummy in re.findall(regex % 4, data)]

# 將傳入Hex字元串以每16字元分割在一個列表內
def ExtractStrings(data):
    return ExtractStringsASCII(data) + ExtractStringsUNICODE(data)

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")

    # 得到.text節內數據
    ref = GetSectionHex(pe)

    # 轉為十六進位格式
    dump_hex = HexDump(ref)
    print(dump_hex)

    # 打包為每16字元一個列表
    dump_list = ExtractStrings(dump_hex)

    print(dump_list)

21.1.6 解析數據目錄表

數據目錄表用於記錄可執行文件的數據目錄項在文件中的位置和大小。數據目錄表共有16個條目,每個條目都對應著一個數據目錄項,每個數據目錄項都描述了可執行文件中某一部分的位置和大小。

數據目錄表的解析可以使用pe.OPTIONAL_HEADER.NumberOfRvaAndSizes首先獲取到數據目錄表的個數,接著二通過迴圈個數依次解包OPTIONAL_HEADER.DATA_DIRECTORY裡面的每一個列表,在迴圈列表時依次解包輸出即可。

import pefile

# 將RVA轉換為FOA的函數
def RVAToFOA(pe,rva):
    for item in pe.sections:
        Section_Start = item.VirtualAddress
        Section_Ends = item.VirtualAddress + item.SizeOfRawData
        if rva >= Section_Start and rva < Section_Ends:
            return rva - item.VirtualAddress + item.PointerToRawData
    return -1

# 掃描數據目錄表
def ScanOptional(pe):
    optional_size = pe.OPTIONAL_HEADER.NumberOfRvaAndSizes
    print("數據目錄表個數: {}".format(optional_size))

    print("-" * 100)
    print("編號 \t\t\t 目錄RVA\t\t 目錄FOA\t\t\t 長度\t\t 描述信息")
    print("-" * 100)

    for index in range(0,optional_size):
        va = int(pe.OPTIONAL_HEADER.DATA_DIRECTORY[index].VirtualAddress)
        print("%03d \t\t 0x%08X\t\t 0x%08X\t\t %08d \t\t"
              %(index,
                pe.OPTIONAL_HEADER.DATA_DIRECTORY[index].VirtualAddress,
                RVAToFOA(pe,va),
                pe.OPTIONAL_HEADER.DATA_DIRECTORY[index].Size
                ),end="")

        if index == 0:
            print("Export symbols")
        if index == 1:
            print("Import symbols")
        if index == 2:
            print("Resources")
        if index == 3:
            print("Exception")
        if index == 4:
            print("Security")
        if index == 5:
            print("Base relocation")
        if index == 6:
            print("Debug")
        if index == 7:
            print("Copyright string")
        if index == 8:
            print("Globalptr")
        if index == 9:
            print("Thread local storage (TLS)")
        if index == 10:
            print("Load configuration")
        if index == 11:
            print("Bound Import")
        if index == 12:
            print("Import Address Table")
        if index == 13:
            print("Delay Import")
        if index == 14:
            print("COM descriptor")
        if index == 15:
            print("NoUse")

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")
    ScanOptional(pe)

21.1.7 解析導入導出表

導入表和導出表都是PE文件中的重要數據結構,分別記錄著一個模塊所導入和導出的函數和數據,如下所示則是使用PeFile模塊實現對導入表與導出表的解析工作,對於導入表ScanImport的解析需要通過pe.DIRECTORY_ENTRY_IMPORT獲取到完整的導入目錄,並通過迴圈的方式輸出x.imports中的數據即可,而對於導出表ScanExport則需要在pe.DIRECTORY_ENTRY_EXPORT.symbols導出符號中解析獲取。

import pefile

# 輸出所有導入表模塊
def ScanImport(pe):
    print("-" * 100)
    try:
        for x in pe.DIRECTORY_ENTRY_IMPORT:
            for y in x.imports:
                print("[*] 模塊名稱: %-20s 導入函數: %-14s" %((x.dll).decode("utf-8"),(y.name).decode("utf-8")))
    except Exception:
        pass
    print("-" * 100)

# 輸出所有導出表模塊
def ScanExport(pe):
    print("-" * 100)
    try:
        for exp in pe.DIRECTORY_ENTRY_EXPORT.symbols:
            print("[*] 導出序號: %-5s 模塊地址: %-20s 模塊名稱: %-15s"
            %(exp.ordinal,hex(pe.OPTIONAL_HEADER.ImageBase + exp.address),(exp.name).decode("utf-8")))
    except:
        pass
    print("-" * 100)

if __name__ == "__main__":
    pe = pefile.PE("d://lyshark.exe")
    ScanImport(pe)
    ScanExport(pe)

本文作者: 王瑞
本文鏈接: https://www.lyshark.com/post/92a3370c.html
版權聲明: 本博客所有文章除特別聲明外,均採用 BY-NC-SA 許可協議。轉載請註明出處!

文章作者:lyshark
文章出處:https://www.cnblogs.com/LyShark/p/17775685.html
本博客所有文章除特別聲明外,均採用 BY-NC-SA 許可協議。轉載請註明出處!
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. vue 使用快速入門三步走 (1) 新建 HTML 頁面,引入 Vue.js文件 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Vue.js 入門示例</title> <script src="https://cdn.j ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 題目 給定兩個數組,判斷兩數組內容是否相等。 不使用排序 不考慮元素位置 例: [1, 2, 3] 和 [1, 3, 2] // true [1, 2, 3] 和 [1, 2, 4] // false 思考幾秒:有了😀😀 1. 直接遍 ...
  • ruoyi框架的vue版本中,對字典的回顯樣式的設計,預設有以下幾種 如果希望添加一種紅色字體的,可以這樣實現,實現後你的回顯就多了一種紅色字體的樣式了 具體實現的方法 在app.vue中,添加對象的css樣式 <style type="text/css"> .el-tag--redColorFon ...
  • 接下來,我們將會用 Vue3 建造響應式的方法,從頭開始製造一個響應式引擎,讓我們一步一步的來解決這個問題! ...
  • 翻出老物件,搭建一個簡單的 IOT 開發環境,也算是廢物利用了 ,接下來加感測器。1. STM32 採集數據: RTOS。 資源相對比較豐富,可以根據項目需求定製。2. ESP32 網路傳輸(AT固件 MQTT協議) : AT:封裝好的介面,擴展性不是那麼好,業務簡單的話將就可以用。 SDK:介面比 ...
  • 代碼可視化是通過使用圖形化手段(架構圖、依賴圖、分散式追蹤、類圖、火焰圖、CallGraph等)使代碼在某些特征上變得可觀測,用於輔助開發人員理解分析項目或建設一些自動化工具。 ...
  • 上游服務和下游服務 在網路通信中,數據流的方向確實通常是由上游到下游,因此,下游服務接收請求併發送響應,而上游服務發送請求並接收響應。感謝您的指正,對於瞭解和描述數據流的方向非常重要,而上游服務通常是請求的發起方,下游服務通常是響應的接收方。 以nginx為例說一下 瀏覽器發去某個功能變數名稱,到達DNS解 ...
  • MySQL欄位的字元類型該如何選擇?千萬數據下varchar和char性能竟然相差30%? 前言 上篇文章MySQL欄位的時間類型該如何選擇?千萬數據下性能提升10%~30%🚀我們討論過時間類型的選擇 本篇文章來討論MySQL中字元類型的選擇並來深入實踐char與varchar類型的最佳使用場景 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...