title author date CreateTime categories
asp dotnet core 通过图片统计 csdn 用户访问
lindexi
2019-11-29 08:26:58 +0800
2019-05-26 11:35:14 +0800
dotnet

在 csdn 的访问统计里面,只能用 csdn 提供的访问统计,因为在 csdn 中不支持在博客加上 js 代码,也就是无法使用友盟等工具统计。 通过在 asp dotnet core 创建一个图片链接的方式,将这个链接作为图片放在 csdn 的博客,可以在链接被访问的时候统计用户访问

新建一个 asp dotnet core 项目,在自己的 controler 里面添加一个方法,这个方法的访问链接是 xx.png 假装这是一张图片,请看代码

    [Route("api/[controller]")]
    [ApiController]
    public class ImageController : ControllerBase
    {
    	[Route("csdn/Image.png")]
        [HttpGet]
        public FileResult GetCSDNImage()
        {
        	// 忽略代码
        }
    }

在 csdn 上通过访问我的域名对应的链接就可以调用 GetCSDNImage 方法

![] (http://bulleimage.service.walterlv.com/api/image/csdn/image.png )

注意路由的命名要清真一点,我本来是用来做广告的,于是命名为广告,然而发现在 csdn 无法被访问到,原因请看 asp dotnet core 图片在浏览器没访问可能原因

通过在 GetCSDNImage 返回一张图片,同时按照这个方法调用的次数就可以用来统计用户的访问了

缓存图片

在 GetCSDNImage 需要返回图片才可以用来假装这是一个静态图片的链接

我将一张图片放在输出文件夹,做法就是在解决方案添加一张图片,右击属性设置复制输出到文件夹

在 Startup 的 ConfigureServices 里面添加内存缓存

        public void ConfigureServices(IServiceCollection services)
        {
        	// 忽略其他代码
            services.AddMemoryCache(); // 添加这一行代码就可以使用内存缓存

            services.AddControllers()
                .AddNewtonsoftJson();
            // 忽略代码
        }

修改 ImageController 在构造注入缓存 IMemoryCache 方法

        public ImageController(IMemoryCache memoryCache)
        {
            _cache = memoryCache;
        }

        private IMemoryCache _cache;

这样就可以在 GetCSDNImage 方法里面使用缓存了

在使用缓存之前需要读取输出文件夹里面的图片,我添加一个方法用来读取输出文件夹里面的图片。在 asp dotnet core 中,很多都是使用指定静态的文件夹作为静态资源的文件夹,直接通过输出文件夹读取的比较少。但是设置一个静态文件夹是另外的知识,本文就直接通过输出文件夹读取

使用 Path.GetDirectoryName(Assembly.GetCallingAssembly().Location) 可以拿到输出文件夹,详细请看dotnet 获取程序所在路径的方法

使用下面代码就可以读取图片文件作为字节了,建议只读取一次,解决多线程访问文件的问题

        private byte[] GetImage()
        {
            var file = Path.Combine(Path.GetDirectoryName(Assembly.GetCallingAssembly().Location), "Image.png");

            return System.IO.File.ReadAllBytes(file);
        }

在 GetCSDNImage 方法里面通过缓存,判断如果缓存里面没有值就从文件读取图片,如果有值就直接从内存返回

如果这段代码是需要我自己写,可能要写几天,因为还存在了多线程访问的问题,如果一开始不存在值,那么就需要创建值,如何作为第一次创建值的时候,刚好多个线程进来,只有一个线程创建等问题。还好微软提供的缓存里面有这样的方法 GetOrCreate 方法,尝试从内存获取,如果获取不到就创建,在这个方法里面第一个参数是传入 key 第二个参数就是传入如何创建的方法。在缓存中就通过 key 来获取或设置值,用法和字典差不多

            var file = _cache.GetOrCreate("Image", entry => GetImage());

现在拿到了值,可以通过 File 方法返回,注意在返回的时候添加 content type 说这是一张图片

        public FileResult GetCSDNImage()
        {     
             var file = _cache.GetOrCreate("Image", entry => GetImage());
      
             return File(file, "image/png");
        }

统计用户访问

我不会告诉大家如何去创建数据库去存放用户访问的数据,因为这些需要的知识点有些多,本文的统计用户访问只是通过一个简单的静态变量获取,不考虑并发的问题

        public FileResult GetCSDNImage()
        {     
        	Count++;

        	Console.WriteLine($"总共有{Count}访问");

             
            // 忽略其他代码
        }

        private static int Count { set; get; }

这样就可以完成了统计用户的访问了,同时代码也很少。我还需要用户的 IP 和使用什么浏览器,于是需要添加一点代码

我的网站是通过 frp 让用户访问,需要从 Frp 获取用户真实 IP 地址用法有点复杂

        private static bool TryGetUserIpFromFrp(HttpRequest httpContextRequest, out StringValues ip)
        {
            return httpContextRequest.Headers.TryGetValue("X-Forwarded-For", out ip);
        }

上面这个方法就可以从 HttpRequest 拿到从 frp 获取的用户 ip 地址

        public FileResult GetCSDNImage()
        {
        	// 忽略代码

            if (TryGetUserIpFromFrp(HttpContext.Request, out var ip))
            {
               
            }

        	// 忽略代码
        }

获取用户的浏览器使用 Headers 里面的 User-Agent 可以拿到

        public FileResult GetCSDNImage()
        {
        	// 忽略代码

            if (HttpContext.Request.Headers.TryGetValue("User-Agent", out var userAgent))
            {
               
            }

        	// 忽略代码
        }

将这些值合并输出

        [Route("csdn/Image.png")]
        [HttpGet]
        public FileResult GetCSDNImage()
        {
            Count++;

            StringBuilder str = new StringBuilder();
            str.Append(DateTime.Now);
            str.Append(" ");
            str.Append("用户访问 ");

            Console.WriteLine(GetUserId());
            Console.WriteLine("用户id =" + HttpContextAccessor.HttpContext.Request.HttpContext.Session.Id);

            if (TryGetUserIpFromFrp(HttpContextAccessor.HttpContext.Request, out var ip))
            {
                str.Append("用户Ip=");
                str.Append(ip);
                str.Append(" ");
            }

            str.Append($"总共有{Count}访问");

            if (HttpContext.Request.Headers.TryGetValue("User-Agent", out var userAgent))
            {
                str.Append("\r\n");
                str.Append("当前用户浏览器");
                str.Append(userAgent);
            }

            Console.WriteLine(str);

            var file = _cache.GetOrCreate("Image", entry => GetImage());

            return File(file, "image/png");
        }

我将这个图片放在 csdn 的博客,运行服务在用户访问的时候可以看到下面的输出

2019/5/26 11:39:24 用户访问 用户Ip=58.209.53.254 总共有13访问
当前用户浏览器Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.157 Safari/537.36

如果要分析什么时候的用户访问有多少,就需要添加数据库,这些需要的知识有点多,本文就不告诉大家

代码很简单,都放在 github

dotnet 获取程序所在路径的方法

asp dotnet core 从 Frp 获取用户真实 IP 地址

asp dotnet core 图片在浏览器没访问可能原因