7. 搞懂LangChain开发库,还有dotenv实战
为帮助开发者快速吃透 LangChain 整体架构、解决入门难题,本文将拆解项目核心模块,并讲解在实际项目中,API Key的常用配置和dotenv代码实战,内容简洁易懂,兼顾理论认知与实际开发落地。
如果喜欢看视频学习的,可以看这个《7. 搞懂LangChain开发库,还有dotenv实战》,喜欢看文章的接着往下看。
1. LangChain项目工程
LangChain 项目采用职责清晰的模块化架构,核心模块定位与能力边界如下:
langchain-core:LangChain 全生态的核心基石,统一定义了生态内所有通用抽象接口、交互协议与核心数据结构,是所有上层组件与集成能力的底层依赖。
langchain-classic:历史版本兼容包,仅用于存量旧项目的平滑兼容与运维支撑,不推荐新项目引入使用。
langchain_v1:当前官方主推的生产级主用版本,核心聚焦大模型智能体(Agent)的全链路构建,封装了开箱即用的核心开发能力。
partners 集成库:由 LangChain 官方维护的第三方生态集成集合;其中 Anthropic、DeepSeek、OpenAI 等主流大模型的核心集成,均由 LangChain 官方与对应厂商联合开发维护,保障接口兼容性与长期稳定性。

其中,OpenAI API 已成为大模型服务领域的行业事实标准,目前绝大多数主流大模型厂商,均提供了与该标准完全兼容的 API 接口,可实现跨模型的低成本无缝切换。

2. 申请API Key
接入任意大模型平台的服务前,均需先在对应平台申请专属的 API 密钥(API Key),用于接口调用时的身份鉴权与调用额度管控。
针对个人开发者的开发测试场景,优先推荐阿里云百炼、火山引擎两大平台,二者均为个人开发者提供了充足的免费调用额度,可充分满足日常功能调试、原型验证的全流程需求。
进到阿里云百炼后台https://bailian.console.aliyun.com,点击【API Key】,进到创建API密钥的窗口。

找到右上角【创建API Key】按钮并点击,这时会弹出一个窗口。

直接点击【确定】按钮即可。

API Key创建好了,点击【复制】按钮并保存到本地文件(自己随意创建一个txt文件)。接着点击【关闭】按钮,之后点击【阿里云百炼logo】,回到主窗口。

在主窗口中,找到【免费额度】并进行点击。

为了避免大模型用量超额,你需要进行设置。找到右边【批量操作免费额度用完即停】按钮并进行点击。

在下拉菜单,点击【批量开启】选项。

操作不能停,还要按下【一键开启所有模型】按钮。

一开始时,我就开启过,现在是开启最近新增的模型,点击【开启免费额度用完即停】按钮。过一会就会全部开启完了。然后要使用的大模型名称并复制保存起来,建议选择最近的(比如qwen3.6-plus),旧版的到一定时间会下架的。接着点击顶部【文档】链接。

找到代码中base_url,复制https://dashscope.aliyuncs.com/compatible-mode/v1并保存起来。注意,这里的示例代码是LangChain旧版本的。

3. 创建工程并进行dotenv实战
用PyCharm创建一个项目工程之后,创建main.py文件,内容如下,这种直接把API_Key写在代码里的做法叫硬编码,它最大的问题是不灵活:换大模型要改代码,还容易泄露密钥。
OPENAI_API_KEY=”你的API Key” OPENAI_BASE_URL=https://dashscope.aliyuncs.com/compatible-mode/v1 MODEL=qwen3.6-plus
为保障调用凭证安全,可将API_KEY和BASE_UR配置至系统环境变量里。用户变量只对当前登录计算机的用户有效,系统变量对所有用户都有效。

配置之后,可以重启计算机生效,也可以通过如下方式生效。打开命令行窗口并运行如下命令进行刷新。接着需要退出命令行窗口,同时需要重启PyCharm。
set PATH=%PATH%
打开main.py文件,输入如下代码。os.getenv()函数,通过指定key获取环境变量的值,如果获取不到,会返回None。
import os import dotenv api_key=os.getenv("OPENAI_API_KEY") base_url=os.getenv("OPENAI_BASE_URL") print(api_key) print(base_url)
在开发产品时,常将用户自己配置的API密钥和URL放在.env文件里。新建.env文件,API密钥的值,加点不一样的。
OPENAI_API_KEY=“你的API密码-666” OPENAI_BASE_URL=“https://dashscope.aliyuncs.com/compatible-mode/v1” MODEL=“qwen3.6-plus”
读写.env文件,需要用pip安装dotenv开发包。
pip install dotenv
用import语法引进dotenv;load_dotenv(),默认加载当前目录下的.env文件。
import os from dotenv import load_dotenv load_dotenv() api_key = os.getenv("OPENAI_API_KEY") base_url = os.getenv("OPENAI_BASE_URL") print(api_key) print(base_url)
运行代码之后,你会发现,返回的还是系统环境变量里配置的值,这是因为同名下,系统环境变量优先级高。删掉系统环境变量里的OPENAI_API_KEY和OPENAI_BASE_URL。

打开命令行窗口并运行如下命令进行刷新。接着需要退出命令行窗口,同时需要重启PyCharm。
set PATH=%PATH%
打开main.py文件并运行之后,就会输出.env里的OPENAI_API_KEY和OPENAI_BASE_URL的值了。
你的API密码-666
https://dashscope.aliyuncs.com/compatible-mode/v1
4. dotenv应用实战
用PyCharm新建一个新的项目工程,用pip安装dotenv和pyside6。
pip install dotenv pyside6
在项目中,新建.env文件,如下是国内AI大模型厂家的API调用地址,在使用的时候,自己核对一下。
# ========== 阿里巴巴-通义千问 ========== QWEN_API_KEY='' QWEN_BASE_URL='https://dashscope.aliyuncs.com/compatible-mode/v1' QWEN_CONSOLE_URL='https://dashscope.console.aliyun.com/' QWEN_MODELS='qwen3.6-plus,qwen-3.5-plus,qwen-3.5-72b-instruct' # ========== 月之暗面 - Kimi ========== KIMI_API_KEY='' KIMI_BASE_URL='https://api.moonshot.cn/v1' KIMI_CONSOLE_URL='https://platform.moonshot.cn/' KIMI_MODELS='moonshot-v1-8k,moonshot-v1-128k,moonshot-v1-256k' # ========== MiniMax ========== MINIMAX_API_KEY='' MINIMAX_BASE_URL='https://api.minimax.chat/v1' MINIMAX_CONSOLE_URL='https://platform.minimax.chat/' MINIMAX_MODELS='abab-6.5-chat,abab-6.5-pro,minimax-m2.7,minimax-m2-her' # ========== 智谱AI-智谱清言 ========== ZHIPU_API_KEY='' ZHIPU_BASE_URL='https://open.bigmodel.cn/api/paas/v4' ZHIPU_CONSOLE_URL='https://open.bigmodel.cn/' ZHIPU_MODELS='glm-4.5-flash,glm-4.5-pro,glm-4-air' # ========== 字节跳动-豆包 ========== DOUBAO_API_KEY='' DOUBAO_BASE_URL='https://ark.cn-beijing.volces.com/api/v3' DOUBAO_CONSOLE_URL='https://console.volcengine.com/ark/' # 重要:豆包不直接使用模型名称,必须替换为火山方舟创建的推理接入点ID DOUBAO_MODELS='doubao-pro-32k,doubao-lite-32k,doubao-pro-128k' # ========== 百度-文心一言 ========== ERNIE_API_KEY='' ERNIE_BASE_URL='https://qianfan.baidubce.com/v2' ERNIE_CONSOLE_URL='https://console.bce.baidu.com/qianfan/' ERNIE_MODELS='ernie-4.0-turbo-8k,ernie-3.5-turbo-128k,ernie-speed-128k' # ========== 腾讯-混元 ========== HUNYUAN_API_KEY='' HUNYUAN_BASE_URL='https://api.hunyuan.cloud.tencent.com/v1' HUNYUAN_CONSOLE_URL='https://console.cloud.tencent.com/hunyuan/' HUNYUAN_MODELS='hunyuan-turbo,hunyuan-pro,hunyuan-lite' # ========== DeepSeek ========== DEEPSEEK_API_KEY='' DEEPSEEK_BASE_URL='https://api.deepseek.com/v1' DEEPSEEK_CONSOLE_URL='https://platform.deepseek.com/' DEEPSEEK_MODELS='deepseek-chat,deepseek-reasoner'
新建一个env_util.py文件,用来封装dotenv对.env文件的操作。
import os from dotenv import load_dotenv, set_key # @老陈说编程 哔哩哔哩、今日头条 class EnvUtil: def __init__(self, env_path=None): if env_path is None: self.env_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), ".env") else: self.env_path = env_path load_dotenv(self.env_path) def load_config(self, providers): config = {} for provider in providers: api_key = os.getenv(f"{provider}_API_KEY", "") base_url = os.getenv(f"{provider}_BASE_URL", "") console_url = os.getenv(f"{provider}_CONSOLE_URL", "") models_str = os.getenv(f"{provider}_MODELS", "") models = [m.strip() for m in models_str.split(",")] if models_str else [] config[provider] = { "api_key": api_key, "base_url": base_url, "console_url": console_url, "models": models } return config def save_config(self, config): if not os.path.exists(self.env_path): return for provider, config_item in config.items(): set_key(self.env_path, f"{provider}_API_KEY", config_item["api_key"]) set_key(self.env_path, f"{provider}_BASE_URL", config_item["base_url"]) set_key(self.env_path, f"{provider}_MODELS", ",".join(config_item["models"])) load_dotenv(self.env_path, override=True)
新建api_page.py文件,复制并粘贴如下代码,代码是用pyside6开发的,没有安装过开发包的话,会报错的。
import sys import os from env_util import EnvUtil from PySide6.QtCore import Qt from PySide6.QtWidgets import ( QWidget, QVBoxLayout, QHBoxLayout, QLabel, QLineEdit, QPushButton, QComboBox, QScrollArea, QFrame, QMessageBox, QApplication ) # @老陈说编程 哔哩哔哩、今日头条 class APIPage(QWidget): def __init__(self, parent=None): super().__init__(parent) self.env_util = EnvUtil() self.config = {} self.current_provider = None self.model_items = [] self.providers_list = ["", "QWEN", "DOUBAO", "KIMI", "MINIMAX", "ZHIPU", "ERNIE", "HUNYUAN", "DEEPSEEK"] self.load_env() layout = QVBoxLayout(self) layout.setContentsMargins(0, 0, 0, 0) layout.setSpacing(0) content = QWidget() content_layout = QVBoxLayout(content) content_layout.setContentsMargins(40, 20, 40, 20) content_layout.setSpacing(20) title = QLabel("大模型配置中心") title.setStyleSheet("font-size: 20px; font-weight: bold; color: #333;") provider_group = QVBoxLayout() provider_label = QLabel("选择厂商") provider_label.setStyleSheet("font-weight: 600; color: #444; margin-bottom: 6px;") self.provider_combo = QComboBox() self.provider_combo.addItem("-- 请选择 --") self.provider_combo.addItem("阿里巴巴 (通义千问)") self.provider_combo.addItem("字节跳动 (豆包)") self.provider_combo.addItem("月之暗面 (Kimi)") self.provider_combo.addItem("MiniMax") self.provider_combo.addItem("智谱AI (智谱清言)") self.provider_combo.addItem("百度 (文心一言)") self.provider_combo.addItem("腾讯 (混元)") self.provider_combo.addItem("DeepSeek") self.provider_combo.setStyleSheet(""" QComboBox { background-color: #ffffff; color: #333333; border: 1px solid #d1d5db; border-radius: 4px; padding: 6px 10px; } """) self.provider_combo.setMinimumHeight(32) self.provider_combo.currentIndexChanged.connect(self.handle_provider_change) provider_group.addWidget(provider_label) provider_group.addWidget(self.provider_combo) self.config_area = QWidget() self.config_area.setVisible(False) config_layout = QVBoxLayout(self.config_area) config_layout.setContentsMargins(0, 0, 0, 0) config_layout.setSpacing(16) api_key_group = QVBoxLayout() api_key_header = QHBoxLayout() api_key_label = QLabel("API Key") api_key_label.setStyleSheet("font-weight: 600; color: #444;") self.get_api_key_link = QLabel('<a href="#" style="color: #0066cc; text-decoration: none;">获取API密钥</a>') self.get_api_key_link.setStyleSheet("color: #0066cc;") self.get_api_key_link.setOpenExternalLinks(False) self.get_api_key_link.linkActivated.connect(self.open_api_key_url) api_key_header.addWidget(api_key_label) api_key_header.addStretch() api_key_header.addWidget(self.get_api_key_link) self.api_key_input = QLineEdit() self.api_key_input.setPlaceholderText("请输入 API Key") self.api_key_input.setEchoMode(QLineEdit.Password) self.api_key_input.setStyleSheet(""" QLineEdit { background-color: #ffffff; border: 1px solid #d1d5db; border-radius: 4px; padding: 6px 10px; font-size: 14px; } """) api_key_group.addLayout(api_key_header) api_key_group.addWidget(self.api_key_input) base_url_group = QVBoxLayout() base_url_label = QLabel("Base URL") base_url_label.setStyleSheet("font-weight: 600; color: #444; margin-bottom: 6px;") self.base_url_input = QLineEdit() self.base_url_input.setPlaceholderText("请输入 Base URL") self.base_url_input.setStyleSheet(""" QLineEdit { background-color: #ffffff; border: 1px solid #d1d5db; border-radius: 4px; padding: 6px 10px; font-size: 14px; } """) base_url_group.addWidget(base_url_label) base_url_group.addWidget(self.base_url_input) model_section = QFrame() model_section.setStyleSheet("border-top: 1px dashed #eee; padding-top: 20px;") model_section_layout = QVBoxLayout(model_section) model_section_layout.setContentsMargins(0, 0, 0, 0) model_header = QHBoxLayout() model_label = QLabel("大模型名称列表") model_label.setStyleSheet("font-weight: 600; color: #444;") self.add_model_btn = QPushButton("+ 添加模型") self.add_model_btn.setFixedWidth(100) self.add_model_btn.setStyleSheet(""" QPushButton { background-color: #e6f7ff; color: #0066cc; border: 1px solid #91d5ff; border-radius: 4px; padding: 6px 14px; font-size: 14px; } QPushButton:hover { background-color: #bae7ff; } """) self.add_model_btn.clicked.connect(self.add_model) model_header.addWidget(model_label) model_header.addStretch() model_header.addWidget(self.add_model_btn) self.model_list_container = QWidget() self.model_list_layout = QVBoxLayout(self.model_list_container) self.model_list_layout.setContentsMargins(0, 8, 0, 8) self.model_list_layout.setSpacing(8) model_section_layout.addLayout(model_header) model_section_layout.addWidget(self.model_list_container) self.save_btn = QPushButton("保 存 配 置") self.save_btn.setStyleSheet(""" QPushButton { background-color: #1890ff; color: white; border: none; border-radius: 4px; padding: 8px 12px; font-size: 14px; } QPushButton:hover { background-color: #40a9ff; } """) self.save_btn.clicked.connect(self.save_config) config_layout.addLayout(base_url_group) config_layout.addLayout(api_key_group) config_layout.addWidget(model_section) config_layout.addWidget(self.save_btn) content_layout.addWidget(title) content_layout.addLayout(provider_group) content_layout.addWidget(self.config_area) content_layout.addStretch() scroll = QScrollArea() scroll.setWidget(content) scroll.setWidgetResizable(True) scroll.setFrameShape(QFrame.NoFrame) layout.addWidget(scroll) def load_env(self): providers = ["QWEN", "DOUBAO", "KIMI", "MINIMAX", "ZHIPU", "ERNIE", "HUNYUAN", "DEEPSEEK"] self.config = self.env_util.load_config(providers) def save_env(self): self.env_util.save_config(self.config) def open_api_key_url(self): if self.current_provider: config = self.config.get(self.current_provider, {}) url = config.get("console_url", "") if url: import webbrowser webbrowser.open(url) def handle_provider_change(self, index): provider = self.providers_list[index] if not provider: self.config_area.setVisible(False) self.current_provider = None return self.current_provider = provider self.load_config_to_form(provider) self.config_area.setVisible(True) def load_config_to_form(self, provider): config = self.config.get(provider, {}) self.api_key_input.setText(config.get("api_key", "")) self.base_url_input.setText(config.get("base_url", "")) self.render_model_list(config.get("models", [])) def render_model_list(self, models): for i in reversed(range(self.model_list_layout.count())): item = self.model_list_layout.takeAt(i) if item.widget(): item.widget().deleteLater() self.model_items = [] for model_name in models: model_item = ModelItem(model_name) model_item.delete_btn.clicked.connect(lambda checked, mi=model_item: self.remove_model(mi)) self.model_list_layout.addWidget(model_item) self.model_items.append(model_item) def add_model(self): if not self.current_provider: return model_item = ModelItem("") model_item.delete_btn.clicked.connect(lambda checked, mi=model_item: self.remove_model(mi)) self.model_list_layout.addWidget(model_item) self.model_items.append(model_item) def remove_model(self, model_item): if not self.current_provider: return index = self.model_items.index(model_item) model_item.deleteLater() self.model_items.pop(index) def save_config(self): if not self.current_provider: return models = [] for item in self.model_items: models.append(item.input.text()) self.config[self.current_provider] = { "api_key": self.api_key_input.text(), "base_url": self.base_url_input.text(), "models": models } self.save_env() QMessageBox.information(self, "成功", "配置已保存!") class ModelItem(QWidget): def __init__(self, model_name="", parent=None): super().__init__(parent) layout = QHBoxLayout(self) layout.setContentsMargins(0, 0, 0, 0) layout.setSpacing(8) self.input = QLineEdit() self.input.setText(model_name) self.input.setPlaceholderText("输入模型名称") self.input.setStyleSheet(""" QLineEdit { background-color: #ffffff; border: 1px solid #d1d5db; border-radius: 4px; padding: 6px 10px; font-size: 14px; } """) self.delete_btn = QPushButton("删除") self.delete_btn.setFixedWidth(60) self.delete_btn.setStyleSheet(""" QPushButton { background-color: #fff1f0; color: #ff4d4f; border: 1px solid #ffa39e; border-radius: 4px; padding: 6px 14px; font-size: 14px; } QPushButton:hover { background-color: #ffccc7; } """) layout.addWidget(self.input, 1) layout.addWidget(self.delete_btn) if __name__ == "__main__": app = QApplication(sys.argv) api_page = APIPage() api_page.setWindowTitle("大模型配置中心") api_page.resize(800, 700) api_page.show() sys.exit(app.exec())
运行api_page.py之后,可以通过dotenv库,对.env文件进行增删改和读取操作。


浙公网安备 33010602011771号