jjzjj

php - 从 php 调用 MSSQL 存储过程

coder 2024-05-02 原文

整个早上我们都被这个难住了。

我有一个处理一些记录的 MSSQL 存储过程。每条记录都会生成一封包含动态内容的电子邮件,这是从另一个存储过程中填充的。

所以第一个存储过程有一个游标,处理每一行都会导致调用另一个存储过程,它本身有一个要循环的游标。第一个存储过程没有输出参数或返回值等,而第二个使用输出参数将字段返回给第一个过程。

这在 Datagrip 中运行良好。

使用 PDO(或使用其他驱动程序)从 php 调用它无法完全运行。它会生成一小批记录,然后停止(往往是 5、9、13 或 45 - 随着我们尝试不同的解决方案而发生变化)。

我们已经设法让它现在使用 PDOStatement::nextRowset 运行。我们使用第一个存储过程的查询,然后使用 while ( $stmt->nextRowset() ) ; 循环遍历(不存在/不需要的)行集。

这行得通。但是由于第一个存储过程没有返回任何东西(只是那个 pdo 似乎想要处理某种内部结果集)这看起来很脏而且效率很低。

有替代方案吗?可能是要传递给 pdo 的参数,还是存储过程中的设置?

下面有一些简化的代码来展示事物是如何相互作用的。

PHP 调用脚本。

<?php
$emailRepository = new EmailRepository(hostname, port, dbname, username, password);

$ret = $emailRepository->sendRenewalEmails();

class EmailRepository
{

    public function __construct($hostname, $port, $dbname, $username, $password)
    {
        $this->hostname = $hostname;
        $this->port = $port;
        $this->dbname = $dbname;
        $this->username = $username;
        $this->password = $password;
        $this->connect();
    }

    private function connect()
    {
        try {
            $this->db = new PDO ("sqlsrv:Server=$this->hostname,$this->port;Database=$this->dbname", $this->username, $this->password);
            $this->db->setAttribute( PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION );
        } catch (PDOException $e) {
            var_dump("Failed to get DB handle: " . $e->getMessage());
        }
    }

    public function sendRenewalEmails()
    {
      try {
        $stmt = $this->db->prepare("EXEC membership_queue_renewal_emails");
        $stmt->execute();
        do {
          echo '>';
        } while($stmt->nextRowset());
        return true;
      } catch (Exception $e) {
        echo $e->getMessage();
      }
    }
}

第一个存储过程(大幅缩减)

CREATE PROCEDURE [dbo].[queue_renewal_emails]
AS
BEGIN

    DECLARE @curr_member_cursor     CURSOR;
    DECLARE @curr_club_cursor       CURSOR;
    DECLARE @g_personid             INT;

    DECLARE @g_emailTemplateId      INT;
    DECLARE @g_email_subject        VARCHAR(200);
    DECLARE @g_email_html           VARCHAR(max);
    DECLARE @g_email_plain          VARCHAR(max);

    DECLARE @g_personEmail          VARCHAR(128);

    SET @curr_club_cursor = CURSOR
    LOCAL STATIC READ_ONLY FORWARD_ONLY
    FOR
    SELECT  DISTINCT
            bgemailTemplate.bgemte_name,
            bgemailTemplate.bgemte_emailtemplateid,
            vpersonpe.pers_emailaddress,
            vpersonpe.pers_personId,
    FROM  company               WITH(NOLOCK)
    INNER JOIN complink             WITH(NOLOCK)    ON complink.clli_companyid = complink.Comp_CompanyId
    AND complink.clli_Deleted is null
    INNER JOIN vpersonpe            WITH(NOLOCK)    ON vpersonpe.pers_personId = complink.clli_personId
    INNER JOIN bgemailTemplate      WITH(NOLOCK)    ON bgemailTemplate.bgemte_Deleted IS NULL
    WHERE vPersonPE.pers_deleted                    IS NULL
    AND   company.comp_deleted                      IS NULL
    AND   vPersonPE.pers_parentid                   IS NULL
    AND   vpersonpe.pers_status                     NOT IN ('Cancelled','Expired','Suspended','Awaiting Approval','Declined','On hold','Revoked','Expelled');

    -- loop through each course
    OPEN @curr_club_cursor;
    FETCH NEXT FROM @curr_club_cursor INTO @g_email_subject, @g_emailTemplateId, @g_personEmail, @g_personid;
    WHILE @@fetch_status = 0
    BEGIN
        EXEC dbo.populateEmail @g_emailtemplateid   /* Email template id */,
                                @g_email_plain OUTPUT /* Plain text email to have the placeholders replaced */,
                                @g_email_subject OUTPUT,
                                @g_personid ;
        FETCH NEXT FROM @curr_club_cursor INTO @g_email_subject, @g_emailTemplateId, @g_personEmail, @g_personid;
    END
    CLOSE @curr_club_cursor
    DEALLOCATE @curr_club_cursor

END
go

第二个存储过程(大大减少)。

CREATE PROCEDURE [dbo].[populateEmail]
    @p_emailtemplateid          INT,
    @p_email_text               VARCHAR(max) OUTPUT,
    @p_email_subject            VARCHAR(200) OUTPUT,
    @p_person_id                INT
AS
BEGIN

    SET NOCOUNT ON;

    -- CURSORs
    DECLARE @curr_field_cursor                  CURSOR;
    DECLARE @g_email_plain                      VARCHAR(MAX) = '';
    DECLARE @g_email_subject                    VARCHAR(200) = '';

    DECLARE @g_emte_emailtemplateid             INT;
    DECLARE @g_EmailPlaceholderId               INT;
    DECLARE @g_place_holder                     VARCHAR(128);
    DECLARE @g_source_column                    VARCHAR(128);
    DECLARE @g_prev_source_query_num            INT;
    DECLARE @g_source_query_num                 INT;

    -- Variables to read results into from each query
    DECLARE @g_q11_comp_name                    VARCHAR(180);
    DECLARE @g_q11_comp_website                 VARCHAR(300);
    DECLARE @g_q11_comp_pers_salutation         VARCHAR(30);
    DECLARE @g_q11_comp_pers_firstname          VARCHAR(90);
    DECLARE @g_q11_comp_pers_lastname           VARCHAR(120);
    -- Start processing

    SET @p_email_text = '';

    SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;

    SET @curr_field_cursor = CURSOR
    LOCAL STATIC READ_ONLY FORWARD_ONLY
    FOR
    SELECT a.emte_emailtemplateid,
            b.emtl_EmailPlaceholderId,
            c.empl_PlaceHolder,
            c.empl_SourceQueryNum,
            c.empl_SourceColumn,
            a.emte_plaintextemail,
            a.emte_subject
    FROM EmailTemplate a with (nolock)
    LEFT OUTER JOIN  EmailTemplateLink b with (nolock)
    ON a.emte_emailtemplateid = b.emtl_EmailTemplateId
    LEFT OUTER JOIN  EmailPlaceholder c with (nolock)
    ON b.emtl_EmailPlaceholderId = c.empl_EmailPlaceholderID
    WHERE a.emte_emailtemplateid = @p_emailtemplateid
    ORDER BY c.empl_SourceQueryNum;

    -- Loop through each required place holder for the passed email template.

    SET @g_prev_source_query_num = 0;

    OPEN @curr_field_cursor
    FETCH NEXT FROM @curr_field_cursor INTO @g_emte_emailtemplateid, @g_EmailPlaceholderId, @g_place_holder, @g_source_query_num, @g_source_column, @g_email_plain, @g_email_subject;
    WHILE @@fetch_status = 0
    BEGIN

        IF @g_prev_source_query_num = 0
        BEGIN
            SET @p_email_text = @g_email_plain;
            SET @p_email_subject = @g_email_subject;
        END;

        IF @g_source_query_num = 11
        BEGIN
            IF @g_prev_source_query_num != @g_source_query_num
            BEGIN
                SELECT @g_q11_comp_name = comp_name,
                        @g_q11_comp_website = comp_website,
                        @g_q11_comp_pers_salutation = Pers_Salutation,
                        @g_q11_comp_pers_firstname = pers_firstname,
                        @g_q11_comp_pers_lastname = pers_lastname
                FROM company with (nolock)
                LEFT OUTER JOIN vPerson with (nolock) ON company.Comp_PrimaryPersonId = vPerson.Pers_PersonId
                LEFT OUTER JOIN address with (nolock) ON company.Comp_PrimaryAddressId = address.Addr_AddressId
                WHERE company.Comp_CompanyId        = @p_person_id;
            END;

            IF @g_source_column = 'comp_name'
            BEGIN
                SET @p_email_text       = REPLACE(@p_email_text, @g_place_holder, COALESCE(@g_q11_comp_name, ''));
                SET @p_email_subject    = REPLACE(@p_email_subject, @g_place_holder, COALESCE(@g_q11_comp_name, ''));
            END;
            ELSE IF @g_source_column = 'comp_website'
            BEGIN
                SET @p_email_text       = REPLACE(@p_email_text, @g_place_holder, COALESCE(@g_q11_comp_website, ''));
                SET @p_email_subject    = REPLACE(@p_email_subject, @g_place_holder, COALESCE(@g_q11_comp_website, ''));
            END;
            ELSE IF @g_source_column = 'comp_primary_person_firstname'
            BEGIN
                SET @p_email_text       = REPLACE(@p_email_text, @g_place_holder, COALESCE(@g_q11_comp_pers_salutation, ''));
                SET @p_email_subject    = REPLACE(@p_email_subject, @g_place_holder, COALESCE(@g_q11_comp_pers_salutation, ''));
            END;
            ELSE IF @g_source_column = 'comp_primary_person_salutation'
            BEGIN
                SET @p_email_text       = REPLACE(@p_email_text, @g_place_holder, COALESCE(@g_q11_comp_pers_firstname, ''));
                SET @p_email_subject    = REPLACE(@p_email_subject, @g_place_holder, COALESCE(@g_q11_comp_pers_firstname, ''));
            END;
            ELSE IF @g_source_column = 'comp_primary_person_lastname'
            BEGIN
                SET @p_email_text       = REPLACE(@p_email_text, @g_place_holder, COALESCE(@g_q11_comp_pers_lastname, ''));
                SET @p_email_subject    = REPLACE(@p_email_subject, @g_place_holder, COALESCE(@g_q11_comp_pers_lastname, ''));
            END;
        END;

        SET @g_prev_source_query_num = @g_source_query_num;
        FETCH NEXT FROM @curr_field_cursor INTO @g_emte_emailtemplateid, @g_EmailPlaceholderId, @g_place_holder, @g_source_query_num, @g_source_column, @g_email_plain, @g_email_subject;
    END;

    CLOSE @curr_field_cursor
    DEALLOCATE @curr_field_cursor

END
go

最佳答案

as the first stored proc isn't returning anything ... is there an alternative to while ( $stmt->nextRowset() ) ;

可能是行计数消息导致客户端看到空行集。添加 SET NOCOUNT ON 作为每个存储过程的第一行。

关于php - 从 php 调用 MSSQL 存储过程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57360872/

有关php - 从 php 调用 MSSQL 存储过程的更多相关文章

  1. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  2. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  3. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  4. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  5. ruby - 调用其他方法的 TDD 方法的正确方法 - 2

    我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent

  6. ruby - Rack:如何将 URL 存储为变量? - 2

    我正在编写一个简单的静态Rack应用程序。查看下面的config.ru代码:useRack::Static,:urls=>["/elements","/img","/pages","/users","/css","/js"],:root=>"archive"map'/'dorunProc.new{|env|[200,{'Content-Type'=>'text/html','Cache-Control'=>'public,max-age=6400'},File.open('archive/splash.html',File::RDONLY)]}endmap'/pages/search.

  7. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

  8. C51单片机——实现用独立按键控制LED亮灭(调用函数篇) - 2

    说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时

  9. ruby - 如何找到调用当前方法的方法 - 2

    如何找到调用此方法的位置?defto_xml(options={})binding.pryoptions=options.to_hifoptions&&options.respond_to?(:to_h)serializable_hash(options).to_xml(options)end 最佳答案 键入caller。这将返回当前调用堆栈。文档:Kernel#caller.例子[0]%rspecspec10/16|===================================================62=====

  10. ruby-on-rails - 使用 HTTParty 的非常基本的 Rails 4.1 API 调用 - 2

    Rails相对较新。我正在尝试调用一个API,它应该向我返回一个唯一的URL。我的应用程序中捆绑了HTTParty。我已经创建了一个UniqueNumberController,并且我已经阅读了几个HTTParty指南,直到我想要什么,但也许我只是有点迷路,真的不知道该怎么做。基本上,我需要做的就是调用API,获取它返回的URL,然后将该URL插入到用户的数据库中。谁能给我指出正确的方向或与我分享一些代码? 最佳答案 假设API为JSON格式并返回如下数据:{"url":"http://example.com/unique-url"

随机推荐