日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

获取真实IP

發(fā)布時間:2023/12/10 编程问答 34 豆豆
生活随笔 收集整理的這篇文章主要介紹了 获取真实IP 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
真正的取真實IP地址及利弊(轉(zhuǎn)自百度空間)
?

?

目前網(wǎng)上流行的所謂“取真實IP地址”的方法,都有bug,沒有考慮到多層透明代理的情況。

多數(shù)代碼類似:

string IpAddress = (HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]!=null? ???????????? && HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"] !=String.Empty) ???????????? ?HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]
???????????? :HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];

事實上,上面的代碼只試用與用戶只使用了1層代理,如果用戶有2層,3層HTTP_X_FORWARDED_FOR 的值是:“本機真實IP,1層代理IP,2層代理IP,.....” ,如果這個時候你的數(shù)據(jù)中保存IP字段的長度很小(15個字節(jié)),數(shù)據(jù)庫就報錯了。

實際應用中,因為使用多層透明代理的情況比較少,所以這種用戶并不多。

其他應用情況,現(xiàn)在越來越多的網(wǎng)站使用了代理加速方式,比如 新浪、SOHU的新聞 都使用Squid做代理方式,利用多臺服務器分流。Squid本身類似透明代理,會發(fā)送“HTTP_X_FORWARDED_FOR” ,HTTP_X_FORWARDED_FOR 中包括客戶的IP地址,如果此時客戶已經(jīng)使用了一層透明代理,那么程序取的 “HTTP_X_FORWARDED_FOR” 就包括兩個IP地址。(我遇到過3個IP地址的情況,4個的未遇到過)

所以取“真正”IP地址的方式,還應該判斷?? “HTTP_X_FORWARDED_FOR”?? 中是否有“,”逗號,或者長度是否超長(超過15字節(jié) xxx.xxx.xxx.xxx)。

所以代碼應該如下:

/** <summary> /// 取得客戶端真實IP。如果有代理則取第一個非內(nèi)網(wǎng)地址 /// </summary> public static string IPAddress { ???? get ???? { ???????? string result = String.Empty;
???????? result = HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"];
???????? if(result!=null&&result!= String.Empty) ???????? {
???????????? //可能有代理 ???????????? if(result.IndexOf(".")==-1)???? //沒有“.”肯定是非IPv4格式 ???????????????? result = null; ???????????? else
???????????? { ???????????????? if(result.IndexOf(",")!=-1)
???????????????? { ???????????????????? //有“,”,估計多個代理。取第一個不是內(nèi)網(wǎng)的IP。
???????????????????? result = result.Replace(" ","").Replace("'","");
???????????????????? string[] temparyip = result.Split(",;".ToCharArray());
???????????????????? for(int i=0;i<temparyip.Length;i++)
???????????????????? { ???????????????????????? if( Text.IsIPAddress(temparyip[i]) ???????????????????????????? && temparyip[i].Substring(0,3)!="10." ???????????????????????????? && temparyip[i].Substring(0,7)!="192.168" ???????????????????????????? && temparyip[i].Substring(0,7)!="172.16.") ???????????????????????? { ???????????????????????????? return temparyip[i];???? //找到不是內(nèi)網(wǎng)的地址
???????????????????????? } ???????????????????? } ???????????????? }
???????????????? else if(Text.IsIPAddress(result)) //代理即是IP格式
???????????????????? return result; ???????????????? else
???????????????????? result = null;???? //代理中的內(nèi)容 非IP,取IP ???????????? }
???????? }
???????? string IpAddress = (HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]!=null && HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"] !=String.Empty)?HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]:HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];
????????
???????? if (null == result || result == String.Empty)
???????????? result = HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];
???? ???????? if (result == null || result == String.Empty)
???????????? result = HttpContext.Current.Request.UserHostAddress;
???????? return result; ???? } }

取“HTTP_X_FORWARDED_FOR” 的弊端。

HTTP_X_FORWARDED_FOR 是HTTP協(xié)議中頭的一部分,不影響TCP的通訊。也就是說實際上客戶端可以發(fā)送任意內(nèi)容的 HTTP_X_FORWARDED_FOR,以就是偽造IP。最簡單的是WEB程序的IP記錄,本來是要記錄真實IP的,反而被“黑客”欺騙。當你的應用程序記錄客戶的訪問IP、拒絕或允許部分IP的訪問、錯誤日志 都會出錯,甚至誤殺。

因此必要的安全日志應該記錄 完整的 “HTTP_X_FORWARDED_FOR” (至少給數(shù)據(jù)庫中的字段分配 3*15+2 個字節(jié),以記錄至少3個IP) 和 “REMOTE_ADDR”。對 HTTP_X_FORWARDED_FOR 的IP格式檢查也是不可少的。
附:(Text是我自定義的一個類,IsIPAddress是其中的一個判斷是否是IP地址格式的方法) #region bool IsIPAddress(str1) 判斷是否是IP格式 /** <summary> /// 判斷是否是IP地址格式 0.0.0.0 /// </summary> /// <param name="str1">待判斷的IP地址</param> /// <returns>true or false</returns> public static bool IsIPAddress(string str1) { ???? if(str1==null||str1==string.Empty||str1.Length<7||str1.Length>15) return false;

???? string regformat = @"^\d{1,3}[\.]\d{1,3}[\.]\d{1,3}[\.]\d{1,3}$";

???? Regex regex = new Regex(regformat,RegexOptions.IgnoreCase ); ???? return regex.IsMatch(str1); } #endregion

原文地址:http://blog.csdn.net/kingwkb/archive/2006/04/05/651127.aspx

獲取客戶端IP:
private string GetClientIP() ???? { ????? string result = HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]; ????? if (null == result || result == String.Empty) ????? { ?????? result = HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"]; ????? }
????? if (null == result || result == String.Empty) ????? { ?????? result = HttpContext.Current.Request.UserHostAddress; ????? } ????? return result; ???? }
獲取MAC地址:
[DllImport("Iphlpapi.dll")] ???? private static extern int SendARP(Int32 dest,Int32 host,ref Int64 mac,ref Int32 length);
???? [DllImport("Ws2_32.dll")] ???? private static extern Int32 inet_addr(string ip);
???? private void Page_Load(object sender, System.EventArgs e) ???? { ????? // 在此處放置用戶代碼以初始化頁面 ????? try ????? { ?????? string userip=Request.UserHostAddress; ?????? string strClientIP = Request.UserHostAddress.ToString().Trim(); ?????? Int32 ldest = inet_addr(strClientIP); //目的地的ip ?????? Int32 lhost = inet_addr("");????? //本地服務器的ip ?????? Int64 macinfo = new Int64(); ?????? Int32 len = 6;
?????? int res = SendARP(ldest,0, ref macinfo, ref len); ?????? string mac_src=macinfo.ToString("X"); ?????? if(mac_src == "0") ?????? { ??????? if(userip=="127.0.0.1") ???????? Response.Write ("正在訪問Localhost!"); ??????? else ???????? Response.Write ("歡迎來自IP為" + userip + "的朋友!" + "<br>"); ??????? return; ?????? }
?????? while(mac_src.Length<12) ?????? { ??????? mac_src = mac_src.Insert(0,"0"); ?????? }
?????? string mac_dest="";
?????? for(int i=0;i<11;i++) ?????? { ??????? if (0 == (i % 2)) ??????? { ???????? if ( i == 10 ) ???????? { ????????? mac_dest = mac_dest.Insert(0,mac_src.Substring(i,2)); ???????? } ???????? else ???????? { ????????? mac_dest ="-" + mac_dest.Insert(0,mac_src.Substring(i,2)); ???????? } ??????? } ?????? }
?????? Response.Write ("歡迎來自IP為"+userip+ "<br>" + ",MAC地址為"+mac_dest+"的朋友!"
??????? +????? "<br>"); ????? } ????? catch(Exception err) ????? { ?????? Response.Write(err.Message); ????? } ???? }

?

?

?

在ASP中使用 Request.ServerVariables("REMOTE_ADDR") 來取得客戶端的IP地址,但如果客戶端是使用代理服務器來訪問,那取到的就是代理服務器的IP地址,而不是真正的客戶端IP地址。要想透過代理服務器取得客戶端的真實IP地址,就要使用 Request.ServerVariables("HTTP_X_FORWARDED_FOR") 來讀取。不過要注意的事,并不是每個代理服務器都能用 Request.ServerVariables("HTTP_X_FORWARDED_FOR") 來讀取客戶端的真實 IP,有些用此方法讀取到的仍然是代理服務器的IP。還有一點需要注意的是:如果客戶端沒有通過代理服務器來訪問,那么用 Request.ServerVariables ("HTTP_X_FORWARDED_FOR") 取到的值將是空的。因此,如果要在程序中使用此方法,可以這樣處理:

?

...... userip = Request.ServerVariables("HTTP_X_FORWARDED_FOR") If userip = "" Then userip = Request.ServerVariables("REMOTE_ADDR") ......
即:如果客戶端通過代理服務器,則取 HTTP_X_FORWARDED_FOR 的值,如果沒通過代理服務器,就取 REMOTE_ADDR 的值。

?

以下是引用片段: <%
'通用函數(shù):如果不能取客戶端真實IP,就會取客戶端的代理IP? Private Function getIP()? Dim strIPAddr? If Request.ServerVariables("HTTP_X_FORWARDED_FOR") = "" OR InStr(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), "unknown") > 0 Then? strIPAddr = Request.ServerVariables("REMOTE_ADDR")? ElseIf InStr(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), ",") > 0 Then? strIPAddr = Mid(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), 1, InStr(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), ",")-1)? ElseIf InStr(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), ";") > 0 Then? strIPAddr = Mid(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), 1, InStr(Request.ServerVariables("HTTP_X_FORWARDED_FOR"), ";")-1)? Else? strIPAddr = Request.ServerVariables("HTTP_X_FORWARDED_FOR")? End If? getIP = Trim(Mid(strIPAddr, 1, 30))? End Function response.write getip()
%>

轉(zhuǎn)載于:https://www.cnblogs.com/zhanghonjiang2011/archive/2012/04/13/2445673.html

總結(jié)

以上是生活随笔為你收集整理的获取真实IP的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。