python實現簡單的學生管理系統

本文實例為大傢分享瞭python實現簡單學生管理系統的具體代碼,供大傢參考,具體內容如下

學生管理系統

相信大傢學各種語言的時候,練習總是會寫各種管理系統吧,管理系統主要有對數據的增刪查改操作,原理不難,適合作為練手的小程序

數據的結構

要保存數據就需要數據結構,比如c裡面的結構體啊,python裡面的列表,字典,還有類都是常用的數據類型
在這裡,我使用瞭鏈表來作為學生數據的數據結構,
即 Node類 和 Student_LinkList類,來實現鏈表

數據的持久化

我們在程序中產生的數據是保存在內存中的,程序一旦退出,下次就不能恢復此次的數據瞭,因此需要把內存種的數據,保存到文件或數據庫中,存儲起來,這個過程就叫數據的持久化

本程序使用瞭python標準庫pickle提供的序列化方法dump()和load()來實現數據的持久化

配置文件

使用配置文件,可以方便程序中使用不同的子類實現,

本程序使用configparser來對配置文件解析
本程序配置文件名為 Student.ini

#Student.ini文件
[Student]
student = Student_LinkList

[Persistence]
persistence = Persistence_Pickle
file = student.pik

類之間的關系

Student #和學生數據有關的抽象類
±- Student_LinkList
Persistence #和持久化有關的抽象類
±- Persistence_Pickle
MyConfigure #和配置文件讀取有關的類
UI #和交互有關的父類
±- Cmd_UI

界面預覽

源碼

'''
使用單鏈表實現的學生管理系統
'''
import pickle
import abc
import configparser

class Student(abc.ABC):
  '''
  抽象學生類
  '''
  @abc.abstractmethod
  def add(self):
    '''
    增加學生結點
    '''
    pass

  @abc.abstractmethod
  def ladd(self):
    '''
    從左側增加學生結點
    '''
    pass

  @abc.abstractmethod
  def delete(self,id_):
    '''
    根據id值來刪除一個結點
    '''
    pass

  @abc.abstractmethod
  def delete_name(self,name):
    '''
    根據姓名來刪除一個結點
    '''
    pass

  @abc.abstractmethod
  def insert(self,idx,val):
    '''
    插入到指定的位置
    '''
    pass

  @abc.abstractmethod
  def show(self):
    '''
    顯示所有的學生結點
    '''
    pass

  @abc.abstractmethod
  def search_id(self):
    '''
    根據id查詢節點
    '''
    pass

  @abc.abstractmethod
  def search_name(self):
    '''
    根據name查詢節點
    '''

  @abc.abstractmethod
  def modity_id(self):
    '''
    根據id找到節點,然後修改
    '''
    pass



class Node(object):
  '''
  學生鏈表結點
  '''
  def __init__(self,id_: int,name: str,sex: str,age: int,score: int):
    self.id = id_
    self.name = name
    self.sex = sex
    self.age = age
    self.score = score

    self.next = None

  def modity(self,id_,name,sex,age,score):
    '''
    修改
    '''
    self.id = id_
    self.name = name
    self.sex = sex
    self.age = age
    self.score = score


  def __str__(self):
    '''
    用於顯示輸出
    '''
    return f"[學生:{self.id:^2}]-->name:{self.name:^10}sex:{self.sex:^10}age:{self.age:^10}score:{self.score:^10}"

class Student_LinkList(Student):
  '''
  學生鏈表
  '''
  def __init__(self):
    self.head = Node(-1,'head','-1',-1,-1)
    self.length = 0
    self.tail = self.head #尾部結點用於尾插

  def add(self,id_,name,sex,age,score):
    '''
    添加一個學生結點,尾插
    '''
    #print('當前tail的值',self.tail)
    temp = Node(id_,name,sex,age,score)
    self.tail.next = temp 
    self.tail = self.tail.next

    self.length += 1
    print('[info]:添加成功')

  def ladd(self,id_,name,sex,age,score):
    '''
    添加一個學生,頭插
    '''
    temp = Node(id_,name,sex,age,score)
    temp.next = self.head.next
    self.head.next = temp

    if self.tail == self.head:
      self.tail = temp

    self.length += 1
    print('[info]:添加成功')

  def delete(self,id_):
    '''
    根據id值來刪除一個結點,用迭代實現
    '''
    p = self.head
    while p.next != None and p.next.id != id_:
      p = p.next

    if p.next == None:
      print('[error]:找不到id')
      return -1
    else:
      temp = p.next
      p.next = temp.next
      #如果刪除的是尾結點,還要移動tail
      if temp.next == None:
        self.tail = p
      del temp
    print('[info]:刪除成功')

  def delete_name(self,name):
    '''
    根據姓名來刪除一個結點,用遞歸實現
    '''
    def _func(node: Node,name: str):
      '''
      遞歸函數
      '''
      #到瞭尾巴節點瞭,還沒有找到
      if node.next == None:
        print('[info]:找不到name')
        return False
      elif node.next.name == name:
        temp = node.next
        node.next = temp.next
        #如果刪除的是尾結點,還要移動tail
        if temp.next == None:
          self.tail = node
        del temp
        print('[info]:刪除成功')
        return True
      else:
        return _func(node.next,name)

    t = self.head
    return _func(t,name)

  def insert(self,idx,id_,name,sex,age,score):
    '''
    在指定位置插入數據
    '''
    if idx > self.length or idx == 0:
      print(f'[error]:你輸入的索引非法(1-{self.length})')
      return 0
    p,cur = self.head,0
    while p != None and cur < idx-1:
      p = p.next

    if cur < idx-1:
      return -1
    else:
      temp = Node(id_,name,sex,age,score)
      temp.next = p.next
      p.next = temp
      return True
    print('[info]:插入成功')

  def search_id(self,id_):
    '''
    根據id查詢節點
    '''
    p = self.head
    while p != None and p.id != id_:
      p = p.next
    if p == None:
      return -1
    else:
      return p

  def search_name(self,name):
    '''
    根據name查詢節點
    '''
    p = self.head
    
    def _func(node: Node,name: str):
      '''
      遞歸函數
      '''
      if node == None:
        return -1
      elif node.name == name:
        return node
      return _func(node.next,name)

    return _func(p,name)

  def modity_id(self,id0,id_,name,sex,age,score):
    '''
    根據id找到節點,然後修改
    '''
    node = self.search_id(id0)
    if node == -1:
      print('[error]:找不到該id')
      return -1
    else:
      node.modity(id_,name,sex,age,score)


  def show(self):
    '''
    顯示所有的學生結點,迭代
    '''
    print(f'\n{"-"*25}以下是系統內數據{"-"*25}')
    temp = []
    p = self.head
    while p != None:
      temp.append(p)
      p = p.next
    return temp

class Student_Array():
  '''
  用數組實現學生數據存儲
  '''
  pass

class Student_Queue():
  '''
  用隊列實現
  '''
  pass

class Student_Dict():
  '''
  用隊列實現
  '''
  pass

class Persistence(abc.ABC):
  '''
  鏈表數據的持久化
  '''
  @abc.abstractmethod
  def save(self):
    '''
    把對象保存
    '''
    pass

  @abc.abstractmethod
  def load(self):
    '''
    加載對象
    '''
    pass

class Persistence_Pickle(Persistence):
  '''
  使用pickle來序列化
  '''
  def __init__(self,cls: Student,file_):
    self.filename = file_
    self.obj = None
    self.cls = cls

  def save(self):
    with open(self.filename,'wb') as f:
      pickle.dump(self.obj,f)

  def load(self):
    try:
      with open(self.filename,'rb') as f:
        temp = pickle.load(f)
    except:
      temp = globals()[self.cls]()
    print('返回temp:',type(temp))
    self.obj = temp
    return temp

class Persistence_File(Persistence):
  '''
  使用文件來持久化
  '''
  pass

class Persistence_Mysql(Persistence):
  '''
  使用Mysql數據庫來持久化
  '''
  pass

class Persistence_Socket(Persistence):
  '''
  使用遠程套接字持久化
  '''
  pass

class MyConfigure(object):
  '''
  用來讀取配置文件的類
  '''
  def __init__(self):
    self.config = configparser.ConfigParser()

  def save(self):
    '''
    保存配置文件
    '''
    with open('Student.ini','w') as f:
      self.config.write(f)

  def load(self):
    '''
    加載配置文件
    '''
    self.config.read('Student.ini')

  def get_student_class(self):
    '''
    獲得Student該使用哪個子類
    '''
    return self.config['Student']['student']

  def get_persistence_class(self):
    '''
    獲得持久化,該使用那個類,
    如果是Pickle或文件,還有file作為保存的文件名
    '''
    temp = {}
    temp['persistence'] = self.config['Persistence']['persistence']
    if 'Persistence_Pickle' in temp['persistence']:
      temp['file'] = self.config['Persistence']['file']
    return temp

class UI(object):
  '''
  界面交互
  '''
  def __init__(self):
    self.config = MyConfigure()
    self.config.load()
    s_class = self.config.get_student_class()
    p_class = self.config.get_persistence_class()

    self.persistence = globals()[p_class['persistence']](s_class,p_class['file'])
    self.student = self.persistence.load()
    print('實例化成功:',self.student,self.persistence)

  def save(self):
    '''
    把數據保存
    '''
    self.persistence.save()

  def quit(self):
    '''
    退出:先保存配置,然後退出
    '''
    self.config.save()
    self.save()

  def _show(self):
    '''
    顯示所有學生節點
    '''
    return self.student.show()


  def _add(self,direction,*temp):
    '''
    增加學生結點,
    direction 1左添加,2右添加
    '''
    if direction == 1:
      self.student.ladd(*temp)
    elif direction == 2:
      self.student.add(*temp)

  def _delete(self,attribute: int,val: str):
    '''
    刪除學生節點
    attribute: 需要根據哪個屬性刪除,1.id 或 2.name
    '''
    if attribute == 1:
      self.student.delete(val)
    elif attribute == 2:
      self.student.delete_name(val)

  def _insert(self,idx,*temp):
    '''
    把學生節點插入到指定的位置
    '''
    self.student.insert(idx,*temp)

  def _search(self,attribute,val):
    '''
    查詢
    '''
    if attribute == 1:
      return self.student.search_id(val)
    elif attribute == 2:
      return self.student.search_name(val)

  def _modity(self,attribute,id_,*temp):
    '''
    修改
    '''
    if attribute == 1:
      self.student.modity_id(id_,*temp)
    elif attribute == 2:
      print('[info]:因為沒實現,所以什麼也不做')
      pass #根據name修改沒有寫




class Cmd_UI(UI):
  '''
  命令行的交互界面
  '''
  def __init__(self):
    super(Cmd_UI,self).__init__()

  def get_input_1_2(self,info: str):
    '''
    獲得輸入,返回1或者2
    info: 描述輸入的信息
    '''
    x = None
    while x == None:
      temp = input(info)
      if temp == '1':
        x = 1
      elif temp == '2':
        x = 2
      else:
        print('你隻能輸入1或者2')
    return x

  def get_input_arg(self):
    '''
    獲得用戶的輸入構造學生節點
    '''
    id_ = input('請輸入id')
    name = input('請輸入姓名')
    sex = input('請輸入性別')
    age = input('請輸入年齡')
    score = input('請輸入成績')
    return (id_,name,sex,age,score)

  def delete(self):
    '''
    刪除節點
    '''
    info = '你想要根據哪個屬性刪除節點:1.id 2.name'
    attribute = self.get_input_1_2(info)
    val = input('輸入你想要刪除的值:')
    self._delete(attribute,val)

  def show(self):
    '''
    顯示
    '''
    rel = self._show()
    for i in rel:
      print(i)

  def add(self):
    '''
    增加學生結點
    '''
    info = '你想要插入的位置:1.左邊 2.右邊'
    direction = self.get_input_1_2(info)
    arg = self.get_input_arg()
    self._add(direction,*arg)

  def insert(self):
    '''
    新學生,插入到指定的位置
    '''
    idx = int(input('輸入要插入的位置'))
    temp = self.get_input_arg()
    self._insert(idx,*temp)

  def search(self):
    '''
    查詢學生
    '''
    info = '你想要根據哪個屬性搜索節點:1.id 2.name'
    attribute = self.get_input_1_2(info)
    val = input('輸入你想要查詢的值:')

    print(self._search(attribute,val))

  def modity(self):
    '''
    修改學生信息
    '''
    info = '你想要根據哪個屬性搜索節點:1.id 2.name'
    attribute = self.get_input_1_2(info)
    val_ = input('輸入要查詢的值:')
    temp = self.get_input_arg()
    self._modity(attribute,val_,*temp)

  def main(self):
    '''
    主流程
    '''
    info = '''
    *******************
    *kalpa學生管理系統*
    *  0.顯示數據  *
    *  1.增加數據  *
    *  2.刪除數據  *
    *  3.查詢數據  *
    *  4.修改數據  *
    *  5.保存並退出 *
    *******************
    '''
    print(info)
    a = '0'
    while a in ['0','1','2','3','4','5']:
      if a == '0':
        self.show()
      elif a == '1':
        self.add()
      elif a == '2':
        self.delete()
      elif a == '3':
        self.search()
      elif a == '4':
        self.modity()
      elif a == '5':
        self.quit()
        return
      a = input('>>')


if __name__ == "__main__":
  ui = Cmd_UI()
  ui.main()

以上就是本文的全部內容,希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。

推薦閱讀: