Hướng dẫn how do you switch bytes in python? - làm thế nào để bạn chuyển đổi byte trong python?

Lấy ví dụ này:

i = 0x12345678
print("{:08x}".format(i))
   # shows 12345678
i = swap32(i)
print("{:08x}".format(i))
   # should print 78563412

Điều gì sẽ là

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
8? Có cách nào để trao đổi byte một
def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
9 trong Python, lý tưởng với các công cụ tích hợp không?

Hỏi ngày 16 tháng 12 năm 2014 lúc 14:05Dec 16, 2014 at 14:05

Hướng dẫn how do you switch bytes in python? - làm thế nào để bạn chuyển đổi byte trong python?

Patrick B.Patrick B.Patrick B.

11.3k8 Huy hiệu vàng55 Huy hiệu bạc97 Huy hiệu Đồng8 gold badges55 silver badges97 bronze badges

5

Một phương pháp là sử dụng mô -đun

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
0:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]

Trước tiên, bạn đóng gói số nguyên của mình vào định dạng nhị phân bằng cách sử dụng một endianness, sau đó bạn giải nén nó bằng cách sử dụng cái kia (thậm chí không quan trọng bạn sử dụng kết hợp nào, vì tất cả những gì bạn muốn làm là hoán đổi độ chính xác).

Đã trả lời ngày 16 tháng 12 năm 2014 lúc 14:16Dec 16, 2014 at 14:16

CarstencarstenCarsten

17,5K4 Huy hiệu vàng46 Huy hiệu bạc53 Huy hiệu Đồng4 gold badges46 silver badges53 bronze badges

5

Big Endian có nghĩa là cách bố trí của 32 bit int có byte quan trọng nhất đầu tiên,

ví dụ. 0x12345678 có bố cục bộ nhớ

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+

Trong khi trên Little Endian, bố cục bộ nhớ là

lsb             msb
+------------------+
| 78 | 56 | 34 | 12|
+------------------+

Vì vậy, bạn chỉ có thể chuyển đổi giữa chúng với một số mặt nạ và thay đổi một chút:

def swap32(x):
    return (((x << 24) & 0xFF000000) |
            ((x <<  8) & 0x00FF0000) |
            ((x >>  8) & 0x0000FF00) |
            ((x >> 24) & 0x000000FF))

Đã trả lời ngày 16 tháng 12 năm 2014 lúc 14:17Dec 16, 2014 at 14:17

Hướng dẫn how do you switch bytes in python? - làm thế nào để bạn chuyển đổi byte trong python?

Nosnosnos

Phim thương hiệu vàng 218K5454 gold badges404 silver badges493 bronze badges

2

Từ Python 3.2, bạn có thể xác định Swap32 () như sau:

def swap32(x):
    return int.from_bytes(x.to_bytes(4, byteorder='little'), byteorder='big', signed=False)

Nó sử dụng mảng byte để thể hiện giá trị và đảo ngược thứ tự của byte bằng cách thay đổi độ nội đầu trong quá trình chuyển đổi trở lại số nguyên.

Đã trả lời ngày 10 tháng 3 năm 2016 lúc 2:20Mar 10, 2016 at 2:20

Hướng dẫn how do you switch bytes in python? - làm thế nào để bạn chuyển đổi byte trong python?

1

Có thể đơn giản sử dụng thư viện ổ cắm.

from socket import htonl

swapped = htonl (i)
print (hex(swapped))

đó là nó. Thư viện này cũng hoạt động theo hướng khác với NTOHL

Đã trả lời ngày 7 tháng 2 năm 2021 lúc 15:01Feb 7, 2021 at 15:01

Hướng dẫn how do you switch bytes in python? - làm thế nào để bạn chuyển đổi byte trong python?

RobdrobdRobD

751 Huy hiệu bạc7 Huy hiệu đồng1 silver badge7 bronze badges

1

Giới thiệu về đặt hàng byte và ndarrays#

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
1 là một đối tượng cung cấp giao diện mảng Python cho dữ liệu trong bộ nhớ.

Nó thường xảy ra rằng bộ nhớ mà bạn muốn xem với một mảng không giống với thứ tự byte giống như máy tính mà bạn đang chạy Python.

Ví dụ, tôi có thể đang làm việc trên một máy tính với CPU nhỏ-chẳng hạn như Intel Pentium, nhưng tôi đã tải một số dữ liệu từ một tệp được viết bởi một máy tính là lớn. Hãy nói rằng tôi đã tải 4 byte từ một tập tin được viết bởi một máy tính mặt trời (lớn). Tôi biết rằng 4 byte này đại diện cho hai số nguyên 16 bit. Trên một máy lớn, một số nguyên hai byte được lưu trữ với byte quan trọng nhất (MSB) trước tiên, và sau đó là byte ít có ý nghĩa nhất (LSB). Do đó, các byte, theo thứ tự bộ nhớ:

  1. Số nguyên MSB 1

  2. LSB Số nguyên 1

  3. Số nguyên MSB 2

  4. LSB Số nguyên 2

Giả sử hai số nguyên trên thực tế 1 và 770. Vì 770 = 256 * 3 + 2, 4 byte trong bộ nhớ sẽ chứa tương ứng: 0, 1, 3, 2. Các byte tôi đã tải từ tệp sẽ có các nội dung này :

>>> big_end_buffer = bytearray([0,1,3,2])
>>> big_end_buffer
bytearray(b'\x00\x01\x03\x02')

Chúng tôi có thể muốn sử dụng

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
1 để truy cập các số nguyên này. Trong trường hợp đó, chúng ta có thể tạo một mảng xung quanh bộ nhớ này và nói với Numpy rằng có hai số nguyên, và chúng là 16 bit và lớn nhất:

>>> import numpy as np
>>> big_end_arr = np.ndarray(shape=(2,),dtype='>i2', buffer=big_end_buffer)
>>> big_end_arr[0]
1
>>> big_end_arr[1]
770

Lưu ý mảng

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
3 ở trên của
msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
4.
msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
5 có nghĩa là ‘lớn endian (
msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
6 là ít endian) và
msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
7 có nghĩa là‘ đã ký 2 byte số nguyên. Ví dụ: nếu dữ liệu của chúng tôi đại diện cho một số nguyên nhỏ 4 byte không dấu duy nhất, chuỗi DTYPE sẽ là
msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
8.

Trong thực tế, tại sao don don chúng ta thử điều đó?

>>> little_end_u4 = np.ndarray(shape=(1,),dtype='<u4', buffer=big_end_buffer)
>>> little_end_u4[0] == 1 * 256**1 + 3 * 256**2 + 2 * 256**3
True

Quay trở lại

msb             lsb
+------------------+
| 12 | 34 | 56 | 78|
+------------------+
9 của chúng tôi-trong trường hợp này, dữ liệu cơ bản của chúng tôi là Big-endian (Data Endianness) và chúng tôi đã đặt DTYPE phù hợp (DTYPE cũng là Big-Endian). Tuy nhiên, đôi khi bạn cần phải lật những thứ này xung quanh.

Cảnh báo

Scalars hiện không bao gồm thông tin thứ tự byte, do đó, trích xuất một vô hướng từ một mảng sẽ trả về một số nguyên theo thứ tự byte gốc. Kể từ đây:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
0

Thay đổi đơn đặt hàng byte#

Như bạn có thể tưởng tượng từ phần giới thiệu, có hai cách bạn có thể ảnh hưởng đến mối quan hệ giữa thứ tự byte của mảng và bộ nhớ cơ bản mà nó đang nhìn vào:

  • Thay đổi thông tin đặt hàng byte trong DTYPE mảng để nó diễn giải dữ liệu cơ bản là theo thứ tự byte khác. Đây là vai trò của

    lsb             msb
    +------------------+
    | 78 | 56 | 34 | 12|
    +------------------+
    
    0

  • Thay đổi thứ tự byte của dữ liệu cơ bản, để lại cách giải thích DTYPE. Đây là những gì

    lsb             msb
    +------------------+
    | 78 | 56 | 34 | 12|
    +------------------+
    
    1 làm.

Các tình huống phổ biến mà bạn cần thay đổi thứ tự byte là:

  1. Dữ liệu và endianness dty của bạn phù hợp với nhau và bạn muốn thay đổi DTYPE để nó khớp với dữ liệu.

  2. Dữ liệu của bạn và DTYPE Endianness không phù hợp và bạn muốn trao đổi dữ liệu để chúng khớp với DTYPE

  3. Dữ liệu của bạn và sự phù hợp với endianness của bạn, nhưng bạn muốn dữ liệu được hoán đổi và DTYPE để phản ánh điều này

Dữ liệu và endianness dtype don don phù hợp, thay đổi dtype để khớp dữ liệu#

Chúng tôi làm một cái gì đó mà họ không phù hợp:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
1

Bản sửa lỗi rõ ràng cho tình huống này là thay đổi DTYPE để nó đưa ra sự chính xác:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
2

Lưu ý rằng mảng không thay đổi trong bộ nhớ:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
3

Dữ liệu và loại endianness don lồng khớp, thay đổi dữ liệu để khớp với DTYPE#

Bạn có thể muốn làm điều này nếu bạn cần dữ liệu trong bộ nhớ là một thứ tự nhất định. Ví dụ: bạn có thể viết bộ nhớ ra một tệp cần đặt hàng byte nhất định.

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
4

Bây giờ mảng đã thay đổi trong bộ nhớ:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
5

Dữ liệu và kết hợp endianness của dtype, dữ liệu hoán đổi và dtype#

Bạn có thể có một mảng DTYPE được chỉ định chính xác, nhưng bạn cần mảng có thứ tự byte đối diện trong bộ nhớ và bạn muốn DTYPE khớp với các giá trị mảng có ý nghĩa. Trong trường hợp này, bạn chỉ cần thực hiện cả hai hoạt động trước đó:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
6

Một cách dễ dàng hơn để chuyển dữ liệu vào một thứ tự DTYPE và byte cụ thể có thể đạt được với phương pháp NDarray Astype:

def swap32(i):
    return struct.unpack("<I", struct.pack(">I", i))[0]
7

Hoán đổi byte là gì?

Để xem xét, hoán đổi byte đảo ngược thứ tự của byte trong một số nguyên (cho dù độ dài 2, 4, hoặc 8 byte). Điều này là cần thiết khi các bộ xử lý X86 lưu trữ byte thứ tự thấp của số nguyên trước ("Little Endian") và bộ xử lý SPARC lưu trữ byte bậc cao trước tiên ("Big Endian").reverses the order of bytes in a integer (whether 2-, 4-, or 8-byte lengths). This is necessary as x86 processors store the low order byte of an integer first ("little endian"), and SPARC processors store the high-order byte first ("big endian").

Làm thế nào để bạn gán một giá trị byte trong Python?

Đối với một byte duy nhất, về cơ bản bạn có ba lựa chọn: một chiều dài 1 byte (hoặc bytearray) đối tượng mychar = b '\ xff' (hoặc mychar = bytearray (b '\ xff')) Phạm vi (256) (hoặc sử dụng mặt nạ để cắt tràn): mychar = 0xff. Một loại CTYPES, ví dụ: mychar = ctypes.A length 1 bytes (or bytearray ) object mychar = b'\xff' (or mychar = bytearray(b'\xff') ) An int that you don't assign values outside range(256) (or use masking to trim overflow): mychar = 0xff. A ctypes type, e.g. mychar = ctypes.

Thứ tự byte trong Python là gì?

Đối số byteorder xác định thứ tự byte được sử dụng để thể hiện số nguyên.Nếu byteorder là "lớn", byte quan trọng nhất là ở đầu mảng byte.Nếu byteorder là "ít", byte quan trọng nhất là ở cuối mảng byte.. If byteorder is "big" , the most significant byte is at the beginning of the byte array. If byteorder is "little" , the most significant byte is at the end of the byte array.

Byte hoạt động như thế nào trong Python?

Loại byte trong Python là bất biến và lưu trữ một chuỗi các giá trị dao động từ 0-255 (8 bit).Bạn có thể nhận được giá trị của một byte duy nhất bằng cách sử dụng một chỉ mục như một mảng, nhưng các giá trị không thể được sửa đổi.You can get the value of a single byte by using an index like an array, but the values can not be modified.