【Pyhton4Delpi】学习笔记(二)安装验证篇

D12环境下安装P4D。

一、下载 Python4Delphi(下称P4D):

下载地址:https://github.com/pyscripter/python4delphi

下载或者克隆P4D到指定的目录,例如:MDS_New,目录结构如下,P4D就是克隆下来的控件全部目录。

 

二、安装P4D

        首先关闭正在运行的Delphi IDE!                

2.1 自动安装:

        在下载的目录InstallMDS_New\P4D\Install)中,点击 MultiInstaller.exe 进行安装。

提示:

        在安装目录中的Setup.ini中默认定义了安装的目录是在P4D目录下,这一步体现在2.1.2选择的目录之下。例如:2.1.2步骤中选择 D:\MDS_New 则试剂安装在 D:\MDS_New\P4D目录中,如果需要更改这个目录,则需要修改Setup.ini文件。

        强烈建议不要修改,这个安装程序是有问题的,默认安装才可以!

[Package - Python4Delphi]
Name=Python4Delphi
Folder=P4D
SearchPath="Source"
2.1.1 选择需要安装的包,默认可以全部安装

2.1.2 选择需要安装的文件夹:

注意:

        勾选 Compile packages and install on IDE 选项,这样就会安装完成后,在控件面板上显示已经安装好的P4D控件。

        另外注意,这里选择目录是上一级目录,实际默认安装在这一级目录下的P4D目录中。

2.1.3 安装完成,点击:Finish 即可

2.2 手动安装

自动安装和手动安装是完全独立的安装方式,但只需要成功执行一次就可以。对于手动安装,比如已经将P4D克隆到了 D:\MDS_New\P4D 目录中。

2.2.1 打开Delphi IDE
2.2.2 设置搜索路径,增加如下地址
  • D:\MDS_New\P4D\Source
  • D:\MDS_New\P4D\Source\vcl
  • D:\MDS_New\P4D\Source\fmx

2.2.3 打开 D:\MDS_New\P4D\Packages\Delphi\P4DComponentSuite.groupproj 工程组

对于10.4+以上版本就选择10.4+,否则选择10.3-。 选择 Build All

2.2.4 选择  dclPythondclPythonVcl 和 dclPythonFmx 运行时包进行安装

三、查看安装是否成功

        打开Delphi 12,不同的Delphi版本都是可以的,从10.4开始,都是一样的,10.3以下的版本有区别。

        如果看到控件面板上包含P4D的控件,这样就算安装成功了!

四、配置验证运行环境,检测是否真正可用

由于P4D是使用Python和Delphi两种语言,那么正常理解,Python的环境是必须要安装的,但事实上,完整安装了Python的环境当然是没有问题的,但是,如果我们给客户开发了一个软件,还需要开户安装Python开发环境,这就太奢侈了,所以也还有个不需要完整安装Python开发环境的路。

对于完整安装,本文就不多说,对于只需要一个极简安装环境的,说明如下,其实Python的运行环境就是一个DLL和一些lib库,只要这两个有了,就可以运行起来了。

以pythpon38为例:

这个DLLs是如果有Python代码引用到三方的DLL,那么三方的DLL就在这个目录中,如果没有引用到,就可以不需要这个目录。特别注意X32和X64是不同的版本,32位的Python就只能用32位的Delphi,必须对应使用。

做一个Delphi 的程序,然后在写Python代码,就可以尝试运行了,举例:

1. Delphi 程序源代码
unit uMainForm;interfaceusesWinapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls, Vcl.Buttons,Vcl.Samples.Spin, Vcl.ExtCtrls,system.Math,System.Threading,System.DateUtils, PythonEngine, Vcl.PythonGUIInputOutput,VarPyth;//PythonEngine, Vcl.PythonGUIInputOutput;typeTForm1 = class(TForm)Splitter1: TSplitter;Panel1: TPanel;Memo1: TMemo;SpeedButton1: TSpeedButton;PythonGUIInputOutput1: TPythonGUIInputOutput;PythonEngine1: TPythonEngine;SpeedButton2: TSpeedButton;PythonModule1: TPythonModule;SpeedButton_LoadDLL: TSpeedButton;OpenDialog1: TOpenDialog;Edit1: TEdit;SpeedButton_UnLoadDLL: TSpeedButton;SpeedButton3: TSpeedButton;Memo2: TMemo;procedure PythonEngine1BeforeLoad(Sender: TObject);procedure SpeedButton1Click(Sender: TObject);procedure SpeedButton2Click(Sender: TObject);procedure PythonModule1Events0Execute(Sender: TObject; PSelf,Args: PPyObject; var Result: PPyObject);procedure SpeedButton_LoadDLLClick(Sender: TObject);procedure SpeedButton_UnLoadDLLClick(Sender: TObject);procedure FormResize(Sender: TObject);procedure PythonModule1Events1Execute(Sender: TObject; PSelf,Args: PPyObject; var Result: PPyObject);procedure SpeedButton3Click(Sender: TObject);procedure FormCreate(Sender: TObject);private{ Private declarations }function IsPrime(n : integer) : Boolean;function CountPrimes(MaxN : Integer) : Integer;public{ Public declarations }end;varForm1: TForm1;implementation{$R *.dfm}{ TForm1 }function TForm1.CountPrimes(MaxN: Integer): Integer;
varCount : integer;
beginTParallel.For(2,MaxN,procedure(i: integer)beginif IsPrime(i) thenAtomicIncrement(Count);end);Result := Count;
end;procedure TForm1.FormCreate(Sender: TObject);
begin{$IFDEF WIN32}{$IFDEF DEBUG}Self.Caption := 'P4D Demo示例 ( Win32 ) Debug';{$ELSE}Self.Caption := 'P4D Demo示例 ( Win32 ) Release';{$ENDIF}{$ELSE}{$IFDEF DEBUG}Self.Caption := 'P4D Demo示例 ( Win64 ) Debug';{$ELSE}Self.Caption := 'P4D Demo示例 ( Win64 ) Release';{$ENDIF}{$ENDIF}
end;procedure TForm1.FormResize(Sender: TObject);
begin//Memo2.Height := Trunc((Self.ClientHeight - Panel1.Height) * 0.7);
end;function TForm1.IsPrime(n: integer): Boolean;
beginif n <= 1 then Exit(False);var q := Floor(Sqrt(n));for var i := 2 to q doif (n mod i = 0) then Exit(False);Exit(True);
end;procedure TForm1.PythonEngine1BeforeLoad(Sender: TObject);
begin//PythonEngine1.SetPythonHome(PythonEngine1.DllPath);   //必须设置这一步
end;procedure TForm1.PythonModule1Events0Execute(Sender: TObject; PSelf,Args: PPyObject; var Result: PPyObject);
varN : integer;
beginwith GetPythonEngine doif PyArg_ParseTuple(Args,'i:delphi_is_prime',@N) <> 0 thenbeginif IsPrime(N) thenResult := PPyObject(Py_True)elseResult := PPyObject(Py_False);Py_INCREF(Result);endelseResult := nil;
end;procedure TForm1.PythonModule1Events1Execute(Sender: TObject; PSelf,Args: PPyObject; var Result: PPyObject);
varN : integer;
beginwith GetPythonEngine doif PyArg_ParseTuple(Args,'i:delphi_count_primes',@N) <> 0 thenbeginResult := PyLong_FromLong(CountPrimes(N));Py_INCREF(Result);endelseResult := nil;
end;procedure TForm1.SpeedButton1Click(Sender: TObject);
beginif not PythonEngine1.Initialized thenbeginMemo1.Lines.Add('请先装载Python DLL');Exit;end;//GetPythonEngine.ExecStrings(SynEdit1.Lines);PythonEngine1.ExecStrings(Memo2.Lines);
end;procedure TForm1.SpeedButton2Click(Sender: TObject);
varaTask: ITask;T    : TDateTime;path : string;PyModule: Variant;
beginif not PythonEngine1.Initialized thenbeginMemo1.Lines.Add('请先装载Python DLL');Exit;end;aTask := TTask.Create(procedurebeginT := Now;TThread.Synchronize(TThread.Current,procedurebeginMemo1.Lines.add('线程运行开始时间: ' + FormatDateTime('hh:mm:ss zzz',T));end);PythonEngine1.ExecStrings(Memo2.Lines);
//      path := 'D:\MDS_New\Demo\camera';
//      PyModule := SysModule;
//      PyModule.path.append(path);
//      PyModule := Import('run');
//      PyModule.start();TThread.Synchronize(TThread.Current,procedurebeginMemo1.Lines.add('线程运行结束时间: ' + FormatDateTime('hh:mm:ss zzz',Now) +  #13#10'线程执行花费时长: ' + (MilliSecondsBetween(Now,T) / 1000).ToString + ' 秒');end);end);aTask.Start;
end;procedure TForm1.SpeedButton3Click(Sender: TObject);
varPyModule: Variant;path : string;
begin{path := 'D:\MDS_New\Demo\camera';PyModule := SysModule;PyModule.path.append(path);PyModule := Import('run');PyModule.start();}//PyModule := Import('hello');//PyModule.SayHello('ABC 你好吗');
end;procedure TForm1.SpeedButton_LoadDLLClick(Sender: TObject);
beginif not OpenDialog1.Execute then Exit;Edit1.Text := OpenDialog1.FileName;//进行运行环境装载PythonEngine1.DllPath := ExtractFilePath(OpenDialog1.FileName);//   'Y:\Documents\sensorwu\Python4Delphi\x86py38';PythonEngine1.DllName := ExtractFileName(OpenDialog1.FileName);PythonEngine1.SetPythonHome(PythonEngine1.DllPath);   //必须设置这一步PythonEngine1.LoadDll;SpeedButton_LoadDLL.Enabled := False;SpeedButton_UnLoadDLL.Enabled := True;
end;procedure TForm1.SpeedButton_UnLoadDLLClick(Sender: TObject);
beginPythonEngine1.UnloadDll;SpeedButton_LoadDLL.Enabled := True;SpeedButton_UnLoadDLL.Enabled := False;
end;end.
2. Python 源代码
#-------------------------------------------------------------------------------
# Name:        module1
# Purpose:     判断素数测试
#
# Author:      wuxihong
#
# Created:     01-02-2024
# Copyright:   (c) wuxihong 2024
# Licence:     <your licence>
#-------------------------------------------------------------------------------#-------- 以下这个包是从delphi 中导入的,打开该引用,同时打开下面的delphi_is_prime实现调用Delphi函数 -------
# ---------- delphi_count_primes 完全使用Delphi的并行计算,速度提高非常多
#from delphi_module import delphi_is_prime
#from delphi_module import delphi_count_primes
#-------------------------------------------
import math
from timeit import Timerdef is_prime(n):if n < 1:return Falseq = math.floor(math.sqrt(n))for i in range(2,q+1):if (n % i == 0):return Falsereturn Truedef count_primes(max_n):res = 0for i in range(2,max_n + 1):#----------- 以下这句是引用 Delphi 包中的函数 delphi_is_prime,打开该语句,屏蔽下面一句 if#if delphi_is_prime(i):#-------------------------------------if is_prime(i):res += 1return resdef test():max_n = 100000print(f'Number of primes between 0 and {max_n} = {count_primes(max_n)}')#----------- 以下这句是引用 Delphi 包中的函数 delphi_count_primes,打开该语句,屏蔽上面一句,即可体会到Delphi的并行运算能力#print(f'Number of primes between 0 and {max_n} = {delphi_count_primes(max_n)}')def main():print(f'Elapsed Time: {Timer(stmt=test).timeit(1)} secs')if __name__ == '__main__':main()
3. 说明

这是一个计算素数个数的演示程序,可以完全通过Python来计算,也可以在Python中调用Delphi的函数来计算,演示了如何设置Python的运行环境。

五、源代码及运行环境(38)下载

1. Delphi 源代码(内涵Python代码)
2. Python38运行环境 

如有任何问题,可以留言交流,因为P4D的资料实在太少,所以需要大家多交流。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/458084.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Msql-数据库死锁

实验案例 CREATE TABLE t1_deadlock ( id int(11) NOT NULL, name varchar(100) DEFAULT NULL, age int(11) NOT NULL, address varchar(255) DEFAULT NULL, PRIMARY KEY (id), KEY idx_age (age) USING BTREE, KEY idx_name (name) USING BTREE ) ENGINEInnoDB DEFAULT CHARS…

C语言之找单身狗

个人主页&#xff08;找往期文章包括但不限于本期文章中不懂的知识点&#xff09;&#xff1a; 我要学编程(ಥ_ಥ)-CSDN博客 题目&#xff1a; 在一个整型数组中&#xff0c;只有一个数字出现一次&#xff0c;其他数组都是成对出现的&#xff0c;请找出那个只出现一次的数字。…

基于微信小程序的医保行政执法案件管理系统

本系统设计的是一个医保行政执法的网站&#xff0c;此网站使用户实现了不需出门就可以在手机或电脑前进行网上查询需求信息等。 用户在注册登陆后&#xff0c;在客户端可以实现&#xff1b;案件信息、结案归档、我的等。然而管理员则可以在服务端直接管理&#xff1b;个人中心、…

H12-821_28

28.如图所示,在一个纯IPv6环境中,若想实现PC1与PC2之间的通信,下列哪组地址可以分别配置在PC1与P2上 A.2001:FDC:1/64 2001:FDC::2/64 B.2001:FDC::1/64 2001:FDC1::2/64 C.FD00:1AC0:872E::1/64 FD00:2BE1:2320::1/64 D.FE80::1/64 FE80::2/64 答案&#xff1a;B 注释&…

少儿编程考级:智慧启迪还是智商税?

在当前科技日新月异的时代背景下&#xff0c;少儿编程教育日益受到家长和社会的广泛关注。与此同时&#xff0c;各类少儿编程考级应运而生&#xff0c;引发了公众对于其价值和意义的深度探讨。一部分人认为这是对孩子逻辑思维与创新能力的有效锻炼&#xff0c;是智慧启迪的重要…

深入解析Elasticsearch的内存架构与管理

在当今数据驱动的世界中&#xff0c;高效、快速地处理和搜索大量数据成为了许多应用的核心需求。Elasticsearch&#xff0c;作为一款功能强大的开源搜索和分析引擎&#xff0c;通过其独特的内存架构和管理策略&#xff0c;实现了对大规模数据集的快速索引和查询。本文将深入解析…

推理系统学习笔记

一些学习资料 最近对MLsys比较感兴趣&#xff0c;遂找些资料开始学习一下 https://fazzie-key.cool/2023/02/21/MLsys/https://qiankunli.github.io/2023/12/16/llm_inference.htmlhttps://dlsyscourse.orghttps://github.com/chenzomi12/DeepLearningSystem/tree/main/04Infe…

最详细PE文件格式讲解(完结篇)

前提情要&#xff1a;各位小伙伴是否已经熟练掌握前半部分知识点&#xff0c;各位可以找一下相关练习&#xff0c;或者等待我下面会发布的代码实践进行练习&#xff0c;如果一切ok&#xff0c;那我们将开始下半部分学习&#xff08;good&#xff09;。 如果还没有看&#xff0c…

【学习笔记】TypeScript学习笔记1 --TypeScript中的类型

文章目录 TS总的变量类型References TS总的变量类型 备注&#xff1a; 如果一个变量设置为了any 类型之后相当于变量关闭了TS的类型检测 let d: any; d 10; d hello;//unknown表示的是未知类型&#xff0c;实际是上一个安全的any,unknown类型的变量不能直接赋值给其他变量le…

docker部署docker运维工具

简介 主要功能:管理容器,管理镜像,管理容器网络 安装 拉取镜像 docker pull joinsunsoft/docker.ui:1.0.1 启动容器 docker run -d --name docker.ui --restart always -v /var/run/docker.sock:/var/run/docker.sock -p 10039:8999 joinsunsoft/docker.ui:1.0.1 使用 打…

汇集全球50+供应链领域企业专家,创新论坛带来最新趋势和实践

过去的几年中&#xff0c;随着世界范围内经济、社会和政治上的巨大变化&#xff0c;供应链管理已成为企业和经济成功的关键因素。面对不断增长的全球挑战&#xff0c;包括经济波动、技术变革、政治不确定性&#xff0c;以及环境可持续性的压力&#xff0c;构建一个创新、高效且…

[leetcode] 29. 两数相除

文章目录 题目描述解题方法倍增java代码复杂度分析 题目描述 给你两个整数&#xff0c;被除数 dividend 和除数 divisor。将两数相除&#xff0c;要求 不使用 乘法、除法和取余运算。 整数除法应该向零截断&#xff0c;也就是截去&#xff08;truncate&#xff09;其小数部分…