Penjelasan Beranotasi Simon Willison tentang Dataklasses David Beazley
David Beazley di Twitter:
Jadi, saya baru saja menerbitkan putaran jahat yang nikmat ini di kelas data. Ini kecil dan kelas yang dihasilkan mengimpor sekitar 15-20 lebih cepat daripada kelas data. https://github.com/dabeaz/dataklasses
Saya memutuskan untuk membuat versi kode aslinya yang beranotasi berat untuk mencari tahu sendiri cara kerjanya.
"x", "y") def __init__ (diri sendiri, x, y): diri sendiri.x=xdiri sendiri.y=y def __repr__( diri sendiri): kembalif"Koordinat({diri sendiri.x!r}, {diri sendiri.y!R})"def__eq__(diri sendiri, lainnya): jikasendiri.__kelas__adalahlainnya. __kelas__: kembali (diri sendiri,x, diri sendiri.y,) == (lainnya.x , lainnya.y,) kalau tidak: kembaliTidak Diimplementasikan
Trik kunci: pembuatan kode
Kunci untuk memahami cara kerja kode adalah dengan memahami bahwa itu menggunakan pembuatan kode. David mengintrospeksi properti kelas beranotasi, dari kelas itu sendiri dan superclass mana pun, lalu menghitung jumlahnya. Dia kemudian menghasilkan metode yang terlihat seperti ini:
def__init__(diri sendiri, _0, _1): sendiri._0=_0diri sendiri ._1=_1
Kemudian gunakan func.__code__.replace(co_names=repl_co_names, co_varnames=repl_co_varnames) metode untuk mengganti nama itu _0 dan _1 variabel kembali ke x dan y.
Saya belum mengerti mengapa dia melakukan ini sebagai lawan menghasilkan kode menggunakan x dan y secara langsung. Saya akan memperbarui ini dengan penjelasan setelah saya menemukannya!
Pembaruan: Jacob Kaplan-Moss menjelaskannya di sini!
Itu benar-benar liar trik inti dari ini, dan apa yang membuatnya cepat:
bytecode untuk
__init__(sendiri, x, y)
persis sama dengan bytecode untuk __init__(self, foo, bar)!
Jadi itu berarti Dave dapat men-cache kode yang dihasilkan untuk "any __init__ berfungsi dengan dua argumen" dan kemudian
re -gunakan kode yang sama persis untuk 2-arity selanjutnya __init__s!
Salinan beranotasi saya
Pertama, termasuk pesan hak cipta seperti yang dipersyaratkan oleh pesan hak cipta:
# dataklasses.py # # https://github.com/dabeaz/dataklasses## Pengarang : David Beazley (@dabeaz).# http://www.dabeaz.com## Hak cipta (C) 2021-2022. ## Izin diberikan untuk menggunakan , salin, dan ubah kode ini di any # cara selama pesan hak cipta dan disclaimer ini tetap ada # kode sumber. Tidak ada garansi. Coba gunakan kode untuk # lebih baik.
Anotasi saya dimulai di sini (Saya juga menjalankannya hingga Hitam).
__semua__=# Ini memastikan "dari impor dataklasses *" hanya akan mengimpor simbol dataklass darifungsiimpor
lru_cache
,
mengurangi # Dekorator ini mengambil fungsi yang mengembalikan kode sumber Python dan mengkompilasinya # itu menjadi fungsi Python. Jadi jika func(fields) mengembalikan string berikut: ## def __init__(sendiri, a): # diri.a=a# # Nilai yang dikembalikan adalah itu fungsi yang dikompilasi. Dengan beberapa penyempurnaan.defkodegen(fungsi): # Ini menyimpan hasil jadi jika Anda melewati numfields yang sama tidak perlu # melakukan pekerjaan yang sama dua kali .## Ini adalah pengoptimalan kinerja utama untuk kode ini – ini berarti # kode yang dihasilkan sama dapat digunakan kembali untuk setiap kelas yang memiliki kesamaan # jumlah argumen!@lru_cachedefmake_func_code(numfields) : # numfields di len(fields) nama= [f”_{n}“forninrange(numfields)] # nama sekarang [“_0”, “_1”, “_2”] tergantung pada numfields # # Kami memanggil func() dengan daftar nama itu dan exec() the# mengembalikan kode sumber. Argumen ketiga untuk exec() adalah# locals() – ini adalah kamus yang fungsi exec() # akan diisi dengan simbol baru. ## Kami menggunakan d:={} operator walrus di sini sehingga kami dapat# lihat kamus itu dengan nama d sebentar lagi.eksekutif(
func(
nama), global(), d := {}) # d.popitem() kembali pasangan (kunci, nilai) pertama di # kamus. Kita tahu bahwa kamus locals() # hanya akan memiliki satu simbol di dalamnya, karena# kita sekarang bahwa kode dikembalikan oleh fungsi(nama) saja# mendefinisikan satu fungsi. Jadi ini mengembalikan nilai# dari item pertama di locals(), function objectkembalid.popitem
()[1] # Dekorasi ini() adalah nilai kembalian dari dekorator,# yang artinya kode berikut: # # @codegen# def make__init__(bidang): # # …# Mengubah make__init__ menjadi fungsi yang ditentukan oleh dekorasi()defdekorasi(bidang): # Seperti yang ditunjukkan di atas, ini menggunakan exec() untuk mengkompilasi dan mengembalikan # badan fungsi yang dibuat menggunakan kode sumber yang dihasilkan fungsi=make_func_code(
len(
bidang)) # Tapi ingat: karena kami men-cache dan menggunakan kembali badan metode, ini memiliki # jelek _0, _1 parameter dan variabel yang ingin kita buat lebih bagus. # co_names: tuple nama selain argumen dan fungsi lokalco_names=fungsi.__code__.co_name s# Untuk contoh Koordinat, nama_bersama=(“_0”, “_1”)# co_varnames: tuple nama argumen dan variabel lokal co_varnames = fungsi.__kode__.co_varnames# Untuk contoh Koordinat, co_varnames=(“self”, “_0”, “_1” )# Kami akan mengganti co_names dan co_varnames dengan modifikasi# versi – jadi kita perlu membuat dua tupel pengganti untuk mereka. # # start :=co_names.index(“_0”) menggunakan operator walrus untuk keduanya# cari tahu indeks berbasis 0 dari simbol _0 dan menetapkannya ke s# # Kami menggunakan (*a, *b, *c) untuk membuat tupel baru yang merupakan hasil dari # menggabungkan ketiga tupel input tersebut.repl_co_names
= (
# Iris semuanya dalam co_names hingga simbol _0 pertama
*co_names[start+len(fields) :],
# Kemudian masukkan bidang, yang merupakan properti kelas beranotasi *bidang, # Sekarang semuanya dalam co_names berikut *co_names[start+len(fields) :], ) # Untuk Koordinat ini sekarang (“x”, “y”)repl_co_varnames= ( # Kami hanya memodifikasi co_varnames jika _0 adalah salah satunya, jika tidak # kami membiarkannya tidak berubah. ( *co_varnames[: (start:=co_varnames.index(“_0”))], *
bidang,
*
co_varnames[start+len(fields) :], ) jika“_0”dico_varnameselseco_varnames ) # Untuk Koordinat ini sekarang (“self”, “x”, “y”)# type(func) mengembalikan objek internal Python yang disebut “fungsi”# yang dapat dipanggil dan memiliki tanda tangan fungsi ini: ## fungsi(kode, global, nama=Tidak ada, argdefs=Tidak ada, penutupan=Tidak ada)## Ini menciptakan objek fungsi baru kembaliTipe(fungsi)( # func.__code__.replace docstring mengatakan: # “Kembalikan salinan objek kode dengan nilai baru untuk bidang yang ditentukan” # Jadi kami menulis ulang co_names dan co_varnames di sini fungsi.__kode__.mengganti(co_names=
repl_co_names
, co_varnames=repl_co_varnames ), fungsi.__globals__, ) kembalidekorasidefall_hints(cls): # Ini mengintrospeksi kelas untuk menemukan semua anggota kelas beranotasi # Dalam Koordinat cas e ini mengembalikan {‘x’: , ‘y’: } # # “x” untuk lambda itu dimulai sebagai {} kamus kosong itu dan setiap kali# adalah kamus dengan hal-hal baru yang ditambahkan ke dalamnya.## reversed(Coordinates.__mro__) loop melalui setiap superclass saat ini # kelas, dimulai dari “objek”.## Jadi lambda dipanggil melawan setiap superclass secara bergantian, dan setiap kali itu# membaca bidang __annotations__, jika tersedia. # # Coordinates.__annotations__ mengembalikan {‘x’: int, ‘y’: int}## dict1 | dict2 di Python mengembalikan dict baru yang menggabungkan dua sebelumnya ## Jadi fungsi ini mengembalikan kamus gabungan dari __annotations__ from# setiap kelas dalam hierarki superclass. kembali kurangi ( lambdax, y: x|
getattr
(y, “__annotations__”, {}),
terbalik
(cls.__mro__), {} ) # Berikutnya adalah fungsi yang menghasilkan metode yang berbeda. Ingat mereka lulus # bidang yang adalah {‘x’: , ‘y’:
} @codegendefmake__init__(bidang): # Memanggil “,”.join(dict) bergabung saja kunci kamus itukode=“def __init__(sendiri, “+“,”.Ikuti(
bidang)
+ “): n” # Jadi di sini kita memiliki: # def __init__ (sendiri, x, y): kembalikode+“n“ .Ikuti(f” diri sendiri.{nama}={nama}n“untuknamadibidang)
# Ini menambahkan:
# self.x=x # self.y=y@kodegendefmake__repr__(bidang): kembali ( “def __repr__(self): n“# type(self).__name__ memberi kita nama kelas: “Koordinat”‘ return f”{type(self).__name__}(‘# Ini memberi kita {self.x!r}, {self.y! r} yang dalam f-string# memberi kami __repr__() versi properti objek tersebut +“, “.Ikuti(“{diri sendiri.”+nama+ “!R}” untuknamadibidang) +‘)”n‘ ) # Jadi ini menghasilkan: # def __repr__(self): # return f”Koordinat({self.x!r}, {self.y!r})” @kodegendefmake__eq__(bidang): selfvals =“,”.Ikuti(f”diri sendiri.{nama}“untuknamadibidang ) lain-lain=“,,”.Ikuti(f”lainnya.{
nama}“
untuk namadibidang ) kembali ( “def __eq__(diri sendiri, orang lain): n“” jika self.__class__ adalah other.__class__: n“f” kembali ({selfvals},)==({lainnya S},)n“ ” kalau tidak: n“” return NotImplementedn“ ) # Ini menghasilkan: # def __eq__(diri sendiri, orang lain): # jika self.__class__ adalah yang lain.__class__: # return (self,x, self.y,)==(other.x, other.y)# kalau tidak: # return NotImplemented@kodegendefmake__iter__(bidang): kembali“def __iter__(sendiri): n” +“n“.Ikuti( f” menghasilkan diri sendiri.{ nama}“untuk namadibidang ) @kodegendefmake__hash__(bidang): self_tuple=“(“+“,,”.Ikuti(f”diri.{nama}” untuknamadi
cls): bidang=all_hints(cls) # bidang sekarang {‘x’: , ‘y’: }clsdict=vars(cls) # clsdict terlihat seperti ini: # {‘__module__’: ‘__main__’,
# '__annotations__': {'x': , 'y': },
# ‘__dict__’: ,# ‘__weakref__’: , # ‘__doc__’: Tidak ada} # # Tujuan dari fungsi ini terutama untuk menambahkan __init__ dan# __repr__ dan __eq__ metode, tetapi hanya jika belum didefinisikan jikabukan“__init__”diclsdict:
(bidang) # Tidak yakin mengapa ini dikomentari sekarang: # jika tidak ‘__iter__’ di clsdict: cls.__iter__=make__iter__(fields) # jika tidak ‘__hash__’ di clsdict: cls.__hash__=make__hash__(fields)cls.__match_args__=bidang# Ini terkait dengan pengetikan struktural Python: # https://www.python.org/dev/peps/pep-0622/#special-attribute-match-argskembali cl s# Contoh penggunaanjika__nama__ ==“__utama__”: @dataklass kelasKoordinat: x: inty: int
Dibuat 2021-12-19T20:30:57-08:00, diperbarui 2021-12-19T21:38:45-08:00 · Riwayat · Edit
Inilah Cara Membuat Seseorang Terbuka kepada Anda… Pernahkah Anda berbicara dengan pelanggan, kolega, karyawan, atau teman, dan berharap Anda bisa membuat mereka terbuka dan memberi tahu Anda apa yang sebenarnya mereka pikirkan? Meskipun Anda tidak dapat—dan tidak…
Kuis Sejarah Indonesia Kelas 11: Uji Pengetahuanmu… Hai semuanya! Apakah kamu sedang mencari kuis interaktif dan menyenangkan untuk menguji pengetahuan sejarah Indonesia kelas 11? Quizizz sejarah Indonesia kelas 11 adalah pilihan yang tepat untukmu! Apakah kamu merasa…
Kevin Holland mengejar, menghentikan pria yang… Pindah ke Derrick Lewis, ada sheriff baru di kota. Menurut serangkaian foto yang diberikan kepada Ariel Helwani, kelas menengah UFC Kevin Holland mengejar dan menghentikan seorang pria yang mencoba mencuri…
Francis Ngannou mengklaim psikiaternya… Gambar melalui Francis Ngannou di Instagram (fotografer tidak terdaftar) Francis Ngannou akan mempertahankan sabuk kelas beratnya untuk pertama kalinya akhir pekan mendatang. Pada hari Sabtu, 22 Januari di UFC 270…
Kunci Jawaban Sejarah Indonesia Kelas 10 Semester 2:… Bagi para pelajar Sekolah Menengah Atas (SMA) kelas X, kini tidak perlu lagi khawatir dalam menghadapi ulangan harian atau ujian semester untuk mata pelajaran sejarah Indonesia. Dalam artikel ini, tersedia…
Rob Font bereaksi atas kekalahan UFC Vegas 44 dari… Tidak. Penantang kelas bantam peringkat 4 Rob Font mendapatkan empat kemenangan beruntunnya pada hari Sabtu, kalah dari Jose Aldo di acara utama UFC Vegas 44. Aldo, mantan juara kelas bulu…
Latihan Ulangan Akhir Bab: Perjuangan Kemerdekaan… Jika kamu sedang mencari informasi tentang latih ulangan akhir bab sejarah indonesia kelas 11 halaman 67, maka kamu telah datang ke tempat yang tepat. Di sini, kami akan membahas secara…
Fakta Mencengangkan: Rahasia Penting Sejarah… Soal Sejarah Indonesia Kelas 10 Semester 2 dan Jawabannya Hai, para pelajar kelas 10! Pastinya sudah tidak sabar ingin mengetahui materi sejarah Indonesia semester 2, kan? Nah, dalam postingan ini,…
Tanya HN: Aku Sangat Kesepian Tanya HN: Aku Sangat Kesepian 23 poin oleh DevToRecruiter 23 menit yang lalu | sembunyikan | masa lalu | favorit | 21 komentar Saya tidak tahu apakah ini tempat yang…
Menguak Jejak Masa Lalu Indonesia: Emodul Sejarah… Belajar Sejarah Indonesia Kelas 12 dengan Emodul Interaktif dan Menyenangkan Apakah Anda merasa kesulitan mempelajari sejarah Indonesia kelas 12? Apakah Anda merasa materi yang diajarkan terlalu berat dan sulit dipahami?…
Soal Ujian Akhir Semester Sejarah Indonesia Kelas 11… Sebagai seorang siswa kelas 11, kamu pasti sudah tidak asing lagi dengan mata pelajaran sejarah Indonesia. Nah, salah satu bagian penting dalam pembelajaran sejarah Indonesia adalah ujian akhir semester (UAS).…
Soal Ujian Akhir Semester Sejarah Indonesia SMA… Apakah Anda seorang siswa kelas 11 yang sedang mencari soal-soal Ujian Akhir Semester (UAS) Sejarah Indonesia kurikulum 2013? Jangan khawatir, Anda berada di tempat yang tepat! Artikel ini akan membahas…
Francis Ngannou Menggoda Tyson Fury Fight Setelah… Katelyn Mulcahy/Getty ImagesFrancis Ngannou tampaknya siap untuk bertarung melawan Tyson Fury berdasarkan tanggapan hari Minggu kepada juara tinju kelas berat: Francis Ngannou @francis_ngannouhttps://t.co/BJRyYfzZQr pic.twitter.com/vLpuDl3SxLNgannou mengalahkan Cyril Gane dengan keputusan di…
Lebih Banyak Kreator Yang Menghasilkan Uang Secara… Dalam hal minat media dan dolar VC, apa yang disebut "ekonomi pencipta" sedang panas-panasnya. Investor menggelontorkan $1,3 miliar ke perusahaan yang berharap dapat membantu pembuat konten online memonetisasi bakat mereka…
Menjadi master Python pada tahun 2022 StackCommerce Python adalah salah satu bahasa pengkodean paling berguna di dunia. Bahkan jika Anda belum pernah melihat sekilas satu baris kode pun, Anda dapat mempelajari Python dengan Bundel Bootcamp Sertifikasi…
7 Cara Meningkatkan Kepercayaan Diri Anda Anda tidak dapat menang sebagai pengusaha atau pemilik bisnis tanpa kepercayaan penuh pada kemampuan Anda sendiri, serta pada solusi dan model bisnis Anda. Sebagai angel investor di startup, saya selalu…
Bellator 275 Hasil: Gegard Mousasi membuat pekerjaan… Bellator 275 Hasil: Gegard Mousasi membuat karya cepat Austin Vanderford Kandang Bellator MMA diadakan di Dublin, Irlandia pada hari Jumat untuk Bellator 275 di 3Arena, dan tidak butuh waktu lama…
Beneil Dariush percaya diri bergulat dengan Islam… 23 Januari 2022 17:10 ET ANAHEIM, California – Beneil Dariush merasa baik menjelang tes berikutnya. Pesaing kelas ringan UFC menghadapi anak didik Khabib Nurmagomedov Islam Makhachev pada 26 Februari di…
Paulo Costa merobek 'macho palsu' Sean… 18 Maret 2022 11:20 ET Paulo Costa memiliki beberapa pilihan kata untuk Sean Strickland setelah dia mengubah nadanya pada pertarungan potensial. Strickland (25-3 MMA, 12-3 UFC) memanggil Costa pada November…
Cain Velasquez mengakui "gesekan" dengan mantan… Mantan juara kelas berat UFC Cain Velasquez mengakui ada “gesekan” dengan mantan rival UFC Brock Lesnar selama menjalankan tugas WWE-nya. Velasquez dan Lesnar bertemu di UFC 121 pada Oktober 2010.…
Tai Tuivasa membagikan prediksinya untuk pertarungan… Gambar melalui FernandLopez.com UFC kelas berat Tai Tuivasa telah memberikan pemikirannya tentang pertempuran yang akan datang antara Francis Ngannou dan Ciryl Gane. Dalam acara utama UFC 270, Francis Ngannou akhirnya…
Charles Oliveira mengatakan kunci rendah hati untuk… LAS VEGAS – Selama 11 tahun terakhir, Charles Oliveira adalah orang yang mengejar emas. Sekarang dia memasuki minggu pertarungan sebagai orang yang memegang sabuk, veteran Brasil mengatakan sedikit yang berubah…
Kunci Jawaban PTS Genap Sejarah Indonesia Kelas 10:… Soal PTS Genap Sejarah Indonesia Kelas 10: Uji Pemahamanmu tentang Perjalanan Bangsa Indonesia! Sebagai siswa kelas 10, kamu tentu sudah tidak asing lagi dengan mata pelajaran Sejarah Indonesia. Nah, untuk…
Semua yang Perlu Anda Ketahui Tentang Menulis Buku,… Hai, saya Jay Steinfeld, dan saya adalah penulis pemula. Saya juga mengumumkan pensiun saya dari menulis buku.Saya akan menjelaskan sedikit tentang pensiun.Tapi mari kita mulai dengan hal-hal menarik , pertama.Saya…
Jelajahi Sejarah Indonesia Kelas 10: Kisah Bangsa… Perjalanan Sejarah Indonesia Melalui Cover Buku yang Menarik Sebagai siswa SMA kelas 10, mempelajari sejarah Indonesia menjadi salah satu mata pelajaran yang wajib diikuti. Namun, tidak sedikit siswa yang merasa…
Israel Adesanya, Robert Whittaker binatang yang sama… 10 Februari 2022 10:00 malam ET HOUSTON – Israel Adesanya dan Robert Whittaker tampaknya tidak terlalu menyukai satu sama lain, tetapi pada konferensi pers UFC 271 Kamis , rasa hormat…
Menguak Sejarah Indonesia: Tantangan dan Kemenangan Bangsa Soal Ujian Nasional Sejarah Indonesia Kelas 12: Persiapan Jitu Raih Nilai Bagus Apakah Anda seorang siswa kelas 12 yang sedang menghadapi ujian nasional sejarah Indonesia? Jika ya, maka Anda perlu…
Sejarah Indonesia Kelas 12 Semester 1: Kisah… Sebagai pelajar kelas 12, memahami sejarah Indonesia semester 1 tentunya menjadi tantangan tersendiri. Banyaknya materi dan menghafalkannya tentu membuat kamu kewalahan. Belum lagi soal ujian yang semakin dekat, membuat kamu…
David Onama bangga menjadi petarung Uganda pertama… Kredit Gambar: UFCDavid Onama membuat sejarah pada Sabtu malam saat ia menjadi petarung pertama dari Uganda yang menang di UFC. Onama menjadi petarung pertama dari Uganda yang mencapai UFC Oktober…
Karyawan Muda Saya Melemahkan Dirinya Dengan… Kolumnis Inc.com, Alison Green, menjawab pertanyaan tentang masalah tempat kerja dan manajemen--mulai dari cara menghadapi bos yang mengelola mikro cara berbicara dengan seseorang di tim Anda tentang bau badan.Pembaca bertanya:…