Rumah pembangunan bahagian belakang Tutorial Python Menulis kod Python seperti Typescript

Menulis kod Python seperti Typescript

Aug 01, 2024 pm 08:18 PM

Saya andaikan anda yang ingin membaca artikel ini tahu apa itu skrip taip. Pembangun Javascript mencipta skrip taip untuk menjadikan javascript lebih selamat jenis. Typesafe menjadikan kod lebih mudah dibaca dan mempunyai kurang pepijat tanpa menulis sebarang ujian. Bolehkah typesafety dicapai dalam python?.

Mengapa kita memerlukan jenis keselamatan?

Bayangkan fungsi yang kelihatan tidak bersalah ini

def send_email(sender, receiver, message):
    ...
Salin selepas log masuk

Saya sengaja menyembunyikan pelaksanaan kodnya. Bolehkah anda meneka fungsi untuk apa dan parameter apa yang kita perlukan untuk menggunakan fungsi ini hanya dengan nama dan parameter fungsinya?. Kami tahu dari nama fungsinya ia berfungsi untuk menghantar e-mel. Bagaimana pula dengan parameternya, apakah yang perlu kita letakkan untuk menggunakan fungsi ini?.

Pengirim tekaan pertama ialah str e-mel, penerima ialah str e-mel, mesej ialah str kandungan e-mel.

send_email(sender="john@mail.com", receiver="doe@mail.com", message="Hello doe! How are you?")
Salin selepas log masuk

Tekaan paling mudah. tetapi ia bukan satu-satunya tekaan.

Penghantar tekaan kedua ialah int user_id pada db, penerima ialah int user_id pada db, mesej ialah str kandungan e-mel.

john_user_id = 1
doe_user_id = 2
send_email(sender=1, receiver=2, message="Hello doe! How are you?")
Salin selepas log masuk

Bayangkan bekerja pada aplikasi. Kebanyakan aplikasi menggunakan beberapa pangkalan data. Pengguna biasanya diwakili oleh idnya.

Penghantar tekaan ketiga ialah kamus, penerima ialah kamus, mesej ialah kamus.

john = {
    "id": 1,
    "username": "john",
    "email": "john@mail.com"
}
doe = {
    "id": 2,
    "username": "doe",
    "email": "doe@mail.com"
}
message = {
    "title": "Greeting my friend doe",
    "body": "Hello doe! How are you?"
}
send_email(sender=john, receiver=doe, message=message)
Salin selepas log masuk

Mungkin send_email memerlukan lebih daripada e-mel dan id pengguna. Untuk menambah lebih banyak data pada setiap parameter, ia menggunakan beberapa struktur kamus. Anda perasan bahawa mesej itu bukan sekadar str mungkin ia memerlukan tajuk dan kandungan.

Penghantar tekaan keempat ialah Pengguna kelas, penerima ialah Pengguna kelas, mesej ialah kamus.

class User():

    def __init__(self, id, username, email):
        self.id = id
        self.username = username
        self.email = email

john = User(id=1, username="john", email="john@mail.com")
doe = User(id=2, username="doe", email="doe@mail.com")
message = {
    "title": "Greeting my friend doe",
    "body": "Hello doe! How are you?"
}
send_email(sender=john, receiver=doe, message=message)
Salin selepas log masuk

Mungkin send_email berintegrasi dengan beberapa orm pangkalan data seperti Django ORM atau Sqlalchemy. Untuk memudahkan pengguna akhir, ia menggunakan kelas ORM secara langsung.

Jadi yang mana satu jawapan yang betul? salah satunya boleh menjadi jawapan yang betul. Mungkin jawapan yang betul boleh menjadi gabungan dua tekaan. Seperti penghantar dan penerima ialah Pengguna kelas (Tekaan Keempat) tetapi mesej itu str (Tekaan pertama). Kami tidak pasti melainkan kami membaca pelaksanaan kodnya. Yang membuang masa jika anda pengguna akhir. Sebagai pengguna akhir yang menggunakan fungsi ini, kami hanya perlukan fungsi apa, parameter apa yang diperlukan dan output fungsi apa.

Penyelesaian

Docstring

Python telah membina dokumentasi fungsi menggunakan docstring. Di sini contoh docstring.

def add(x, y):
    """Add two number

    Parameter:\n
    x -- int\n
    y -- int\n

    Return: int
    """
    return x + y

def send_email(sender, receiver, message):
    """Send email from sender to receiver

    Parameter:\n
    sender -- email sender, class User\n
    receiver -- email receiver, class User\n
    message -- body of the email, dictionary (ex: {"title": "some title", "body": "email body"}\n

    Return: None
    """
    ...
Salin selepas log masuk

Apa yang menarik tentang docstring ia serasi dengan editor. Dalam vscode ia akan menunjukkan docstring apabila anda menuding pada fungsi. Kebanyakan perpustakaan dalam python menggunakan docstring untuk mendokumentasikan fungsinya.

Writing Python code like Typescript

Masalah dengan docstring ialah penyegerakan dokumen. Cara anda memastikan bahawa docstring sentiasa disegerakkan dengan pelaksanaan kod. Anda tidak boleh mengujinya dengan betul. Saya mendengarnya daripada orang rawak di internet "Memiliki dokumentasi yang lapuk adalah lebih teruk daripada tidak mempunyai dokumentasi".

Doctest

Btw Anda boleh menguji docstring menggunakan doctest kinda. Doctest menguji docstring anda dengan menjalankan contoh pada docstring anda. Doctest sudah diprapasang dalam python jadi anda tidak memerlukan pergantungan luaran. mari lihat contoh ini buat fail baharu bernama my_math.py kemudian letakkan kod ini.

# my_math.py
def add(x, y):
    """Add two integer

    Parameter:\n
    x -- int\n
    y -- int\n

    Return: int
    >>> add(1, 2)
    3
    """
    return x + y


if __name__ == "__main__":
    import doctest

    doctest.testmod()

Salin selepas log masuk

Ia adalah kod yang sama seperti contoh docstring tetapi saya menambah contoh dan doctest pada baris terakhir kod. Untuk menguji docstring hanya jalankan fail python my_math.py. Jika tiada output bermakna contoh anda lulus ujian. Jika anda mahu melihat output menjalankannya dalam mod verbose python my_math.py -v, anda akan melihat output ini.

Trying:
    add(1, 2)
Expecting:
    3
ok
1 items had no tests:
    __main__
1 items passed all tests:
   1 tests in __main__.add
1 tests in 2 items.
1 passed and 0 failed.
Test passed
Salin selepas log masuk

Jika anda membuat kesilapan pada contoh kod ia akan mengembalikan ralat.

# my_math.py
def add(x, y):
    """Add two integer

    Parameter:\n
    x -- int\n
    y -- int\n

    Return: int
    >>> add(2, 2) # <-- I change it here
    3
    """
    return x + y


if __name__ == "__main__":
    import doctest

    doctest.testmod()

Salin selepas log masuk

keluaran:

**********************************************************************
File "~/typescript-in-python/my_math.py", line 12, in __main__.add
Failed example:
    add(2, 2) # <-- I change it here
Expected:
    3
Got:
    4
**********************************************************************
1 items had failures:
   1 of   1 in __main__.add
***Test Failed*** 1 failures.
Salin selepas log masuk

Hebat! kini saya boleh menguji docstring saya. Tetapi kaveatnya ialah:

  1. doctest hanya semak contoh ia tidak menyemak parameter fungsi komen dan kembali
  2. doctest perlu menjalankan kod seperti alat ujian lain untuk menyemak ia betul atau tidak doctest perlu menjalankan contoh kod. Jika kod anda memerlukan beberapa alatan luaran seperti pangkalan data atau pelayan smtp (seperti menghantar e-mel), sukar untuk menguji menggunakan doctest.

Menaip Python

Kadangkala anda tidak perlu menjalankan kod untuk menyemak sama ada kod betul atau tidak. Anda hanya memerlukan jenis input dan jenis output. Bagaimana? Pertimbangkan contoh ini.

def add(x, y):
    """Add two integer

    Parameter:\n
    x -- int\n
    y -- int\n

    Return: int
    """
    return x + y

def sub(x, y):
    """Substract two integer

    Parameter:\n
    x -- int\n
    y -- int\n

    Return: int
    """
    return x - y

a = add(2, 1)
b = add(1, 1)
c = sub(a, b)
Salin selepas log masuk

fungsi tambah kembalikan int dan sub fungsi memerlukan dua int sebagai parameter input. Jika saya menggunakan dua pulangan dari fungsi tambah kemudian letakkannya pada sub paramater seperti contoh di atas adakah ia ralat? sudah tentu bukan kerana sub fungsi memerlukan int dan anda meletakkan int juga.

Memandangkan python 3.5 python telah dibina dalam jenis yang dipanggil menaip. Dengan menaip anda boleh menambah jenis pada fungsi anda seperti contoh di bawah.

def add(x: int, y: int) -> int:
    """Add two integer"""
    return x + y

a = add(1, 2)
Salin selepas log masuk

Instead put it on your docstring you put it on the function. Typing is supported on many editor. If you use vscode you can hover on variable and it will shown it's type.
Writing Python code like Typescript

Nice now our code will have a type safety. eeehhhh not realy. If I intentionally use function incorrectlly like this.

def add(x: int, y: int) -> int:
    """Add two integer"""
    return x + y

res = add(1, [])
print(res)
Salin selepas log masuk

It will show error

Traceback (most recent call last):
  File "~/typescript-in-python/main.py", line 5, in <module>
    res = add(1, [])
          ^^^^^^^^^^
  File "~/typescript-in-python/main.py", line 3, in add
    return x + y
           ~~^~~
TypeError: unsupported operand type(s) for +: 'int' and 'list'
Salin selepas log masuk

But it doesn't show that you put incorrect type. Even worse if you use it like this.

def add(x: int, y: int) -> int:
    """Add two integer"""
    return x + y

res = add("hello", "world")
print(res)
Salin selepas log masuk
Salin selepas log masuk

It will succeed. It must be error because you put incorrect type.

helloworld
Salin selepas log masuk

Why python typing doesn't have type checker by default??. Based on pep-3107 it said

Before launching into a discussion of the precise ins and outs of Python 3.0’s function annotations, let’s first talk broadly about what annotations are and are not:

  1. Function annotations, both for parameters and return values, are completely optional.
  2. Function annotations are nothing more than a way of associating arbitrary Python expressions with various parts of a function at compile-time. By itself, Python does not attach any particular meaning or significance to annotations. Left to its own, Python simply makes these expressions available as described in Accessing Function Annotations below.

The only way that annotations take on meaning is when they are interpreted by third-party libraries. ...

So in python typing is like a decorator in typescript or java it doesn't mean anything. You need third party libraries todo type checking. Let's see some library for typechecking.

Python typing + type checker

Here are libraries for typechecking in python. For example we will typecheck this wrong.py file

def add(x: int, y: int) -> int:
    """Add two integer"""
    return x + y

res = add("hello", "world")
print(res)
Salin selepas log masuk
Salin selepas log masuk

1.mypy

The "OG" of python type checker. To install it just using pip pip install mypy. Now let's use mypy to typecheck this file. Run mypy wrong.py. It will shown type error which is nice.

wrong.py:5: error: Argument 1 to "add" has incompatible type "str"; expected "int"  [arg-type]
wrong.py:5: error: Argument 2 to "add" has incompatible type "str"; expected "int"  [arg-type]
Found 2 errors in 1 file (checked 1 source file)
Salin selepas log masuk

btw you can run mypy on entire project by using mypy ..

2.pyright

Another typechecker is pyright. It created by microsoft. It's same like mypy install through pip pip install pyright. Then run it pyright wrong.py. It will shown this error.

~/typescript-in-python/wrong.py
  ~/typescript-in-python/wrong.py:5:11 - error: Argument of type "Literal['hello']" cannot be assigned to parameter "x" of type "int" in function "add"
    "Literal['hello']" is incompatible with "int" (reportArgumentType)
  ~/typescript-in-python/wrong.py:5:20 - error: Argument of type "Literal['world']" cannot be assigned to parameter "y" of type "int" in function "add"
    "Literal['world']" is incompatible with "int" (reportArgumentType)
2 errors, 0 warnings, 0 informations
Salin selepas log masuk

It said that it's more faster than mypy but I found that's not much diffrent. Maybe my code base it's to small. Also pyright implement more python standard than mypy you can see on https://microsoft.github.io/pyright/#/mypy-comparison. Personaly I prefer mypy than pyright because the error message were more readable.

3.pylyzer

Speaking of performance and speed another new python typechecker pylyzer. It's written in rust. You can install it through pip pip install pylyzer or through cargo (rust package manager) cargo install pylyzer --locked. Then run it pylyzer wrong.py. It will shown this error.

Start checking: wrong.py
Found 2 errors: wrong.py
Error[#2258]: File wrong.py, line 5, <module>.res

5 | res = add("hello", "world")
  :           -------
  :                 |- expected: Int
  :                 `- but found: {"hello"}

TypeError: the type of add::x (the 1st argument) is mismatched

Error[#2258]: File wrong.py, line 5, <module>.res

5 | res = add("hello", "world")
  :                    -------
  :                          |- expected: Int
  :                          `- but found: {"world"}

TypeError: the type of add::y (the 2nd argument) is mismatched
Salin selepas log masuk

So far this is the most readable and beautiful error message. It's reminds me of rust compiler error. Speed, performance and most readable error message, I think I will choose to using pylyzer if the package already stable. The problem is at the time I write this blog, pylyzer still in beta. It can only typecheck your code base, it haven't support external depedencies.

Conclusion

Alright we successfully write python code like typescript (kinda). There is more way to using python typing module other than check simple type (str, int, bool etc). Maybe I will cover more advance type it in next blog. Maybe you guys have opinion about this, know better typechecker other then those 3, found other way to do typecheck in python or other. let me know on comment section below. As always Happy Coding.

Atas ialah kandungan terperinci Menulis kod Python seperti Typescript. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn

Alat AI Hot

Undresser.AI Undress

Undresser.AI Undress

Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover

AI Clothes Remover

Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool

Undress AI Tool

Gambar buka pakaian secara percuma

Clothoff.io

Clothoff.io

Penyingkiran pakaian AI

AI Hentai Generator

AI Hentai Generator

Menjana ai hentai secara percuma.

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
1 bulan yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Tetapan grafik terbaik
1 bulan yang lalu By 尊渡假赌尊渡假赌尊渡假赌
Akan R.E.P.O. Ada Crossplay?
1 bulan yang lalu By 尊渡假赌尊渡假赌尊渡假赌

Alat panas

Notepad++7.3.1

Notepad++7.3.1

Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina

SublimeText3 versi Cina

Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1

Hantar Studio 13.0.1

Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6

Dreamweaver CS6

Alat pembangunan web visual

SublimeText3 versi Mac

SublimeText3 versi Mac

Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Bagaimana untuk menyelesaikan masalah kebenaran yang dihadapi semasa melihat versi Python di Terminal Linux? Bagaimana untuk menyelesaikan masalah kebenaran yang dihadapi semasa melihat versi Python di Terminal Linux? Apr 01, 2025 pm 05:09 PM

Penyelesaian kepada Isu Kebenaran Semasa Melihat Versi Python di Terminal Linux Apabila anda cuba melihat versi Python di Terminal Linux, masukkan Python ...

Bagaimana cara menyalin seluruh lajur satu data ke dalam data data lain dengan struktur yang berbeza di Python? Bagaimana cara menyalin seluruh lajur satu data ke dalam data data lain dengan struktur yang berbeza di Python? Apr 01, 2025 pm 11:15 PM

Apabila menggunakan Perpustakaan Pandas Python, bagaimana untuk menyalin seluruh lajur antara dua data data dengan struktur yang berbeza adalah masalah biasa. Katakan kita mempunyai dua DAT ...

Bagaimana Mengajar Asas Pengaturcaraan Pemula Komputer Dalam Kaedah Projek dan Masalah Dikemukakan Dalam masa 10 Jam? Bagaimana Mengajar Asas Pengaturcaraan Pemula Komputer Dalam Kaedah Projek dan Masalah Dikemukakan Dalam masa 10 Jam? Apr 02, 2025 am 07:18 AM

Bagaimana Mengajar Asas Pengaturcaraan Pemula Komputer Dalam masa 10 jam? Sekiranya anda hanya mempunyai 10 jam untuk mengajar pemula komputer beberapa pengetahuan pengaturcaraan, apa yang akan anda pilih untuk mengajar ...

Bagaimana untuk mengelakkan dikesan oleh penyemak imbas apabila menggunakan fiddler di mana-mana untuk membaca lelaki-dalam-tengah? Bagaimana untuk mengelakkan dikesan oleh penyemak imbas apabila menggunakan fiddler di mana-mana untuk membaca lelaki-dalam-tengah? Apr 02, 2025 am 07:15 AM

Cara mengelakkan dikesan semasa menggunakan fiddlerevery di mana untuk bacaan lelaki-dalam-pertengahan apabila anda menggunakan fiddlerevery di mana ...

Apakah ungkapan biasa? Apakah ungkapan biasa? Mar 20, 2025 pm 06:25 PM

Ekspresi biasa adalah alat yang berkuasa untuk memadankan corak dan manipulasi teks dalam pengaturcaraan, meningkatkan kecekapan dalam pemprosesan teks merentasi pelbagai aplikasi.

Bagaimanakah uvicorn terus mendengar permintaan http tanpa serving_forever ()? Bagaimanakah uvicorn terus mendengar permintaan http tanpa serving_forever ()? Apr 01, 2025 pm 10:51 PM

Bagaimanakah Uvicorn terus mendengar permintaan HTTP? Uvicorn adalah pelayan web ringan berdasarkan ASGI. Salah satu fungsi terasnya ialah mendengar permintaan HTTP dan teruskan ...

Apakah beberapa perpustakaan Python yang popular dan kegunaan mereka? Apakah beberapa perpustakaan Python yang popular dan kegunaan mereka? Mar 21, 2025 pm 06:46 PM

Artikel ini membincangkan perpustakaan Python yang popular seperti Numpy, Pandas, Matplotlib, Scikit-Learn, Tensorflow, Django, Flask, dan Permintaan, memperincikan kegunaan mereka dalam pengkomputeran saintifik, analisis data, visualisasi, pembelajaran mesin, pembangunan web, dan h

Bagaimana secara dinamik membuat objek melalui rentetan dan panggil kaedahnya dalam Python? Bagaimana secara dinamik membuat objek melalui rentetan dan panggil kaedahnya dalam Python? Apr 01, 2025 pm 11:18 PM

Di Python, bagaimana untuk membuat objek secara dinamik melalui rentetan dan panggil kaedahnya? Ini adalah keperluan pengaturcaraan yang biasa, terutamanya jika perlu dikonfigurasikan atau dijalankan ...

See all articles