|
|
|
|
# -*- encoding: utf-8 -*-
|
|
|
|
|
'''
|
|
|
|
|
@File : X1_new.py
|
|
|
|
|
@License : (C)Copyright 2018-2022
|
|
|
|
|
|
|
|
|
|
@Modify Time @Author @Version @Desciption
|
|
|
|
|
------------ ------- -------- -----------
|
|
|
|
|
2023/6/30 15:51 zart20 1.0 None
|
|
|
|
|
'''
|
|
|
|
|
'''
|
|
|
|
|
【编程16.1】编制程序:对卷积神经网络模型的数据结构赋值
|
|
|
|
|
【目的及编程说明】读者通过编程16.1可理解可视化构建卷积神经网络模型相关的数据结构及其作用。这里给出的数据结构,
|
|
|
|
|
能够支持16.2及16.4节的输出界面,即能够绘制出一张卷积神经网络结构图。编程16.1要求为模型对象总表AllModelObj
|
|
|
|
|
和AllModelConn赋值。要求包含所有类型的图元对象(见下表述),并建立适当的连接关系。
|
|
|
|
|
网络对象总表AllModelObj(ObjID,ObjType,ObjLable,ProcFunc, SetParaFunc, ParaString, ObjX,ObjY),
|
|
|
|
|
其中:ObjID 表示图元号;ObjType表示图元类别(1—数据集,2—卷积,3—池化,4—全连接,5—非线性,6—分类,7—误差计算,
|
|
|
|
|
8—卷积调整,9—全连接调整);ObjLable为对象标签;ProcFunc为该图元被执行时的基本操作函数;
|
|
|
|
|
SetParaFunc为给基本操作函数赋参数值的函数;ParaString 为同类不同图元的参数,ObjX,
|
|
|
|
|
ObjY为对象在仿真输出屏幕中的位置(示例中未给出,由图形界面中的位置产生)。
|
|
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
# 定义一个函数,根据给定的参数生成一个AllModelObj对象
|
|
|
|
|
def create_almodelobj():
|
|
|
|
|
all_model = []
|
|
|
|
|
obj_label = ["数据集", "卷积", "池化", "全连接", "非线性",
|
|
|
|
|
"分类", "误差计算", "卷积调整", "全连接调整"]
|
|
|
|
|
proc_func = ['LoadData', 'ConvProc', 'MaxPoolProc', 'FullConnProc',
|
|
|
|
|
'NonlinearProc', 'ClassifierProc', 'ErrorProc',
|
|
|
|
|
'AjConvProc', 'AjFullconnProc']
|
|
|
|
|
set_para_func = ['SetDataPara', 'SetConvPara', 'SetPollPara',
|
|
|
|
|
'SetFullConnPara', 'SetNonLPara', 'SetClassifyPara',
|
|
|
|
|
'SetErrorPara','SetAjConvPara', 'SetAjFCPara']
|
|
|
|
|
for i in range(9):
|
|
|
|
|
obj = [] # 创建一个列表,存储对象的属性
|
|
|
|
|
obj.append(i) # 对象ID
|
|
|
|
|
obj.append(i % 9 + 1) # 对象类型
|
|
|
|
|
obj.append(obj_label[i % 9] + f"{i // 9 + 1}") # 对象标签
|
|
|
|
|
obj.append(proc_func[i % 9] + f"{i // 9 + 1}") # 对象基本功能函数
|
|
|
|
|
obj.append(set_para_func[i % 9] + f"{i // 9 + 1}") # 对象功能函数参数
|
|
|
|
|
obj.append('.')
|
|
|
|
|
obj.append(i * 20)
|
|
|
|
|
obj.append(i * 20)
|
|
|
|
|
all_model.append(obj)
|
|
|
|
|
return all_model
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
# all_model = create_almodelobj()
|
|
|
|
|
# print(all_model)
|
|
|
|
|
|
|
|
|
|
# 定义一个函数,生成一个AllModelConn对象的列表
|
|
|
|
|
def create_allmodelconn():
|
|
|
|
|
all_conn = [] # 创建一个空列表,存储所有对象
|
|
|
|
|
conn_obj_id = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] #储连接线编号
|
|
|
|
|
conn_type = [1, 1, 1, 1, 1, 1, 2, 2, 2, 2] #储连接线类别
|
|
|
|
|
# 定义一个列表,存储源图元对象
|
|
|
|
|
nobj_s = ["DataSet1", "Conv1", "Pool1", "FullConn1", "Nonline1",
|
|
|
|
|
"Classifier1", "Error1", "Error1", "AjFullconn1", "AjConv1"]
|
|
|
|
|
# 定义一个列表,存储目标图元对象
|
|
|
|
|
nobj_e = ["Conv1", "Pool1", "FullConn1", "Nonline1", "Classifier1",
|
|
|
|
|
"Error1", "AjFullconn1", "AjConv1", "FullConn1", "Conv1"]
|
|
|
|
|
all_conn = [ # 使用列表推导式,根据给定的参数生成所有对象
|
|
|
|
|
[ # 创建一个列表,存储对象的属性
|
|
|
|
|
conn_obj_id[i], # 连接线编号
|
|
|
|
|
conn_type[i], # 连接线类别
|
|
|
|
|
nobj_s[i], # 源图元对象
|
|
|
|
|
nobj_e[i] # 目标图元对象
|
|
|
|
|
]
|
|
|
|
|
for i in range(len(conn_obj_id)) # 循环遍历连接线编号的长度
|
|
|
|
|
]
|
|
|
|
|
return all_conn
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
# all_conn = create_allmodelconn()
|
|
|
|
|
# print(all_conn)
|
|
|
|
|
# 导入需要的库
|
|
|
|
|
import numpy as np # 用于处理数组
|
|
|
|
|
from PIL import Image # 用于读取图片
|
|
|
|
|
'''
|
|
|
|
|
【编程16.2】编制程序:练习加载输入数据。
|
|
|
|
|
【目的及编程说明】读者通过编程16.2可理解加载数据的过程。本项目中假定输入数据的类型为图片数据,
|
|
|
|
|
要求读者能够实现将程序指定的图片转换为卷积神经网络模型能够接受的数据格式即数组。具体实现要求:
|
|
|
|
|
(1)程序读取一个固定路径的图片,;
|
|
|
|
|
(2)将图片缩放为 32x32固定大小,并将其转换为数组格式;
|
|
|
|
|
(3)将数组中具体数值归一化。该程序为编制LoadData()奠定基础。
|
|
|
|
|
'''
|
|
|
|
|
# 定义一个函数,实现图片转换为数组的功能
|
|
|
|
|
def image_to_array(path,height,width):
|
|
|
|
|
img = Image.open(path).convert("L") #转换为灰度模式
|
|
|
|
|
img = img.resize((height, width)) # 将图片缩放为height*width的固定大小
|
|
|
|
|
data = np.array(img) # 将图片转换为数组格式
|
|
|
|
|
data = data / 255.0 # 将数组中的数值归一化,除以255
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
# path = 'data/train/1.jpg' # 路径
|
|
|
|
|
# height, width =28, 28 # 尺寸:h,w
|
|
|
|
|
# data_image = image_to_array(path,height,width)
|
|
|
|
|
# print(data_image)
|
|
|
|
|
|
|
|
|
|
'''
|
|
|
|
|
【编程16.3】编制程序:完成指定卷积核参数的卷积操作。
|
|
|
|
|
【目的及编程说明】读者通过编程16.3可理解卷积操作的具体计算过程。该函数的输入数据为数组格式的数据。具体实现要求:
|
|
|
|
|
(1)程序中给出卷积核大小为3*3的矩阵,并给矩阵赋初始值。例如初值[[ 1, -1, 2], [ 3, -2, 1], [ 0, 1, 0]]
|
|
|
|
|
或[[ 0, 1, 0], [ 1, -4, 1], [ 0, 1, 0]]或自己设定;
|
|
|
|
|
(2)默认步长(stride)为1;(3)完成卷积计算;
|
|
|
|
|
(4)返回卷积后的数组。该程序为编制ConvProc()奠定基础。
|
|
|
|
|
'''
|
|
|
|
|
# 导入需要的库
|
|
|
|
|
import numpy as np # 用于处理数组
|
|
|
|
|
# 定义一个函数,实现卷积操作的功能
|
|
|
|
|
def convolve(data, kernel):
|
|
|
|
|
# 获取数据和卷积核的形状
|
|
|
|
|
data_height, data_width = data.shape # 获取数据高宽
|
|
|
|
|
kernel_height, kernel_width = kernel.shape # 获取卷积核高宽
|
|
|
|
|
output_height = data_height - kernel_height + 1 # 以步长为1来计算输出的高宽
|
|
|
|
|
output_width = data_width - kernel_width + 1
|
|
|
|
|
output = np.zeros((output_height, output_width)) # 创建一个空的输出数组
|
|
|
|
|
for i in range(output_height): # 遍历输出数组的每个元素,计算卷积值
|
|
|
|
|
for j in range(output_width):
|
|
|
|
|
data_region = data[i:i+kernel_height, j:j+kernel_width] # 获取数据和卷积核的对应区域
|
|
|
|
|
conv_value = np.sum(data_region * kernel) # 计算对应区域的元素乘积之和,作为卷积值
|
|
|
|
|
output[i, j] = conv_value # 将卷积值赋给输出数组的对应元素
|
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
# path = 'data/train/1.jpg' # 路径
|
|
|
|
|
# height, width = 28, 28 # 尺寸:h,w
|
|
|
|
|
# data_image = image_to_array(path, height, width)
|
|
|
|
|
#
|
|
|
|
|
# # 定义一个卷积核,可以自己修改
|
|
|
|
|
# kernel = np.array([[1, -1, 2],
|
|
|
|
|
# [3, -2, 1],
|
|
|
|
|
# [0, 1, 0]])
|
|
|
|
|
# output = convolve(data_image, kernel) # 传入数据和卷积核
|
|
|
|
|
# print(output)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
'''
|
|
|
|
|
【编程16.4】编制程序:完成指定参数的全连接操作。
|
|
|
|
|
【目的及编程说明】读者通过编程16.4可理解全连接操作的具体计算过程。
|
|
|
|
|
该函数的输入数据为数组格式的数据,输出为计算后的数组。具体实现要求:
|
|
|
|
|
(1)程序中给定权重矩阵[[0.1, 0.2, 0.3], [0.2, 0.3, 0.4]]和偏置向量[0.1, 0.2]
|
|
|
|
|
(注:可自己自主设定参数);
|
|
|
|
|
(2)计算输入数据和权重矩阵的乘积(注意矩阵的维度),并加上偏置向量;
|
|
|
|
|
(3)返回计算后的数组。该程序为编制FullConnProc()奠定基础。
|
|
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
# 定义一个函数,实现全连接操作
|
|
|
|
|
def full_connect(input_data, weight_matrix, bias_vector):
|
|
|
|
|
output_data = np.dot(input_data, weight_matrix) # 计算输入数据和权重矩阵的乘积
|
|
|
|
|
bias_vector = np.expand_dims(bias_vector, axis=0) # 扩展偏置向量的形状
|
|
|
|
|
bias_vector = np.expand_dims(bias_vector, axis=0)
|
|
|
|
|
output_data = output_data + bias_vector # 加上偏置向量
|
|
|
|
|
return output_data # 返回输出数据
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
# weight_matrix = np.array([[0.1, 0.2, 0.3], [0.2, 0.3, 0.4]]) # 给定权重矩阵和偏置向量,可以自己调整
|
|
|
|
|
# bias_vector = np.array([0.1, 0.2, 0.3])
|
|
|
|
|
# # 假设输入数据为一个三维数组,每一层代表一个样本,每一行代表一个特征,每一列代表一个通道
|
|
|
|
|
# input_data = np.array([[[1, 2], [3, 4], [5, 6]],
|
|
|
|
|
# [[7, 8], [9, 10], [11, 12]],
|
|
|
|
|
# [[13, 14], [15, 16], [17, 18]]])
|
|
|
|
|
# output_data = full_connect(input_data,weight_matrix,bias_vector)
|
|
|
|
|
# print(output_data)
|
|
|
|
|
|