日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

基于telepath庫實現Python和JavaScript之間交換數據

瀏覽:50日期:2022-06-19 09:03:44
目錄它有什么作用?安裝方法簡介它有什么作用?

它提供了一種將包括Python對象在內的結構化數據打包為JSON可序列化格式的機制。通過向相應的JavaScript實現注冊該機制,可以擴展該機制以支持任何Python類。然后,打包的數據可以包含在HTTP響應中,并在JavaScript中解壓縮以獲得與原始數據等效的數據結構。

安裝方法

pip install telepath

并將’telepath’添加到項目的INSTALLED_APPS。

簡介

假設我們正在構建一個用于玩跳棋的Django應用。我們已經花費了數天或數周的時間,構建了游戲規則的Python實現,并提供了代表當前游戲狀態和各個部分的類。但是,我們還希望為播放器提供一個適當友好的用戶界面,這意味著該是我們編寫JavaScript前端的時候了。我們的UI代碼不可避免地將擁有自己的對象,這些對象代表不同的角色,鏡像我們正在服務器上跟蹤的數據結構——但我們無法發送Python對象,因此將這些數據發送到客戶端通常將意味著設計游戲狀態的JSON表示形式,并在兩端分別設計大量樣式代碼,遍歷數據結構以在本機對象之間來回轉換。讓我們看看telepath如何簡化該過程。一個完整的跳棋游戲對于本教程來說有點太多了,所以我們只選擇渲染這一步...在Python環境中,創建一個新的Django項目:

pip install 'Django>=3.1,<3.2'django-admin startproject draughtscd draughts./manage.py startapp games

在draughts / settings.py的INSTALLED_APPS列表中添加’games’。為簡單起見,在本示例中,我們將不涉及數據庫,而是將游戲狀態表示為普通的Python類而不是Django模型。修改games/views.py,如下所示:

from django.shortcuts import renderclass Piece: def __init__(self, color, position):self.color = colorself.position = positionclass GameState: def __init__(self, pieces):self.pieces = pieces @staticmethod def new_game():black_pieces = [ Piece(’black’, (x, y)) for y in range(0, 3) for x in range((y + 1) % 2, 8, 2)]white_pieces = [ Piece(’white’, (x, y)) for y in range(5, 8) for x in range((y + 1) % 2, 8, 2)]return GameState(black_pieces + white_pieces)def game(request): game_state = GameState.new_game() return render(request, ’game.html’, {})

如下所示創建games/templates/game.html:

<!doctype html><html> <head><title>Draughts</title><script> document.addEventListener(’DOMContentLoaded’, event => {const gameElement = document.getElementById(’game’);gameElement.innerHTML = ’TODO: render the board here’ });</script> </head> <body><h1>Draughts</h1><div id='game'></div> </body></html>

將新視圖添加到draughts/urls.py:

from django.contrib import adminfrom django.urls import pathfrom games.views import gameurlpatterns = [ path(’’, game), path(’admin/’, admin.site.urls),]

現在,使用./manage.py runserver啟動服務器,并訪問http:// localhost:8000 /。到目前為止,我們已經創建了一個代表新游戲的GameState對象——現在是時候引入telepath,以便我們可以將該對象傳輸到客戶端。執行下面命令:

pip install telepath

并將’telepath’添加到draughts / settings.py中的INSTALLED_APPS列表中。現在編輯games/views.py文件:

import jsonfrom django.shortcuts import renderfrom telepath import JSContext# ...def game(request): game_state = GameState.new_game() js_context = JSContext() packed_game_state = js_context.pack(game_state) game_state_json = json.dumps(packed_game_state) return render(request, ’game.html’, {’game_state_json’: game_state_json, })

這里JSContext是一個幫助工具,用于管理游戲狀態對象到我們可以在Javascript中使用的表示形式的轉換。js_context.pack接受該對象并將其轉換為可以JSON序列化并傳遞到我們的模板的值。但是,現在重新加載頁面失敗,并出現以下形式的錯誤:don’t know how to pack object: <games.views.GameState object at 0x10f3f2490>這是因為GameState是Telepath尚不知道如何處理的自定義Python類型。傳遞給pack的任何自定義類型必須鏈接到相應的JavaScript實現;這是通過定義Adapter對象并將其注冊到telepath來完成的。如下更新game / views.py:

import jsonfrom django.shortcuts import renderfrom telepath import Adapter, JSContext, register# ...class GameState: # keep definition as beforeclass GameStateAdapter(Adapter): js_constructor = ’draughts.GameState’ def js_args(self, game_state):return [game_state.pieces] class Media:js = [’draughts.js’]register(GameStateAdapter(), GameState)

此處js_constructor是JavaScript構造函數的標識符,該標識符將用于在客戶端上構建GameState實例,并且js_args定義了將傳遞給此構造函數的參數列表,以重新創建給定game_state對象的JavaScript對應對象 。Media類指示文件,該文件遵循Django對格式媒體的約定,可在其中找到GameState的JavaScript實現。稍后我們將看到此JavaScript實現的外觀,現在,我們需要為Piece類定義一個類似的適配器,因為我們對GameStateAdapter的定義取決于是否能夠打包Piece實例。將以下定義添加到games/views.py:

class Piece: # keep definition as beforeclass PieceAdapter(Adapter): js_constructor = ’draughts.Piece’ def js_args(self, piece):return [piece.color, piece.position] class Media:js = [’draughts.js’]register(PieceAdapter(), Piece)

重新加載頁面,您將看到錯誤提示消失了,這表明我們已成功將GameState對象序列化為JSON并將其傳遞給模板?,F在,我們可以將其包含在模板中-編輯games/templates/game.html:

<body><h1>Draughts</h1><div id='game' data-game-state='{{ game_state_json }}'></div> </body>

再次重新加載頁面,并在瀏覽器的開發人員工具中檢查游戲元素(在Chrome和Firefox中,右鍵單擊TODO注釋,然后選擇Inspect或Inspect Element),您將看到GameState對象的JSON表示,準備好 解壓成完整的JavaScript對象。除了將數據打包成JSON可序列化的格式外,JSContext對象還跟蹤將數據解壓縮所需的JavaScript媒體定義,作為其媒體屬性。讓我們更新游戲視圖,以將其也傳遞給模板-在games/views.py中:

def game(request): game_state = GameState.new_game() js_context = JSContext() packed_game_state = js_context.pack(game_state) game_state_json = json.dumps(packed_game_state) return render(request, ’game.html’, {’game_state_json’: game_state_json,’media’: js_context.media, })

將下面代碼添加到games / templates / game.html中的HTML頭文件中:

<head><title>Draughts</title>{{ media }}<script> document.addEventListener(’DOMContentLoaded’, event => {const gameElement = document.getElementById(’game’);gameElement.innerHTML = ’TODO: render the board here’ });</script> </head>

重新加載頁面并查看源代碼,您將看到這帶來了兩個JavaScript包括 —— telepath.js(客戶端telepath庫,提供解包機制)和我們在適配器定義中指定的draughts.js文件。后者尚不存在,所以讓我們在games / static / draughts.js中創建它:

class Piece { constructor(color, position) {this.color = color;this.position = position; }}window.telepath.register(’draughts.Piece’, Piece);class GameState { constructor(pieces) {this.pieces = pieces; }}window.telepath.register(’draughts.GameState’, GameState);

這兩個類定義實現了我們先前在適配器對象中聲明的構造函數-構造函數接收的參數是js_args定義的參數。window.telepath.register行將這些類定義附加到通過js_constructor指定的相應標識符。現在,這為我們提供了解壓縮JSON所需的一切-回到games / templates / game.html中,更新JS代碼,如下所示:

<script> document.addEventListener(’DOMContentLoaded’, event => {const gameElement = document.getElementById(’game’);const gameStateJson = gameElement.dataset.gameState;const packedGameState = JSON.parse(gameStateJson);const gameState = window.telepath.unpack(packedGameState);console.log(gameState); })</script>

您可能需要重新啟動服務器以獲取新的games/static文件夾。重新加載頁面,然后在瀏覽器控制臺中,您現在應該看到填充了Piece對象的GameState對象。現在,我們可以繼續在games/static/draughts.js中填寫渲染代碼:

class Piece { constructor(color, position) {this.color = color;this.position = position; } render(container) {const element = document.createElement(’div’);container.appendChild(element);element.style.width = element.style.height = ’24px’;element.style.border = ’2px solid grey’;element.style.borderRadius = ’14px’;element.style.backgroundColor = this.color; }}window.telepath.register(’draughts.Piece’, Piece)class GameState { constructor(pieces) {this.pieces = pieces; } render(container) {const table = document.createElement(’table’);container.appendChild(table);const cells = [];for (let y = 0; y < 8; y++) { let row = document.createElement(’tr’); table.appendChild(row); cells[y] = []; for (let x = 0; x < 8; x++) {let cell = document.createElement(’td’);row.appendChild(cell);cells[y][x] = cell;cell.style.width = cell.style.height = ’32px’;cell.style.backgroundColor = (x + y) % 2 ? ’silver’: ’white’; }}this.pieces.forEach(piece => { const [x, y] = piece.position; const cell = cells[y][x]; piece.render(cell);}); }}window.telepath.register(’draughts.GameState’, GameState)

在games/templates/game.html中添加對render方法的調用:

<script> document.addEventListener(’DOMContentLoaded’, event => {const gameElement = document.getElementById(’game’);const gameStateJson = gameElement.dataset.gameState;const packedGameState = JSON.parse(gameStateJson);const gameState = window.telepath.unpack(packedGameState);gameState.render(gameElement); })</script>

重新加載頁面,您將看到我們的跳棋程序已準備就緒,可以開始游戲了。讓我們快速回顧一下我們已經取得的成果:

我們已經打包和解包了自定義Python / JavaScript類型的數據結構,而無需編寫代碼來遞歸該結構。如果我們的GameState對象變得更復雜(例如,“棋子”列表可能變成棋子和國王對象的混合列表,或者狀態可能包括游戲歷史),則無需重構任何數據打包/拆包邏輯,除了為每個使用的類提供一個適配器對象。 僅提供了解壓縮頁面數據所需的JS文件-如果我們的游戲應用程序擴展到包括Chess,Go和Othello,并且所有生成的類都已通過Telepath注冊,則我們仍然只需要提供與跳棋知識相關的代碼。 即使我們使用任意對象,也不需要動態內聯JavaScript —— 所有動態數據都以JSON形式傳遞,并且所有JavaScript代碼在部署時都是固定的(如果我們的網站強制執行CSP,這一點很重要)。

以上就是基于telepath庫實現Python和JavaScript之間交換數據的詳細內容,更多關于Python和JavaScript之間交換數據的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
精品国产鲁一鲁****| 欧美激情三区| 成人污污视频| 国产夫妻在线| 色一区二区三区| 国产亚洲久久| 国产精品v日韩精品v欧美精品网站| 日本免费在线视频不卡一不卡二| 免费一区二区视频| 一区二区三区午夜视频| 日韩影院精彩在线| 免播放器亚洲一区| 亚洲精品无播放器在线播放| 日韩欧美三区| 牛牛精品成人免费视频| 黑森林国产精品av| 亚洲一区日韩| 日韩精品免费视频一区二区三区 | 国产乱码精品一区二区三区四区| 欧美a一区二区| 黄色在线网站噜噜噜| 久久精品免费一区二区三区 | 国产精品白丝av嫩草影院| 精品国产a一区二区三区v免费| 日韩亚洲一区在线| 久久av一区| 麻豆传媒一区二区三区| 欧美中文一区二区| 性欧美xxxx免费岛国不卡电影| 三级欧美韩日大片在线看| 免费日韩一区二区三区| 亚洲夜间福利| 你懂的亚洲视频| 国产高清久久| 你懂的国产精品| 91国语精品自产拍| 免费日韩一区二区三区| 伊人精品视频| 日韩中文在线电影| 日韩av中文在线观看| 91精品高清| 首页国产精品| 欧美日韩一区二区三区在线电影| 国产精品传媒麻豆hd| 欧美日韩四区| 国产精品黑丝在线播放| 国产日产精品一区二区三区四区的观看方式| 欧美黄色网页| 精品国产精品国产偷麻豆| 久久精品99国产精品| 亚洲欧美日韩国产一区| 国产99久久| 成人亚洲欧美| 久久av综合| 国产麻豆一区二区三区| 日本欧美在线看| 美日韩精品视频| 国产精品呻吟| 成人综合一区| аⅴ资源天堂资源库在线| 国产欧美日韩一区二区三区四区| 亚洲一区av| 视频一区在线视频| 亚洲一区二区小说| 美女久久一区| 日本精品一区二区三区在线观看视频| 亚洲精品国产精品粉嫩| 亚洲三级精品| 91精品国产自产在线丝袜啪| 欧美日韩亚洲三区| 久久久亚洲欧洲日产| 国产精品xxx| 国产精品原创| 精品一区二区三区在线观看视频| 美女视频网站久久| 日韩不卡在线| 亚洲一区激情| 欧美日韩午夜| 日韩免费福利视频| 亚洲激情社区| 日韩av一区二区三区| 美女精品久久| 91精品国产调教在线观看| 一区免费在线| 日韩国产精品久久久| 日韩88av| 亚洲精品日韩久久| 久久永久免费| 久久精品国产大片免费观看| 日韩中文字幕麻豆| 精品三区视频| 99国产一区| 国产亚洲久久| 黄色欧美日韩| 激情久久99| 在线日韩一区| 国产精品成人3p一区二区三区| 久久一级电影| 国产精品入口久久| 国产亚洲精品久久久久婷婷瑜伽| 国产精品chinese| 久久最新视频| 日韩中文字幕高清在线观看| 欧美日韩激情| 日本久久综合| 国产福利一区二区精品秒拍| 国产综合亚洲精品一区二| 国产精品夜夜夜| 国产精品美女| 亚洲精品国产嫩草在线观看| 国产欧美日韩一级| 视频国产精品| 日韩一区二区免费看| 天堂√8在线中文| 国产欧美三级| 亚洲日本三级| 在线视频日韩| 99精品小视频| 精品日产乱码久久久久久仙踪林| 日韩1区2区3区| 久久午夜影视| 男人的天堂亚洲一区| 激情综合网址| 久久精品一区二区不卡| 国产精品久久久久久久免费观看| 国产精品成人国产| 国产伦精品一区二区三区在线播放 | 视频一区二区欧美| 亚洲精品极品少妇16p| 蜜桃成人av| 在线日韩欧美| 2023国产精品久久久精品双| 久久精品亚洲人成影院 | 久久精品xxxxx| 欧美aⅴ一区二区三区视频| 欧美激情视频一区二区三区免费| 国产亚洲精aa在线看| 国产欧美日韩| 精品午夜视频| 久久久久久美女精品| 99久久精品国产亚洲精品| 国产欧美日韩精品一区二区免费| 日韩有吗在线观看| 麻豆一区二区三| 在线一区视频观看| 欧美成人高清| 无码日韩精品一区二区免费| 久久亚洲电影| 欧美激情 亚洲a∨综合| 性感美女一区二区在线观看| 欧美成人午夜| 亚洲18在线| 麻豆国产欧美一区二区三区 | 国产亚洲午夜| 欧美欧美黄在线二区| 日韩福利视频导航| 久久99国产精品视频| 亚洲人成在线网站| 日韩一区二区免费看| 国产精品毛片aⅴ一区二区三区| 成人在线视频免费看| 黑丝一区二区三区| 久久精品一区二区国产| 自拍日韩欧美| 牛牛精品成人免费视频| 欧美日韩在线播放视频| 欧美亚洲自偷自偷| 色天使综合视频| 日本特黄久久久高潮| 三级精品视频| 日本欧美在线看| 久久裸体视频| 国产剧情在线观看一区| 国产精品婷婷| 九九精品调教| 欧美精品不卡| 中文字幕av一区二区三区人 | 国产精品腿扒开做爽爽爽挤奶网站| 国产精品麻豆成人av电影艾秋| 黄色欧美日韩| 精品欧美一区二区三区在线观看| 久久成人精品| 久久国产亚洲| av免费不卡国产观看| 日韩区欧美区| 中文一区一区三区免费在线观| 亚洲成人不卡| 久久一区视频| 红杏一区二区三区| 国产精品成人一区二区网站软件| 亚洲精品在线a| 亚洲一区二区日韩| 日韩精品一级二级| 日韩影院精彩在线| 国产精品蜜月aⅴ在线| 亚洲一区二区三区高清不卡| 少妇精品久久久一区二区| 日韩视频一区| 国产视频一区在线观看一区免费| 欧美/亚洲一区|