Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

17 KiB

Отчет по теме 8. Модули и структурирование программы

Цель работы: Изучение способов разработки программ, состоящих из нескольких частей – модулей, и обеспечения взаимодействия этих частей при выполнении программы.

Терехов Фёдор Валерьевич, А-01-23

1. Создание и использование модулей в среде Python.

Модулем в среде Python называется любая часть программного кода на этом языке, записанная в отдельном файле. В языке Python модули также являются объектами класса module.

1.1. Запуск модуля на выполнение путем его импорта.

Файл Mod1.py:

perm1=input('Mod1:Введите значение = ')
print('Mod1:Значение perm1=',perm1)

Класс и атрибуты объекта Mod1:

<class 'module'>
['__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'perm1']
print('perm1=',Mod1.perm1)

Ответ программы:

Mod1:Значение perm1= 3
perm1= 3

1.2. Атрибут sys.modules.

Импортированные модули заносятся в словарь – значение атрибута sys.modules. Их можно увидеть по инструкции.

print("До удаления модуля Mod1:",sorted(sys.modules.keys()))
sys.modules.pop('Mod1')
print("После удаления модуля Mod1:",sorted(sys.modules.keys()))
import Mod1
print("После восстановления модуля Mod1:",sorted(sys.modules.keys()))
sys.modules.pop('Mod1')
print("После повторного удаления модуля Mod1:",sorted(sys.modules.keys()))

Ответ программы:

До удаления модуля Mod1: ['Mod1', 'Mod2', '__main__', '_abc', '_bootlocale', '_codecs', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_imp', '_io', '_locale', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'encodings', 'encodings.aliases', 'encodings.cp437', 'encodings.latin_1', 'encodings.utf_8', 'genericpath', 'io', 'marshal', 'math', 'os', 'os.path', 'posix', 'posixpath', 'site', 'stat', 'sys', 'time', 'zipimport']
После удаления модуля Mod1: ['Mod2', '__main__', '_abc', '_bootlocale', '_codecs', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_imp', '_io', '_locale', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'encodings', 'encodings.aliases', 'encodings.cp437', 'encodings.latin_1', 'encodings.utf_8', 'genericpath', 'io', 'marshal', 'math', 'os', 'os.path', 'posix', 'posixpath', 'site', 'stat', 'sys', 'time', 'zipimport']
Mod1:Введите значение = 3
Mod1:Значение perm1= 3
После восстановления модуля Mod1: ['Mod1', 'Mod2', '__main__', '_abc', '_bootlocale', '_codecs', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_imp', '_io', '_locale', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'encodings', 'encodings.aliases', 'encodings.cp437', 'encodings.latin_1', 'encodings.utf_8', 'genericpath', 'io', 'marshal', 'math', 'os', 'os.path', 'posix', 'posixpath', 'site', 'stat', 'sys', 'time', 'zipimport']
После повторного удаления модуля Mod1: ['Mod2', '__main__', '_abc', '_bootlocale', '_codecs', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_imp', '_io', '_locale', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'encodings', 'encodings.aliases', 'encodings.cp437', 'encodings.latin_1', 'encodings.utf_8', 'genericpath', 'io', 'marshal', 'math', 'os', 'os.path', 'posix', 'posixpath', 'site', 'stat', 'sys', 'time', 'zipimport']

1.3. Запуск модуля на выполнение с помощью функции exec().

Запуск модуля на выполнение может быть произведен с использованием ранее изученной функции exec. При этом модуль не требуется импортировать. Функция exec действует так, как будто на месте обращения к ней в программу вставлен код из объекта-аргумента функции. Внимание: объект-модуль при этом не создается! Созданные при выполнении модуля объекты становятся объектами главной программы!

import Mod1
import os

exec(open(os.getcwd() + '/TEMA8/Mod1.py').read())
exec(open(os.getcwd() + '/TEMA8/Mod1.py').read())
exec(open(os.getcwd() + '/TEMA8/Mod1.py').read())

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
Mod1:Введите значение = 2
Mod1:Значение perm1= 2
Mod1:Введите значение = 3
Mod1:Значение perm1= 3
Mod1:Введите значение = 1
Mod1:Значение perm1= 1

1.4. Использование инструкции from … import ….

В одном модуле может содержаться несколько программных единиц (например, функций или пользовательских объектов). Тогда можно осуществлять импорт модуля не целиком, а только часть содержащихся в нем объектов.

from Mod1 import perm1
print('perm1=',perm1)
print(dir())

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
perm1= 1
['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'perm1']
  • Объект Mod1 не появился в памяти. При использовании from Mod1 import perm1 импортируется только функция perm1, но не сам модуль.
  • Модуль выполнился полностью. При импорте любого объекта из модуля весь модуль выполняется один раз.
  • Объект perm1 ссылается на функцию perm1 из модуля Mod1. После выполнения from Mod1 import perm1 мы можем напрямую вызывать perm1() без указания имени модуля.
from Mod2 import beta
g=beta(2)
print(g)
print(dir())

Ответ программы:

****BETA****
535.4916555247646
['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'beta', 'g', 'perm1']

Объект Mod2 не появился в памяти. При использовании from Mod2 import beta импортируется только функция beta, но не сам модуль.

alpha()
Traceback (most recent call last):
  File "/Users/fedorterekhov/Library/Mobile Documents/com~apple~CloudDocs/Учёба/3 курс/POAS_labs/python-labs/TEMA8/task1.4.py", line 9, in <module>
    alpha()
NameError: name 'alpha' is not defined

Функция alpha не была импортированна из Mod2, код выдаёт ошибку.

from Mod2 import alpha as al
print(dir())
['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'al', 'beta', 'g', 'perm1']

Из Mod2 была импортирована функция alpha как al, поэтому в памяти есть объект с названием al, но он выполняет действия функции alpha.

del al,beta
print("После первого удаления:", dir())
from Mod2 import alpha as al, beta as bt
print("После добавления al и bt:", dir())
del al,bt
print("После второго удаления:", dir())
from Mod2 import *
tt=alpha()  #На запрос введите значение 0.12
uu=beta(float(tt))
print(uu)

Ответ программы:

После первого удаления: ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'g', 'perm1']
После добавления al и bt: ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'al', 'bt', 'g', 'perm1']
После второго удаления: ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'g', 'perm1']
****ALPHA****
Значение t=0.12
****BETA****
1.4578913609506803

2. Создание многомодульных программ.

2.1. Пример простой многомодульной программы.

Mod0:

#Модуль Mod0
import Mod1
print('perm1=',Mod1.perm1)
from Mod2 import alpha as al
tt=al()
print('tt=',tt)
from Mod2 import beta
qq=beta(float(tt))
print('qq=',qq)
import Mod0
print("Mod0.qq:", Mod0.qq)
print("Mod0.tt:", Mod0.tt)
print("Mod0.Mod1.perm1:", Mod0.Mod1.perm1)

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
perm1= 1
****ALPHA****
Значение t=1
tt= 1
****BETA****
qq= 23.140692632779267
Mod0.qq: 23.140692632779267
Mod0.tt: 1
Mod0.Mod1.perm1: 1

2.2. Многомодульная программа.

Модуль MM1:

def realdvig(xtt,kk1,TT,yti1,ytin1):
		#Модель реального двигателя
		yp=kk1*xtt  #усилитель
		yti1=yp+yti1  #Интегратор
		ytin1=(yti1+TT*ytin1)/(TT+1)
		return [yti1,ytin1]

def tahogen(xtt,kk2,yti2):
		#Модель тахогенератора
		yp=kk2*xtt   #усилитель
		yti2=yp+yti2 #интегратор
		return yti2

def nechus(xtt,gran):
#зона нечувствит
    if xtt<gran and xtt>(-gran):
        ytt=0
    elif xtt>=gran:
        ytt=xtt-gran
    elif xtt<=(-gran):
        ytt=xtt+gran
    return ytt

Модуль MM2:

znach=input('k1,T,k2,Xm,A,F,N=').split(',')
k1=float(znach[0])
T=float(znach[1])
k2=float(znach[2])
Xm=float(znach[3])
A=float(znach[4])
F=float(znach[5])
N=int(znach[6])

import math
vhod=[]
for i in range(N):
	vhod.append(A*math.sin((2*i*math.pi)/F))

import MM1 as mod
yi1=0;yin1=0;yi2=0
vyhod=[]
for xt in vhod:
	xt1=xt-yi2   #отрицательная обратная связь
	[yi1,yin1]=mod.realdvig(xt1,k1,T,yi1,yin1)
	yi2=mod.tahogen(yin1,k2,yi2)
	yt=mod.nechus(yin1,Xm)
	vyhod.append(yt)

Ответ программы:

k1,T,k2,Xm,A,F,N=2.0,5.0,0.5,0.3,1.0,20.0,15
[0, 0, 0.06760452541690626, 0.4793408756948438, 0.931148162040943, 1.2274710293386355, 1.1233305479712674, 0.400902344081069, -0.42732659995249095, -2.4154244800859455, -4.520514725079971, -6.022685151402792, -6.061564289571653, -3.8945172200444302, 0.16254138526353118]

Модуль MM0:

import MM2
print('y=',MM2.vyhod)

Ответ программы:

k1,T,k2,Xm,A,F,N=2.0,5.0,0.5,0.3,1.0,20.0,15   
y= [0, 0, 0.06760452541690626, 0.4793408756948438, 0.931148162040943, 1.2274710293386355, 1.1233305479712674, 0.400902344081069, -0.42732659995249095, -2.4154244800859455, -4.520514725079971, -6.022685151402792, -6.061564289571653, -3.8945172200444302, 0.16254138526353118]

2.3. Области действия объектов в модулях.

В ранее созданных модулях вводятся и используются следующие объекты: Mod1: perm1 Mod2: функции alpha, beta; переменные t, expi Mod0: переменные tt,qq Проведите следующие эксперименты:

  • Попробуйте вставить в функции alpha обращение к функции beta и, наоборот, из beta – к alpha.
  • Попробуйте отобразить на экране в модуле Mod0 значения объектов t и expi.
  • Попробуйте в модуле Mod0 увеличить в 3 раза значение объекта perm1 и отобразить его после этого на экране.
  • Попробуйте в командной строке (в главном модуле) увеличить в 2 раза значения объектов perm1, tt, qq.
    Обобщите результаты экспериментов и сделайте выводы об областях действия объектов.
  • Попробуйте вставить в функции alpha обращение к функции beta и, наоборот, из beta – к alpha.

Mod2.py:

import math
def alpha():
    print('****ALPHA****')
    t=input('Значение t=')
    result_beta = beta(float(t))
    return t

def beta(q):
    print('****BETA****')
    expi=q*math.pi
    alpha_result = alpha()
    return math.exp(expi)

Программа запуска:

from Mod2 import alpha, beta
alpha()
beta(2)

Ответ программы:

****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=1
****BETA****
****ALPHA****
Значение t=^Z
zsh: suspended  /usr/bin/python3

В конце завершил действие программы комбинацией "contol+z".

Вывод: программа попадает в бесконечную рекурсию (зацикливание).

  • Попробуйте отобразить на экране в модуле Mod0 значения объектов t и expi.

Mod0.py:

#Модуль Mod0
import Mod1
print('perm1=',Mod1.perm1)
from Mod2 import alpha as al
tt=al()
import Mod2
print(Mod2.t)   #отображение t
print(Mod2.expi)    #отображение expi
print('tt=',tt)
from Mod2 import beta
qq=beta(float(tt))
print('qq=',qq)

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
perm1= 1
****ALPHA****
Значение t=1
Traceback (most recent call last):
  File "/Users/fedorterekhov/Library/Mobile Documents/com~apple~CloudDocs/Учёба/3 курс/POAS_labs/python-labs/TEMA8/Mod0.py", line 7, in <module>
    print(Mod2.t)   #отображение t
AttributeError: module 'Mod2' has no attribute 't'

Вывод: программа завершается с ошибкой AttributeError, потому что пытается обратиться к локальным переменным функций как к атрибутам модуля.

  • Попробуйте в модуле Mod0 увеличить в 3 раза значение объекта perm1 и отобразить его после этого на экране.

Mod0.py:

#Модуль Mod0
import Mod1
print('perm1=',Mod1.perm1)
Memory=Mod1.perm1
Mod1.perm1=Mod1.perm1*3
print('Увеличенное perm1=',Mod1.perm1)
Mod1.perm1=Memory
Mod1.perm1=int(Mod1.perm1)*3
print('Увеличенное perm1 как инт=',Mod1.perm1)
from Mod2 import alpha as al
tt=al()
import Mod2
print('tt=',tt)
from Mod2 import beta
qq=beta(float(tt))
print('qq=',qq)

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
perm1= 1
Увеличенное perm1= 111
Увеличенное perm1 как инт= 3
****ALPHA****
Значение t=1
tt= 1
****BETA****
qq= 23.140692632779267

Вывод: значение perm1 сохраняется как строка, а не как число. Чтобы увеличить его в 3 раза, необходимо преобразовать perm1 в число.

  • Попробуйте в командной строке (в главном модуле) увеличить в 2 раза значения объектов perm1, tt, qq.
import Mod0

from Mod1 import perm1
from Mod0 import tt, qq
perm1 *= 2 
tt *= 2
qq *= 2
print(perm1, tt, qq)

Ответ программы:

Mod1:Введите значение = 1
Mod1:Значение perm1= 1
perm1= 1
****ALPHA****
Значение t=1
tt= 1
****BETA****
qq= 23.140692632779267
11 11 46.281385265558534

Вывод: perm1 и tt - строки, при их умножении на два строки удваиваются. qq - численная переменная, при умножении на два удваивается именно значение числа.