객체 지향 파이썬-객체 직렬화

데이터 저장의 맥락에서 직렬화는 데이터 구조 또는 개체 상태를 저장 (예 : 파일 또는 메모리 버퍼에)하거나 나중에 전송 및 재구성 할 수있는 형식으로 변환하는 프로세스입니다.

직렬화에서 객체는 저장 가능한 형식으로 변환되어 나중에 직렬화 해제하고 직렬화 된 형식에서 원래 객체를 다시 만들 수 있습니다.

간물

피클 링은 파이썬 객체 계층이 바이트 스트림 (일반적으로 사람이 읽을 수 없음)으로 변환되어 파일에 기록되는 프로세스이며,이를 직렬화라고도합니다. 언 피클 링은 반대 작업으로, 바이트 스트림이 작동하는 Python 객체 계층으로 다시 변환됩니다.

Pickle은 객체를 저장하는 가장 간단한 방법입니다. Python Pickle 모듈은 객체를 특수 저장소 형식으로 직접 저장하는 객체 지향 방식입니다.

뭘 할 수 있는데?

  • Pickle은 사전과 목록을 매우 쉽게 저장하고 재현 할 수 있습니다.
  • 오브젝트 속성을 저장하고 동일한 상태로 다시 복원합니다.

피클이 할 수없는 것은 무엇입니까?

  • 개체 코드를 저장하지 않습니다. 속성 값입니다.
  • 파일 핸들이나 연결 소켓을 저장할 수 없습니다.

간단히 말해서, 피클 링은 변수가 목록, 클래스 등이 될 수있는 파일에서 데이터 변수를 저장하고 검색하는 방법입니다.

피클하려면 반드시-

  • 수입 피클
  • 파일에 변수를 작성하십시오.
pickle.dump(mystring, outfile, protocol),

세 번째 인수 프로토콜은 선택 사항입니다.

피클 가져 오기

파일에 변수를 작성하십시오.

myString = pickle.load(inputfile)

행동 양식

pickle 인터페이스는 네 가지 방법을 제공합니다.

  • dump() − dump () 메서드는 열린 파일 (파일 류 객체)로 직렬화합니다.

  • dumps() − 문자열로 직렬화

  • load() − 개방형 객체에서 역 직렬화합니다.

  • loads() − 문자열에서 역 직렬화합니다.

위의 절차에 따라 아래는“산 세척”의 예입니다.

산출

My Cat pussy is White and has 4 legs
Would you like to see her pickled? Here she is!
b'\x80\x03c__main__\nCat\nq\x00)\x81q\x01}q\x02(X\x0e\x00\x00\x00number_of_legsq\x03K\x04X\x05\x00\x00\x00colorq\x04X\x05\x00\x00\x00Whiteq\x05ub.'

따라서 위의 예에서 Cat 클래스의 인스턴스를 만든 다음이를 피클 링하여 "Cat"인스턴스를 간단한 바이트 배열로 변환했습니다.

이렇게하면 바이너리 파일이나 데이터베이스 필드에 바이트 배열을 쉽게 저장하고 나중에 스토리지 지원에서 원래 형태로 복원 할 수 있습니다.

또한 피클 된 객체로 파일을 생성하려는 경우 열린 바이너리 파일도 전달하는 dump () 메서드 (dumps * () * 하나 대신)를 사용할 수 있으며 피클 링 결과는 파일에 자동으로 저장됩니다.

[….]
binary_file = open(my_pickled_Pussy.bin', mode='wb')
my_pickled_Pussy = pickle.dump(Pussy, binary_file)
binary_file.close()

산세 제거

이진 배열을 가져 와서 객체 계층 구조로 변환하는 프로세스를 언 피클 링이라고합니다.

언 피클 링 프로세스는 pickle 모듈의 load () 함수를 사용하여 수행되며 간단한 바이트 배열에서 완전한 객체 계층 구조를 반환합니다.

이전 예제에서로드 함수를 사용해 보겠습니다.

산출

MeOw is black
Pussy is white

JSON

JSON (JavaScript Object Notation)은 Python 표준 라이브러리의 일부로 경량 데이터 교환 형식입니다. 사람이 읽고 쓰는 것은 쉽습니다. 구문 분석 및 생성이 쉽습니다.

단순성 때문에 JSON은 데이터를 저장하고 교환하는 방법으로, JSON 구문을 통해 수행되며 많은 웹 애플리케이션에서 사용됩니다. 사람이 읽을 수있는 형식이므로 API로 작업 할 때의 효율성 외에도 데이터 전송에 사용하는 이유 중 하나 일 수 있습니다.

JSON 형식 데이터의 예는 다음과 같습니다.

{"EmployID": 40203, "Name": "Zack", "Age":54, "isEmployed": True}

Python을 사용하면 Json 파일 작업이 간단 해집니다. 이 목적으로 사용되는 모듈은 JSON 모듈입니다. 이 모듈은 Python 설치에 포함 (내장)되어야합니다.

이제 Python 사전을 JSON으로 변환하고 텍스트 파일에 쓰는 방법을 살펴 보겠습니다.

JSON에서 Python으로

JSON을 읽는다는 것은 JSON을 Python 값 (객체)으로 변환하는 것을 의미합니다. json 라이브러리는 JSON을 Python의 사전 또는 목록으로 구문 분석합니다. 이를 위해 다음과 같이 loads () 함수 (문자열에서로드)를 사용합니다.

산출

다음은 샘플 json 파일입니다.

data1.json
{"menu": {
   "id": "file",
   "value": "File",
   "popup": {
      "menuitem": [
         {"value": "New", "onclick": "CreateNewDoc()"},
         {"value": "Open", "onclick": "OpenDoc()"},
         {"value": "Close", "onclick": "CloseDoc()"}
      ]
   }
}}

위의 내용 (Data1.json)은 일반적인 사전처럼 보입니다. pickle을 사용하여이 파일을 저장할 수 있지만 그 출력은 사람이 읽을 수있는 형식이 아닙니다.

JSON (Java Script Object Notification)은 매우 간단한 형식이며 이것이 인기의 이유 중 하나입니다. 이제 아래 프로그램을 통해 json 출력을 살펴 보겠습니다.

산출

위에서 우리는 읽기를 위해 json 파일 (data1.json)을 열고 파일 핸들러를 얻고 json.load에 전달하고 객체를 다시 가져옵니다. 객체의 출력을 인쇄하려고 할 때 json 파일과 동일합니다. 객체의 유형은 사전이지만 Python 객체로 나옵니다. 이 피클을 본 것처럼 json에 쓰는 것은 간단합니다. 위에서 json 파일을로드하고 다른 키 값 쌍을 추가하고 동일한 json 파일에 다시 작성합니다. 이제 data1.json을 보면 이전과 같은 형식이 아니라 다르게 보입니다.

출력이 동일하게 보이게하려면 (사람이 읽을 수있는 형식) 프로그램의 마지막 줄에 몇 개의 인수를 추가합니다.

json.dump(conf, fh, indent = 4, separators = (‘,’, ‘: ‘))

마찬가지로 pickle과 마찬가지로 덤프로 문자열을 인쇄하고로드로로드 할 수 있습니다. 아래는 그 예입니다.

YAML

YAML은 모든 프로그래밍 언어에 대해 가장 인간 친화적 인 데이터 직렬화 표준 일 수 있습니다.

Python yaml 모듈은 pyaml이라고합니다.

YAML은 JSON의 대안입니다-

  • Human readable code − YAML은 사람이 가장 읽을 수있는 형식이므로 앞 페이지 내용도 YAML로 표시되어이 점을 알 수 있습니다.

  • Compact code − YAML에서는 공백 들여 쓰기를 사용하여 대괄호가 아닌 구조를 나타냅니다.

  • Syntax for relational data − 내부 참조의 경우 앵커 (&)와 별칭 (*)을 사용합니다.

  • One of the area where it is used widely is for viewing/editing of data structures − 예를 들어 구성 파일, 디버깅 중 덤프 및 문서 헤더.

YAML 설치

yaml은 내장 모듈이 아니므로 수동으로 설치해야합니다. Windows 시스템에 yaml을 설치하는 가장 좋은 방법은 pip를 사용하는 것입니다. Windows 터미널에서 아래 명령을 실행하여 yaml을 설치하십시오.

pip install pyaml (Windows machine)
sudo pip install pyaml (*nix and Mac)

위의 명령을 실행하면 현재 최신 버전에 따라 화면에 아래와 같은 내용이 표시됩니다.

Collecting pyaml
Using cached pyaml-17.12.1-py2.py3-none-any.whl
Collecting PyYAML (from pyaml)
Using cached PyYAML-3.12.tar.gz
Installing collected packages: PyYAML, pyaml
Running setup.py install for PyYAML ... done
Successfully installed PyYAML-3.12 pyaml-17.12.1

테스트하려면 Python 셸로 이동하여 yaml 모듈을 가져오고, yaml을 가져옵니다. 오류가 없으면 설치가 성공했다고 말할 수 있습니다.

pyaml 설치 후 아래 코드를 살펴 보겠습니다.

script_yaml1.py

위에서 우리는 사전, 목록 및 튜플의 세 가지 데이터 구조를 만들었습니다. 각 구조에서 yaml.dump를 수행합니다. 중요한 점은 출력이 화면에 표시되는 방식입니다.

산출

사전 출력이 깨끗해 보입니다. 핵심 가치.

다른 개체를 구분하는 공백입니다.

목록은 대시 (-)로 표시됩니다.

튜플은 먼저 !! Python / tuple로 표시된 다음 목록과 동일한 형식으로 표시됩니다.

yaml 파일로드

여기에 다음이 포함 된 yaml 파일이 하나 있다고 가정 해 보겠습니다.

---
# An employee record
name: Raagvendra Joshi
job: Developer
skill: Oracle
employed: True
foods:
   - Apple
   - Orange
   - Strawberry
   - Mango
languages:
   Oracle: Elite
   power_builder: Elite
   Full Stack Developer: Lame
education:
   4 GCSEs
   3 A-Levels
   MCA in something called com

이제 yaml.load 함수를 통해이 yaml 파일을로드하는 코드를 작성해 보겠습니다. 아래는 동일한 코드입니다.

출력물이 그다지 읽기 어렵 기 때문에 결국 json을 사용하여 예쁘게 만듭니다. 우리가 얻은 출력과 우리가 가진 실제 yaml 파일을 비교하십시오.

산출

소프트웨어 개발의 가장 중요한 측면 중 하나는 디버깅입니다. 이 섹션에서는 기본 제공 디버거 또는 타사 디버거를 사용하여 Python을 디버깅하는 다양한 방법을 살펴 봅니다.

PDB – Python 디버거

모듈 PDB는 중단 점 설정을 지원합니다. 중단 점은 프로그램 상태에 대한 추가 정보를 얻을 수있는 프로그램의 의도적 인 일시 중지입니다.

중단 점을 설정하려면 다음 줄을 삽입하십시오.

pdb.set_trace()

pdb_example1.py
import pdb
x = 9
y = 7
pdb.set_trace()
total = x + y
pdb.set_trace()

이 프로그램에 몇 가지 중단 점을 삽입했습니다. 프로그램은 각 중단 점 (pdb.set_trace ())에서 일시 중지됩니다. 변수 내용을 보려면 변수 이름을 입력하면됩니다.

c:\Python\Python361>Python pdb_example1.py
> c:\Python\Python361\pdb_example1.py(8)<module>()
-> total = x + y
(Pdb) x
9
(Pdb) y
7
(Pdb) total
*** NameError: name 'total' is not defined
(Pdb)

c를 누르거나 다음 중단 점까지 프로그램 실행을 계속합니다.

(Pdb) c
--Return--
> c:\Python\Python361\pdb_example1.py(8)<module>()->None
-> total = x + y
(Pdb) total
16

결국 훨씬 더 큰 프로그램, 즉 서브 루틴을 사용하는 프로그램을 디버깅해야합니다. 그리고 때로는 찾으려고하는 문제가 서브 루틴 내부에있을 것입니다. 다음 프로그램을 고려하십시오.

import pdb
def squar(x, y):
   out_squared = x^2 + y^2
   return out_squared
if __name__ == "__main__":
   #pdb.set_trace()
   print (squar(4, 5))

이제 위의 프로그램을 실행하면

c:\Python\Python361>Python pdb_example2.py
> c:\Python\Python361\pdb_example2.py(10)<module>()
-> print (squar(4, 5))
(Pdb)

우리는 사용할 수 있습니다 ?도움을받을 수 있지만 화살표는 곧 실행될 줄을 나타냅니다. 이 시점에서 s를 눌러s 그 라인으로 발걸음을 옮깁니다.

(Pdb) s
--Call--
>c:\Python\Python361\pdb_example2.py(3)squar()
-> def squar(x, y):

이것은 함수에 대한 호출입니다. 코드에서 현재 위치에 대한 개요를 보려면 l을 시도하십시오.

(Pdb) l
1 import pdb
2
3 def squar(x, y):
4 -> out_squared = x^2 + y^2
5
6 return out_squared
7
8 if __name__ == "__main__":
9 pdb.set_trace()
10 print (squar(4, 5))
[EOF]
(Pdb)

n을 눌러 다음 줄로 이동할 수 있습니다. 이 시점에서 당신은 out_squared 메소드 안에 있고 함수 .ie x와 y 안에 선언 된 변수에 접근 할 수 있습니다.

(Pdb) x
4
(Pdb) y
5
(Pdb) x^2
6
(Pdb) y^2
7
(Pdb) x**2
16
(Pdb) y**2
25
(Pdb)

따라서 ^ 연산자가 우리가 원하는 것이 아니라는 것을 알 수 있습니다. 대신 ** 연산자를 사용하여 제곱을해야합니다.

이렇게하면 함수 / 메소드 내에서 프로그램을 디버깅 할 수 있습니다.

벌채 반출

로깅 모듈은 Python 버전 2.3부터 Python 표준 라이브러리의 일부였습니다. 기본 제공 모듈이기 때문에 모든 Python 모듈이 로깅에 참여할 수 있으므로 애플리케이션 로그에 타사 모듈의 메시지와 통합 된 자체 메시지가 포함될 수 있습니다. 많은 유연성과 기능을 제공합니다.

로깅의 이점

  • Diagnostic logging − 응용 프로그램의 동작과 관련된 이벤트를 기록합니다.

  • Audit logging − 비즈니스 분석을위한 이벤트를 기록합니다.

메시지는 "심각도"& minu 수준으로 작성 및 기록됩니다.

  • DEBUG (debug()) − 개발을위한 진단 메시지.

  • INFO (info()) − 표준 "진행"메시지.

  • WARNING (warning()) − 심각하지 않은 문제를 감지했습니다.

  • ERROR (error()) − 심각한 오류가 발생했습니다.

  • CRITICAL (critical()) − 일반적으로 치명적인 오류 (프로그램 중지).

아래의 간단한 프로그램을 살펴 보겠습니다.

import logging

logging.basicConfig(level=logging.INFO)

logging.debug('this message will be ignored') # This will not print
logging.info('This should be logged') # it'll print
logging.warning('And this, too') # It'll print

위에서 우리는 심각도 수준에서 메시지를 로깅합니다. 먼저 모듈을 가져오고 basicConfig를 호출하고 로깅 수준을 설정합니다. 위에서 설정 한 레벨은 INFO입니다. 그런 다음 디버그 문, 정보 문 및 경고 문의 세 가지 문이 있습니다.

logging1.py의 출력

INFO:root:This should be logged
WARNING:root:And this, too

info 문이 debug 문 아래에 있으므로 디버그 메시지를 볼 수 없습니다. 출력 터미널에서도 디버그 문을 얻으려면 basicConfig 수준 만 변경하면됩니다.

logging.basicConfig(level = logging.DEBUG)

출력에서 볼 수 있습니다.

DEBUG:root:this message will be ignored
INFO:root:This should be logged
WARNING:root:And this, too

또한 기본 동작은 로깅 수준을 설정하지 않으면 경고임을 의미합니다. 위의 프로그램에서 두 번째 줄을 주석 처리하고 코드를 실행하십시오.

#logging.basicConfig(level = logging.DEBUG)

산출

WARNING:root:And this, too

로깅 수준에 내장 된 Python은 실제로 정수입니다.

>>> import logging
>>>
>>> logging.DEBUG
10
>>> logging.CRITICAL
50
>>> logging.WARNING
30
>>> logging.INFO
20
>>> logging.ERROR
40
>>>

로그 메시지를 파일에 저장할 수도 있습니다.

logging.basicConfig(level = logging.DEBUG, filename = 'logging.log')

이제 모든 로그 메시지는 화면 대신 현재 작업 디렉토리의 파일 (logging.log)로 이동합니다. 이것은 우리가받은 메시지의 사후 분석을 할 수있게 해주므로 훨씬 더 나은 접근 방식입니다.

로그 메시지로 날짜 스탬프를 설정할 수도 있습니다.

logging.basicConfig(level=logging.DEBUG, format = '%(asctime)s %(levelname)s:%(message)s')

출력은 다음과 같습니다.

2018-03-08 19:30:00,066 DEBUG:this message will be ignored
2018-03-08 19:30:00,176 INFO:This should be logged
2018-03-08 19:30:00,201 WARNING:And this, too

벤치마킹

벤치마킹 또는 프로파일 링은 기본적으로 코드 실행 속도와 병목 현상이있는 위치를 테스트하는 것입니다. 이렇게하는 주된 이유는 최적화를위한 것입니다.

timeit

Python에는 timeit이라는 내장 모듈이 있습니다. 작은 코드 조각의 시간을 측정하는 데 사용할 수 있습니다. timeit 모듈은 플랫폼 별 시간 함수를 사용하므로 가능한 한 가장 정확한 타이밍을 얻을 수 있습니다.

따라서 각 코드에서 가져온 두 개의 코드를 비교 한 다음 스크립트를 최적화하여 더 나은 성능을 제공 할 수 있습니다.

timeit 모듈에는 명령 줄 인터페이스가 있지만 가져올 수도 있습니다.

스크립트를 호출하는 방법에는 두 가지가 있습니다. 먼저 스크립트를 사용하여 아래 코드를 실행하고 출력을 확인하십시오.

import timeit
print ( 'by index: ', timeit.timeit(stmt = "mydict['c']", setup = "mydict = {'a':5, 'b':10, 'c':15}", number = 1000000))
print ( 'by get: ', timeit.timeit(stmt = 'mydict.get("c")', setup = 'mydict = {"a":5, "b":10, "c":15}', number = 1000000))

산출

by index: 0.1809192126703489
by get: 0.6088525265034692

위에서 우리는 두 가지 다른 방법, 즉 아래 첨자를 사용하고 사전 키 값에 액세스합니다. 아주 작은 데이터에 대해서는 너무 빨리 실행되므로 문을 100 만 번 실행합니다. 이제 get에 비해 훨씬 빠르게 인덱스 액세스를 볼 수 있습니다. 코드를 여러 번 실행할 수 있으며 더 나은 이해를 위해 시간 실행에 약간의 차이가있을 것입니다.

또 다른 방법은 명령 줄에서 위의 테스트를 실행하는 것입니다. 해보자

c:\Python\Python361>Python -m timeit -n 1000000 -s "mydict = {'a': 5, 'b':10, 'c':15}" "mydict['c']"
1000000 loops, best of 3: 0.187 usec per loop

c:\Python\Python361>Python -m timeit -n 1000000 -s "mydict = {'a': 5, 'b':10, 'c':15}" "mydict.get('c')"
1000000 loops, best of 3: 0.659 usec per loop

위의 출력은 시스템 하드웨어와 현재 시스템에서 실행중인 모든 응용 프로그램에 따라 다를 수 있습니다.

함수를 호출하려는 경우 아래에서 timeit 모듈을 사용할 수 있습니다. 테스트 할 함수 내에 여러 문을 추가 할 수 있습니다.

import timeit

def testme(this_dict, key):
   return this_dict[key]

print (timeit.timeit("testme(mydict, key)", setup = "from __main__ import testme; mydict = {'a':9, 'b':18, 'c':27}; key = 'c'", number = 1000000))

산출

0.7713474590139164