扩展dll和规则dll貌似没区别啊?

wzuxian2012 2013-04-08 09:11:02
别贴网上的资料

我看过很多遍了,也试验过。

感觉2者的区别(目前发现的只有这么一点)
扩展dll 的导出函数里不能够创建对话框,只要创建立即报错。

规则dll则可以。


另外:2者都可以导出mfc类的派生类,

网上说:扩展dll则可以,我试验了一下,发现规则dll也可以导出。

...全文
206 19 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
19 条回复
切换为时间正序
请发表友善的回复…
发表回复
lm_whales 2013-05-30
  • 打赏
  • 举报
回复
MFC 类的派生类或者组合类
lm_whales 2013-05-30
  • 打赏
  • 举报
回复
扩展dll导出C++ 类,并且是MFC(dll版)的扩展类(即导出某个动态库中的 MFC 类的派生类); 这种DLL是MFC类库(dll版)的扩展,所以称为扩展dll C 使用这种 DLL应该是很吃力的,除非你根本没有做什么扩展(没有导出某个MFC 类的派生类)。 规则DLL只是使用MFC类库(dll版或者静态库),导出函数是C形式的。
看不见的裂痕 2013-04-22
  • 打赏
  • 举报
回复
扩展DLL只能用在MFC里面,针对MFC生成的一些对话框程序。 其他C#和Java是没办法调用的。 规则DLL都是可以的。
wzuxian2012 2013-04-22
  • 打赏
  • 举报
回复
人呢? 人呢? 人呢?
看不见的裂痕 2013-04-22
  • 打赏
  • 举报
回复
我说带窗体的,你C语言调用带窗体的扩展DLL能成功? 你C语言最多调用extern “C”的非窗体化得DLL,那是可以的。
wzuxian2012 2013-04-22
  • 打赏
  • 举报
回复
引用 15 楼 tiger9991 的回复:
扩展DLL只能用在MFC里面,针对MFC生成的一些对话框程序。 其他C#和Java是没办法调用的。 规则DLL都是可以的。
你试验过了吗? 我做过实验,发现,c语言也可以调用啊,所以才来问人 如果你会c#那更好做实验了. 呵呵
linlinlinxi007 2013-04-10
  • 打赏
  • 举报
回复
规则MFC DLL,可以被各种语言调用,只要你的接口参数标准 扩展MFC DLL,需要MFC程序调用,这个是一个限制的地方 MFC的扩展DLL和MFC的常规(regular)DLL的区别在于:前者可以导出MFC类,而后者不能,只能导出自己写的C++类
  • 打赏
  • 举报
回复
引用 2 楼 wzuxian2012 的回复:
引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 你的这个,是不是太草率了, 我举个极端的例子来 推翻: 无论是规则的还是扩展的, 都可以提供 一般的函数吧。 比如这样的函数 int fun() { return 0; } 就这么简单的一个接口函数……
这有啥极端的,创建mfc的对话框需要一些初始化操作,不管在dll还是在exe中。 但在扩展dll中,都没做这些初始化操作,出错很正常。 除了 AfxSetResourceHandle 外,还需要一个CWinApp类型的theApp,当然还有其他操作。 具体参考 VC\atlmfc\src\mfc\dllmodul.cpp 这些操作你都做的话创建对话框自然没有问题。
wzuxian2012 2013-04-10
  • 打赏
  • 举报
回复
wzuxian2012 2013-04-10
  • 打赏
  • 举报
回复
引用 12 楼 akirya 的回复:
引用 11 楼 akirya 的回复:引用 10 楼 wzuxian2012 的回复:引用 9 楼 akirya 的回复:引用 8 楼 wzuxian2012 的回复:引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 C/C++ code?123456789101112131415161……
谁知道你里面用了啥。 可以提供一个demo让我区分吗? 邮箱: liulin2017@163.com 多谢
  • 打赏
  • 举报
回复
引用 11 楼 akirya 的回复:
引用 10 楼 wzuxian2012 的回复:引用 9 楼 akirya 的回复:引用 8 楼 wzuxian2012 的回复:引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 C/C++ code?1234567891011121314151617181920212223242526……
向导只是生成一部分固定代码的 ,你完全可以删掉自己写啊。 你提供的代码编译后 就只能看到一个C可以调用的函数。谁知道你里面用了啥。 C能调用就是标准的dll。不能调用就是扩展的。
  • 打赏
  • 举报
回复
引用 10 楼 wzuxian2012 的回复:
引用 9 楼 akirya 的回复:引用 8 楼 wzuxian2012 的回复:引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 C/C++ code?1234567891011121314151617181920212223242526272829303132333435363738……
向导只是生成一部分固定代码的 ,你完全可以删掉自己写啊。 你提供的代码 C编译后 就只能看到一个C可以调用的函数。谁知道你里面用了啥。 C能调用就是标准的dll。不能调用就是扩展的。
wzuxian2012 2013-04-10
  • 打赏
  • 举报
回复
引用 9 楼 akirya 的回复:
引用 8 楼 wzuxian2012 的回复:引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 C/C++ code?1234567891011121314151617181920212223242526272829303132333435363738394041424344454647……
那到底 根据什么区分? 我是用类向导区分的, 类向导 如果不准确,那么 还有什么可以区分呢?
  • 打赏
  • 举报
回复
引用 8 楼 wzuxian2012 的回复:
引用 1 楼 akirya 的回复:C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。 C/C++ code?12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152扩展dll代码: exte……
你还没明白 "不是根据工程类型区分的" 的意思
wzuxian2012 2013-04-10
  • 打赏
  • 举报
回复
引用 1 楼 akirya 的回复:
C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。
扩展dll代码:

extern "C" int fun();


extern "C" int fun()
{
	return 11110;
}

测试工程代码:

//main.c

#include<windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <Shlwapi.h>
#include <string.h>
#include <assert.h>



int main()
{
	HINSTANCE hInst=NULL;
	int testVal=-1;
	CHAR strModuleFileName[MAX_PATH]={0};
	CHAR strDllName[MAX_PATH]="test9.dll";
	CHAR strDllComName[MAX_PATH]={0};
	char* pstr=NULL;
	typedef int(* pFun)();
	pFun myfun;
	GetModuleFileName(GetModuleHandle(NULL),strModuleFileName,MAX_PATH);
	pstr=strrchr(strModuleFileName,'\\');
	assert(pstr);
	*pstr='\0';
	PathCombine(strDllComName,strModuleFileName,strDllName);
	hInst=LoadLibrary(strDllComName);
	if(!hInst)
	{
		printf("加载dll失败,错误码为:%d\r\n",GetLastError());
		return 0;
	}
	myfun=(pFun)GetProcAddress(hInst,"fun");
	testVal=myfun();
	printf("%d",testVal);
	return 0;
}


自己验证吧, 无论扩展还是规则dll,均可以调用。


wzuxian2012 2013-04-09
  • 打赏
  • 举报
回复
引用 3 楼 huangbeyond 的回复:
对于扩展dll,确实只要在接口函数里创建一个对话框,立即导致程序崩溃。 ----------------------------------------------- 这个问题,要使用AfxSetResourceHandle来解决。 详细的,参考MSDN。
你要自己试试,比如 一个函数创建一个模式对话框就行 用扩展dll 你试试把, 看崩溃不 你加或不加都崩溃。 不信的话,我等你消息。
huangbeyond 2013-04-08
  • 打赏
  • 举报
回复
对于扩展dll,确实只要在接口函数里创建一个对话框,立即导致程序崩溃。 ----------------------------------------------- 这个问题,要使用AfxSetResourceHandle来解决。 详细的,参考MSDN。
wzuxian2012 2013-04-08
  • 打赏
  • 举报
回复
引用 1 楼 akirya 的回复:
C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。
你的这个,是不是太草率了, 我举个极端的例子来 推翻: 无论是规则的还是扩展的, 都可以提供 一般的函数吧。 比如这样的函数 int fun() { return 0; } 就这么简单的一个接口函数, 2种dll都可以 实现, exe都可以成功的调用。 我在主贴里的那个 创建对话框的例子,我是亲自测试过的。 对于扩展dll,确实只要在接口函数里创建一个对话框,立即导致程序崩溃。 还有没有其他答案?
  • 打赏
  • 举报
回复
C可以调用规则dll,如果不能调用,那么就是扩展dll。 不是根据工程类型区分的。
最近帮朋友改一个小东西的时候才发现的,微软专门为PHP出了个SQL Server的扩展(当然是Windows版本的),用了一下发现还是很好用的,对于Windows下使用php开发SQL Server应用来说,这个扩展有利于兼容性以及充分利用SQL Server特性。 PHP本身有个php_mssql.dll扩展用来连接Sql server,但是貌似这个dll只是用来连接低版本 Sql server的(2000以下版本),在Sql server 2005以后版本则根本无法使用mssql_connect连接到数据库。 先到微软网站下载 SQL Server Driver for PHP :http://www.microsoft.com/downloads/details.aspx?familyid=CCDF728B-1EA0-48A8-A84A-5052214CAAD9&displaylang=en , 这是一个自解压的EXE文件,解压缩后你会得到这么几个文件: 其中的52、53表示就是php的5.2.x和5.3.x 版本,你必须选择跟你php版本相匹配的; vc6、vc9表示的是编译这个dll所使用的vc++编译器版本,基本上大多数时候,选vc6的; nts、ts表示的是否是 ThreadSafe的,得根据安装的php版本来选择,如果你不确定,就两种情况分别尝试一下好了; 我服务器上安装的是 ThreadSafe版本的php-5.2.x,所以选择的是php_sqlsrv_52_nts_vc6.dll,把这个文件拷贝到php的ext目录,比如: 1 C:\php\ext 然后修改php.ini在适当的地方加上一行: 1 extension=php_sqlsrv_52_nts_vc6.dll 然后重启web服务器就可以了。 运行一个phpinfo看看: 另外需要注意的是,如果使用这个扩展连接Sql server 2005以上版本的sql server,你还需要在机器上先安装 SQL Server Native Client :http://download.microsoft.com/download/0/E/6/0E67502A-22B4-4C47-92D3-0D223F117190/sqlncli.msi 这个扩展为php新增了一系列sqlsrv_开头的函数, 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 sqlsrv_begin_transaction sqlsrv_cancel sqlsrv_client_info sqlsrv_close sqlsrv_commit sqlsrv_configure sqlsrv_connect sqlsrv_errors sqlsrv_execute sqlsrv_fetch sqlsrv_fetch_array sqlsrv_fetch_object sqlsrv_fetch_metadata sqlsrv_free_stmt sqlsrv_get_config sqlsrv_get_field sqlsrv_has_rows sqlsrv_next_result sqlsrv_num_fields sqlsrv_num_rows sqlsrv_prepare sqlsrv_query sqlsrv_rollback sqlsrv_rows_affected sqlsrv_send_stream_data sqlsrv_server_info 详细的使用文档可以在之前下载的自解压文件里面找到。 摘抄一段文档里面的例子: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 /*===================================================================== This file is part of a Microsoft SQL Server Shared Source Application. Copyright (C) Microsoft Corporation. All rights reserved. THIS CODE AND INFORMATION ARE PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE. ======================================================= */ $serverName = "(local)"; $connectionInfo = array( "Database"=>"AdventureWorks"); /* Connect using Windows Authentication. */ $conn = sqlsrv_connect( $serverName, $connectionInfo); if( $conn === false ) { echo "Could not connect.\n"; die( print_r( sqlsrv_errors(), true)); } /* Get the product picture for a given product ID. */ $tsql = "SELECT LargePhoto FROM Production.ProductPhoto AS p JOIN Production.ProductProductPhoto AS q ON p.ProductPhotoID = q.ProductPhotoID WHERE ProductID = ?"; $params = array($_REQUEST['productId']); /* Execute the query. */ $stmt = sqlsrv_query($conn, $tsql, $params); if( $stmt === false ) { echo "Error in statement execution.
"; die( print_r( sqlsrv_errors(), true)); } /* Retrieve the image as a binary stream. */ if ( sqlsrv_fetch( $stmt ) ) { $image = sqlsrv_get_field( $stmt, 0, SQLSRV_PHPTYPE_STREAM(SQLSRV_ENC_BINARY)); fpassthru($image); } else { echo "Error in retrieving data.
"; die(print_r( sqlsrv_errors(), true)); } /* Free the statement and connectin resources. */ sqlsrv_free_stmt( $stmt ); sqlsrv_close( $conn ); 是不是很简单?跟平时用的mysql、mssql函数差不多的。
Url重写系列篇: ----主讲:天涯浪子 本讲将通过实例比较ASP.NET下的三种典型URL重写方案——ISAPI重写(使用开源组件IIRF),ASP.NET2.0内置的urlMappings和基于自定义HTTPModule的URL重写(使用NBear.Web中的UrlRewriteModule实现),并探讨URL重写中可能遇到的陷阱及处理办法。 需要手动为UrlRewriteWebSite目录添加一个到http://localhost/UrlRewriteWebSite的同名虚拟目录,允许匿名访问,并设置目录默认页为default.aspx。 另外,为了启用IIRF的URL重写支持,需要将UrlRewriteWebSite/bin目录下的IsapiRewrite4.dll添加为IIS默认网站的ISAPI过滤器。[相应资料打压缩包] 重写原理: 用户(A)----->系统运行(C) 真实地址 用真实地址运行 用户(A)----->转换器(B)----->系统运行(C) 假地址 将用户假地址 用真实地址运行 转换为真实地址 重写规则 注意,我们的演示程序中将混合使用三种方式的URL重写,因此,需要为三种实现分别设置一些URL重写规则: 1、IIRF,对于IIRF,对应于IsapiRewrite4.dll,在相同的目录会有一个IsapiRewrite4.ini文件,除了默认的一些设置,我们在文件末尾添加了几条自定义规则如下: # Custom RewriteRules RewriteRule ^/UrlRewriteWebSite/test(.*).aspx /UrlRewriteWebSite/Default.aspx?page=$1 RewriteRule ^/UrlRewriteWebSite/folder/(.*).aspx /UrlRewriteWebSite/Default.aspx?folder=$1 RewriteRule ^/UrlRewriteWebSite/folder/? /UrlRewriteWebSite/Default.aspx?folder=default 熟悉正则表达式的朋友应该很容易理解上面这三条规则规则一将形如testXXX.aspx这样的页面访问,重写为Default.aspx?page=XXX这样的页面; 规则二将形如folder/XXX.aspx的路径,重写为Default.aspx?folder=XXX这样的页面; 规则三将不带任何文件的folder目录的访问,重写为Default.aspx?folder=default这样的页面。 2、urlMappings是ASP.NET2.0内置支持的URL重写配置块,它应该包含在web.config的配置块中。但是,这个内置的URL重写支持不支持正则表达式,因而只能用来实现一对一的路径和页面的重写。urlMappings的配置内容包含在下面的Web.config文件中。 3、NBear.Web.Modules.UrlRewriteModule则是NBear中实现的一个基于HTTPModule的URL重写实现,它允许使用正则表达式来描述重写规则。 注意,代码中包含了urlMappings配置和用于NBear.Web.Modules.UrlRewriteModule重写规则。为了比较着几种重写方案,正则表达式基本上是和前面的IIRF定义中的规则类似的。 页面测试 定义完这些重写规则,我们就可以试着在页面中使用它们了。例如,如果我们写一个测试页面如下: Default.aspx Default.aspx.cs public partial class _Default : System.Web.UI.Page { protected void Page_Load(object sender, EventArgs e) { Response.Write(Request.Path + "?" + Request.ServerVariables["QUERY_STRING"]); } } 注意,Default.aspx页面会输出当前呈现的实际页面及其QueryString参数。 运行该页面,分别点击页面中的链接,我们会看到,貌似所有的URL重写一切正常。但是,当试着点击页面中的按钮,我们马上会发现,页面postback后,浏览器地址栏中的链接变成了那个被重写后的地址,而不是,原来显示于地址栏的虚拟地址了。这是一个严重的不一致,道理我点击页面的按钮,在有跳转到其他页面的情况下,地址栏显示另一个页面地址,不是吗? 要解决这个问题,我们只需要为form添加一个onsubmit事件处理如下:
添加该事件处理,就能在页面postback提交之前,重置页面的地址。 为前面的页面添加onsubmit之后,我们发现,postback不再会改变地址栏地址显示了。 注意:你确认试过点击最后一组链接中的buildin default page和section default page了吗? 你会发现,这两个链接根本不能显示。为什么呢?为什么类似的folder default page可以正常显示,而另两个不能显示呢? 回到前面的规则定义部分,我们就能发现,folder default page使用的是由IIRF这个ISAPI定义的规则,而另两个则使用的是内置于ASP.NET2.0的HTTPModule的重写规则(本质上,urlMappings也是使用HTTPModule来实现重写的,所以,除了不支持正则表达式之外,它也包含自定义HTTPModule方式实现的所有缺点)。 在IIS的ISAPI层面,是可以截获所有的页面请求的,哪怕指定的页面、目录根本不存在。但是,ASP.NET解析器则只有在对页面的请求被IIS转发过来时,才能处理。我们知道,IIS可以忽略对链接的虚拟目录是否存在的检测,但是,却无法检测非ASP.NET支持的文件扩展名的链接(我们固然可以在IIS中将所有类型的扩展名都映射到ASP.NET解析器,但是,如果我们有设置IIS的权限,为什么还要用性能更低,限制更多的ASP.NET方式的URL重写,而不使用基于ISAPI方式的重写呢?)。所以,为了让这两个不能显示的页面能正常显示,一方面,我们要在IIS中设置默认页,如default.aspx,另一方面,需要让IIS对某个不带aspx扩展名的链接,如这里只包含某个目录的名称的链接转发到默认页。 要做到这一点,我们需要在我们的应用程序中,为buildin和section分别将两个对应的目录,并且,在目录中创建两个空的default.aspx页面。尽管这样的default.aspx页面实际上永远不会被真正执行,但是有他们的帮助,就能让IIS顺利地将页面请求转发至ASP.NET解析器,从而,使得基于HTTPModule的URL重写规则,被执行。 好了,创建这两个目录及default.aspx文件,我们就能修复该问题了。 接下来我们先做点测试: 我们来对页面上的链接反复点击点击,folder page -> section page -> folder page -> section page...等等,打住,看到浏览器地址栏发生了什么吗?这不是恐怖活动,但是。。。也差不多了。我们看到我们可爱的地址,变成了...folder/section/folder/section...aspx。 想想是为什么呢?看看我们的aspx文件。。。我想你一定想到了。对了,都是相对路径惹的祸! 我们可爱的的相对路径一顶是同学们最常使用的,但是,浏览器在处理相对路径时,是以浏览器上接受的url地址为基础进行计算的,也就是说,如果当前的地址为folder/1.aspx,那么,很显然,./section/2.aspx这个页面,对应的自然是folder/section/2.aspx了,问题就出在这儿了!有URL重写时,不会有这样的情况出现。但是URL重写,并且,将一个带假目录的虚拟地址重写到一个不带假目录的页面时,由于浏览器客户端和服务端此时的当前页面计算方法是不同的,就会发生相对路径的匹配错误问题!真实很严重的问题啊! 解决的办法,只有使用绝对路径!但是,我们当然不会傻到对每个链接直接使用绝对路径的,呵呵:) 将Default.aspx中的所有相对路径都使用Page.ResolveUrl进行包装如下,在输出页面时就将地址转换为绝对路径,就能解决这个恐怖的相对路径陷阱了。当然,也别忘了加上onsubmit事件处理代码: Default.aspx 好了,再试一试点击页面上的链接和按钮,多点几次 谢谢………… 参考资料: 引言 让我们花点时间来看一下网站上的一些 URL。您是否发现一些类似于 http://yoursite.com/info/dispEmployeeInfo.aspx?EmpID=459-099&type=summary 的 URL?或者,您可能将一系列网页从一个目录或网站移动到另一个目录或网站,结果导致已将旧 URL 用作书签的访问者断开链接。在本文中,我们将了解如何通过将 http://yoursite.com/info/dispEmployeeInfo.aspx?EmpID=459-099&type=summary 替换为类似于 http://yoursite.com/people/sales/chuck.smith 的网址,使用 URL 重写将那些冗长的 URL 缩写为富有意义且容易记忆的 URL。我们还将了解如何将 URL 重写用于创建智能 404 错误。 URL 重写是截取传入 Web 请求并自动将请求重定向到其他资源的过程。执行 URL 重写时,通常会检查被请求的 URL,并基于 URL 的值将请求重定向到其他 URL。例如,在进行网站重组而将 /people/ 目录下的所有网页移动到 /info/employees/ 目录中时,您可能希望使用 URL 重写来检查 Web 请求是否指向了 /people/ 目录中的文件。如果请求指向 /people/ 目录中的文件,您可能希望自动将请求重定向到 /info/employees/ 目录中的同一文件。 使用传统的 ASP,应用 URL 重写的唯一方法是编写 ISAPI 筛选器,或者购买提供 URL 重写功能的第三方产品。但是,使用 Microsoft® ASP.NET,您可以通过很多方法来轻松地创建您自己的 URL 重写软件。本文讨论了可供 ASP.NET 开发人员实现 URL 重写的各种技术,然后讨论了 URL 重写的一些实际使用情况。在深入讨论 URL 重写的技术细节之前,让我们先看一些可以使用 URL 重写的日常情景。 URL 重写的常见用法 创建数据驱动的 ASP.NET 网站时,通常会产生一个单个的网页,该网页基于查询字符串参数显示数据库数据的子集。例如,在设计电子商务站点时,您的任务之一便是允许用户浏览待售产品。为此,您可以创建一个名为 displayCategory.aspx 的页面,该页面将显示给定类别的产品。可以通过查询字符串参数来指定要查看的该类别的产品。也就是说,如果用户要浏览待售的 Widget 产品,并且所有 Widget 产品的 CategoryID 均为 5,则用户可以访问以下网址:http://yousite.com/displayCategory.aspx?CategoryID=5。 创建具有此类 URL 的网站有两点不足:首先,从最终用户的角度考虑,URL http://yousite.com/displayCategory.aspx?CategoryID=5 比较杂乱。可用性专家 Jakob Neilsen 建议遵循以下标准来选择 URL: • 简短。 • 易于键入。 • 可以看出站点的结构。 • “可删节”,允许用户通过删除 URL 的组成部分来浏览站点。 我还要增加一条标准,即,URL 应该便于记忆。URL http://yousite.com/displayCategory.aspx?CategoryID=5 不符合 Neilsen 的任何标准,也不容易记住。要求用户键入查询字符串值将使 URL 的键入变得非常困难,并且只有了解查询字符串参数的用途及其名称/值对结构的富有经验的 Web 开发人员才能够对 URL 进行“删节”。 较好的方法是允许使用切合实际且容易记忆的 URL,如 http://yoursite.com/products/Widgets。只要看一眼 URL,您便可以推断出将要显示的内容 -- 有关 Widget 的信息。此 URL 也很容易记住和共享。我可以告诉我的同事“请查看 yoursite.com/products/Widgets,”,她可能无需再次问我 URL 是什么即可打开该页面。(尝试一下,您只需说出“Amazon.com 页面”即可!)此 URL 还将显示出来,并且应该是“可删节”的。也就是说,如果用户删去 URL 的末端,键入 http://yoursite.com/products,他们应该看到所有产品的列表,或者至少应该看到他们可以查看的所有类别的产品列表。 注意:要获得“可删节”URL 的最好示例,可考虑使用由许多 blog 引擎生成的 URL。要查看 2004 年 1 月 28 日的帖子,用户可以访问诸如 http://someblog.com/2004/01/28 的 URL。如果该 URL 被删节为 http://someblog.com/2004/01,用户将看到 2004 年 1 月的所有帖子。将该 URL 进一步删节为 http://someblog.com/2004 将显示 2004 年的所有帖子。 除了简化 URL 之外,URL 重写还经常用于处理网站重组,以免导致大量链接断开或书签过期。 请求到达 IIS 时将会发生什么情况 在正式研究 URL 如何实现重写之前,应首先了解 Microsoft® Internet Information Services (IIS) 如何处理传入请求,这一点非常重要。当请求到达 IIS Web 服务器时,IIS 检查被请求文件的扩展名以确定如何处理该请求。IIS 可以自行处理请求(如 HTML 页面、图像以及其他静态内容),或者将请求路由到 ISAPI 扩展。(ISAPI 扩展是一个处理传入 Web 请求的非托管编译类。其任务是生成被请求资源的内容。) 例如,当传入针对 Info.asp 网页的请求时,IIS 会将此消息路由到 asp.dll ISAPI 扩展。然后,该 ISAPI 扩展将加载被请求的 ASP 页面,执行该页面,并将所呈现的 HTML 返回给 IIS,然后,IIS 将该 HTML 发送回请求客户端。对于 ASP.NET 页面,IIS 会将此消息路由到 aspnet_isapi.dll ISAPI 扩展。然后,aspnet_isapi.dll ISAPI 扩展将处理操作传递给托管的 ASP.NET 辅助进程,该辅助程序将处理请求,并返回 ASP.NET 网页的呈现 HTML。 您可以自定义 IIS,以指定扩展名与 ISAPI 扩展的映射关系。图 1 显示了 Internet Information Services 管理工具的“应用程序配置”对话框。请注意,与 ASP.NET 有关的扩展名(.aspx、ascx、config、asmx、rem、cs、vb 及其他)均已映射到 aspnet_isapi.dll ISAPI 扩展。 图 1. 已配置的文件扩展名映射 讨论 IIS 如何管理传入请求稍稍超出了本文范围。但是可以在 Michele Leroux Bustamante 的文章 Inside IIS and ASP.NET 中找到对此内容的深入讨论。ASP.NET 引擎仅处理那些扩展名已明确映射至 IIS 中的 aspnet_isapi.dll 的传入 Web 请求,了解这一点非常重要。 使用 ISAPI 筛选器检查请求 IIS 除了可以将传入 Web 请求的文件扩展名映射到相应的 ISAPI 扩展之外,还将执行许多其他任务。例如,IIS 将尝试对发出请求的用户进行身份验证,并确定通过身份验证的用户是否有权限访问被请求的文件。在处理请求的有效期内,IIS 将经历几个状态。在每个状态下,IIS 都将引发可以使用 ISAPI 筛选器以编程方式进行处理的事件。 与 ISAPI 扩展一样,ISAPI 筛选器是在 Web 服务器上安装的非托管代码块。ISAPI 扩展被设计为可以响应针对特定文件类型的请求。另一方面,ISAPI 筛选器还包含可以对 IIS 引发的事件进行响应的代码。ISAPI 筛选器可以截取甚至修改传入和传出的数据。ISAPI 筛选器可以应用于很多方面,包括: • 身份验证和授权。 • 记录和监视。 • HTTP 压缩。 • URL 重写。 虽然 ISAPI 筛选器可用于执行 URL 重写,但本文将讨论如何使用 ASP.NET 实现 URL 重写。不过,我们将对使用 ISAPI 筛选器与使用 ASP.NET 中的技术实现 URL 重写进行权衡。 请求进入 ASP.NET 引擎时将会发生什么情况 在 ASP.NET 之前,需要使用 ISAPI 筛选器来实现 IIS Web 服务器上的 URL 重写。由于 ASP.NET 引擎与 IIS 非常相似,因此可以使用 ASP.NET 进行 URL 重写。存在相似之处的原因在于 ASP.NET 引擎可以实现以下功能: • 在处理请求时可以引发事件。 • 允许任意数量的 HTTP 模块处理所引发的事件,这与 IIS 的 ISAPI 筛选器相似。 • 将呈现被请求资源这项任务委托给 HTTP 处理程序,该处理程序与 IIS 的 ISAPI 扩展相似。 与 IIS 一样,ASP.NET 引擎在请求的有效期内将会触发事件,通过发信号来表示其处理过程从一个状态改变为了另一个状态。例如,当 ASP.NET 引擎首次响应请求时,BeginRequest 事件将被触发。接下来触发的是 AuthenticateRequest 事件,该事件在已建立用户标识时出现。(此外,还有大量的其他事件:AuthorizeRequest、ResolveRequestCache 和 EndRequest,等等。这些事件属于 System.Web.HttpApplication 类;有关详细信息,请参阅位于以下网址的技术文档:HttpApplication Class Overview。) 正如上一部分所讨论的,可以创建 ISAPI 筛选器以响应 IIS 引发的事件。同样,ASP.NET 提供了 HTTP 模块,该模块可以响应由 ASP.NET 引擎引发的事件。可以将 ASP.NET Web 应用程序配置为具有多个 HTTP 模块。对于由 ASP.NET 引擎处理的每个请求,将初始化每个已配置的 HTTP 模块,并允许将事件处理程序绑定到处理请求期间所引发的事件。请注意,对每个请求均使用了许多内置 HTTP 模块。其中的一个内置 HTTP 模块是 FormsAuthenticationModule,该模块首先检查是否使用了窗体身份验证,如果使用,将检查是否对用户进行了身份验证。如果有使用,会自动将用户重定向到指定的登录页面。 如上所述,通过使用 IIS,传入请求将最终发送给 ISAPI 扩展,而 ISAPI 扩展的任务是返回特定请求的数据。例如,在请求传统的 ASP 网页时,IIS 将请求传递给 asp.dll ISAPI 扩展,该扩展的任务是返回被请求的 ASP 页面的 HTML 标记。ASP.NET 引擎使用相似的方法。初始化 HTTP 模块后,ASP.NET 引擎的下一项任务是确定应由哪个 HTTP 处理程序来处理请求。 所有通过 ASP.NET 引擎传递的请求最终都将到达 HTTP 处理程序或 HTTP 处理程序工厂(HTTP 处理程序工厂仅返回 HTTP 处理程序的实例,然后使用该实例来处理请求)。最终的 HTTP 处理程序将返回响应,即呈现被请求的资源。此响应将被发送回 IIS,然后 IIS 将响应返回给提出请求的用户。 ASP.NET 包括许多内置的 HTTP 处理程序。例如,PageHandlerFactory 用于呈现 ASP.NET 网页。WebServiceHandlerFactory 用于呈现 ASP.NET Web 服务的响应 SOAP 信封。TraceHandler 将向 trace.axd 呈现请求的 HTML 标记。 图 2 描述了如何处理对 ASP.NET 资源的请求。首先,IIS 接收到请求,并将请求调度给 aspnet_isapi.dll。接下来,ASP.NET 引擎对已配置的 HTTP 模块进行初始化。最后将调用正确的 HTTP 处理程序,并呈现被请求的资源,将所生成的标记返回给 IIS 和请求客户端。 图 2. IIS 和 ASP.NET 正在处理请求

16,548

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • AIGC Browser
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧