René's URL Explorer Experiment


Title: ClassManager Cache Performance Impact · Issue #1407 · pythonnet/pythonnet · GitHub

Open Graph Title: ClassManager Cache Performance Impact · Issue #1407 · pythonnet/pythonnet

X Title: ClassManager Cache Performance Impact · Issue #1407 · pythonnet/pythonnet

Description: Environment Pythonnet version: Latest Dev 3.0.0 Python version: 3.6.8 Operating System: Windows 10 Details I was doing some performance testing on the latest PythonNet and noticed a significant slow down in the conversion of CLR objects ...

Open Graph Description: Environment Pythonnet version: Latest Dev 3.0.0 Python version: 3.6.8 Operating System: Windows 10 Details I was doing some performance testing on the latest PythonNet and noticed a significant slo...

X Description: Environment Pythonnet version: Latest Dev 3.0.0 Python version: 3.6.8 Operating System: Windows 10 Details I was doing some performance testing on the latest PythonNet and noticed a significant slo...

Opengraph URL: https://github.com/pythonnet/pythonnet/issues/1407

X: @github

direct link

Domain: github.com


Hey, it has json ld scripts:
{"@context":"https://schema.org","@type":"DiscussionForumPosting","headline":"ClassManager Cache Performance Impact","articleBody":"### Environment\r\n\r\n-   Pythonnet version: Latest Dev 3.0.0\r\n-   Python version: 3.6.8\r\n-   Operating System: Windows 10\r\n\r\n### Details\r\nI was doing some performance testing on the latest PythonNet and noticed a significant slow down in the conversion of CLR objects to Python. After some digging I believe I found where the slow down is coming from. \r\n\r\nIn #1287 , the addition of the `MaybeType` changed the `ClassManager` dictionary cache, which ended up storing `ClassBases` with `MaybeType` as the key. But because the `ClassManager` 's `GetClass` function attempts to fetch a value out of the cache using a `Type`, the object is implicitly converted to a MaybeType first. This conversion and then comparison (finding obj in dictionary) leads to a major difference in performance when converting large quantities of an object to Python since we have to initialize a new object every time the cache is used. \r\n\r\n\r\n\r\n-   What commands did you run to trigger this issue? If you can provide a\r\n    [Minimal, Complete, and Verifiable example](http://stackoverflow.com/help/mcve)\r\n    this will help us understand the issue.\r\n\r\nHere is a unit test without any pass/fail that I was using to debug this issue. I paired it with dotTrace to see the difference in performance.\r\n```\r\n        [Test]\r\n        public void Test()\r\n        {\r\n            for (int i = 0; i \u003c 1000000; i++)\r\n            {\r\n                var testSlice = new Slice();\r\n                testSlice.ToPython();\r\n            }\r\n        }\r\n\r\n        public class Slice\r\n        {\r\n            public Dictionary\u003cchar, int\u003e intDict = new Dictionary\u003cchar, int\u003e();\r\n            public List\u003cint\u003e intList = new List\u003cint\u003e();\r\n            public string test = \"pepe\";\r\n            public decimal testDecimal = 5.5m;\r\n\r\n            public Slice()\r\n            {\r\n                for (int i = 0; i \u003c 20; i++)\r\n                {\r\n                    var key = 'A' + i;\r\n                    intDict.Add((char)key, i);\r\n                    intList.Add(i);\r\n                }\r\n\r\n            }\r\n        }\r\n```\r\n\r\n#### Current Branch Performance on 1 million calls of same object:\r\n![image](https://user-images.githubusercontent.com/41277998/109896485-d4d13480-7c45-11eb-8031-065b0db03997.png)\r\n\r\n#### Without MaybeType in ClassManager (Cache dict is \u003cType, ClassBase\u003e) performance:\r\n![image](https://user-images.githubusercontent.com/41277998/109898073-7eb1c080-7c48-11eb-88a1-5613ef566e2c.png)\r\n\r\n\r\n\r\n\r\n### Attempt at a fix while maintaining `MaybeType`\r\n\r\n\r\n\r\nI did attempt to improve the performance by overriding the `Equals()` and `GetHashCode()` functions for `MaybeType` which did show a little improvement and a bit more visibility into the slowdown. But unfortunately the object is already implicit converted before `Equals()` because the ClassManager cache dictionary is \u003cMaybeType, ClassBase\u003e\r\n\r\n`public static implicit operator MaybeType (Type ob) =\u003e new MaybeType(ob);`\r\n\r\nHere was my attempt.\r\n```\r\n        /// \u003csummary\u003e\r\n        /// Determines whether the specified \u003csee cref=\"T:System.Object\"/\u003e is equal to the current \u003csee cref=\"T:System.Object\"/\u003e.\r\n        /// \u003c/summary\u003e\r\n        /// \u003creturns\u003e\r\n        /// true if the specified object  is equal to the current object; otherwise, false.\r\n        /// \u003c/returns\u003e\r\n        /// \u003cparam name=\"obj\"\u003eThe object to compare with the current object. \u003c/param\u003e\u003cfilterpriority\u003e2\u003c/filterpriority\u003e\r\n        public override bool Equals(object obj)\r\n        {\r\n            if (ReferenceEquals(null, obj)) return false;\r\n\r\n            // Compare as a Type\r\n            if (obj is Type typeObj)\r\n            {\r\n                return this.Value == typeObj;\r\n            }\r\n\r\n            // If MaybeType just compare Types\r\n            if (obj is MaybeType maybeObj)\r\n            {\r\n                return Value == maybeObj.Value;\r\n            }\r\n\r\n            // It must be false\r\n            return false;\r\n        }\r\n\r\n        /// \u003csummary\u003e\r\n        /// Serves as a hash function for a particular type.\r\n        /// \u003c/summary\u003e\r\n        /// \u003creturns\u003e\r\n        /// A hash code for the current \u003csee cref=\"T:System.Object\"/\u003e.\r\n        /// \u003c/returns\u003e\r\n        /// \u003cfilterpriority\u003e2\u003c/filterpriority\u003e\r\n        public override int GetHashCode()\r\n        {\r\n            // Use Type HashCode\r\n            unchecked { return Value.GetHashCode(); }\r\n        }\r\n```\r\n\r\nAnd results:\r\n![image](https://user-images.githubusercontent.com/41277998/109897607-d4399d80-7c47-11eb-8539-bcd577843212.png)\r\n\r\n\r\nIt appears that the slowest part is actually the fetching of the fully qualified name, Which could be something that is changed to be lazily fetched when requested instead of during construction.\r\n\r\n\r\n","author":{"url":"https://github.com/C-SELLERS","@type":"Person","name":"C-SELLERS"},"datePublished":"2021-03-04T01:52:11.000Z","interactionStatistic":{"@type":"InteractionCounter","interactionType":"https://schema.org/CommentAction","userInteractionCount":2},"url":"https://github.com/1407/pythonnet/issues/1407"}

route-pattern/_view_fragments/issues/show/:user_id/:repository/:id/issue_layout(.:format)
route-controllervoltron_issues_fragments
route-actionissue_layout
fetch-noncev2:cc283f3c-9031-00c1-2038-e6f702f443e5
current-catalog-service-hash81bb79d38c15960b92d99bca9288a9108c7a47b18f2423d0f6438c5b7bcd2114
request-idC104:DD6B:1FB0929:2AD89FA:69702DA4
html-safe-nonce9e19d55aba213a2a2b5c5fd105135a4bbd5cca6ca594519fc3813c082e35aab3
visitor-payloadeyJyZWZlcnJlciI6IiIsInJlcXVlc3RfaWQiOiJDMTA0OkRENkI6MUZCMDkyOToyQUQ4OUZBOjY5NzAyREE0IiwidmlzaXRvcl9pZCI6IjUwMjY5NzUzMzI5ODg2MjgzODgiLCJyZWdpb25fZWRnZSI6ImlhZCIsInJlZ2lvbl9yZW5kZXIiOiJpYWQifQ==
visitor-hmac8ccefb775407cfe358888e6fcd91dab9ce681f0d7755f2e5098cc469d9a0bf6a
hovercard-subject-tagissue:821672709
github-keyboard-shortcutsrepository,issues,copilot
google-site-verificationApib7-x98H0j5cPqHWwSMm6dNU4GmODRoqxLiDzdx9I
octolytics-urlhttps://collector.github.com/github/collect
analytics-location///voltron/issues_fragments/issue_layout
fb:app_id1401488693436528
apple-itunes-appapp-id=1477376905, app-argument=https://github.com/_view_fragments/issues/show/pythonnet/pythonnet/1407/issue_layout
twitter:imagehttps://opengraph.githubassets.com/9016ac1c3f2fb6a08fe470456397ff6589c0be050335c48e1e864f5048cd339c/pythonnet/pythonnet/issues/1407
twitter:cardsummary_large_image
og:imagehttps://opengraph.githubassets.com/9016ac1c3f2fb6a08fe470456397ff6589c0be050335c48e1e864f5048cd339c/pythonnet/pythonnet/issues/1407
og:image:altEnvironment Pythonnet version: Latest Dev 3.0.0 Python version: 3.6.8 Operating System: Windows 10 Details I was doing some performance testing on the latest PythonNet and noticed a significant slo...
og:image:width1200
og:image:height600
og:site_nameGitHub
og:typeobject
og:author:usernameC-SELLERS
hostnamegithub.com
expected-hostnamegithub.com
None01fa379f5de85ef8e791d09724e69709ce9eb9595278316e0a921312dc88e0bc
turbo-cache-controlno-preview
go-importgithub.com/pythonnet/pythonnet git https://github.com/pythonnet/pythonnet.git
octolytics-dimension-user_id6050430
octolytics-dimension-user_loginpythonnet
octolytics-dimension-repository_id14748123
octolytics-dimension-repository_nwopythonnet/pythonnet
octolytics-dimension-repository_publictrue
octolytics-dimension-repository_is_forkfalse
octolytics-dimension-repository_network_root_id14748123
octolytics-dimension-repository_network_root_nwopythonnet/pythonnet
turbo-body-classeslogged-out env-production page-responsive
disable-turbofalse
browser-stats-urlhttps://api.github.com/_private/browser/stats
browser-errors-urlhttps://api.github.com/_private/browser/errors
releasedda91974c069382b0dfa47b2da7e28bd061c8331
ui-targetfull
theme-color#1e2327
color-schemelight dark

Links:

Skip to contenthttps://github.com/pythonnet/pythonnet/issues/1407#start-of-content
https://github.com/
Sign in https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fpythonnet%2Fpythonnet%2Fissues%2F1407
GitHub CopilotWrite better code with AIhttps://github.com/features/copilot
GitHub SparkBuild and deploy intelligent appshttps://github.com/features/spark
GitHub ModelsManage and compare promptshttps://github.com/features/models
MCP RegistryNewIntegrate external toolshttps://github.com/mcp
ActionsAutomate any workflowhttps://github.com/features/actions
CodespacesInstant dev environmentshttps://github.com/features/codespaces
IssuesPlan and track workhttps://github.com/features/issues
Code ReviewManage code changeshttps://github.com/features/code-review
GitHub Advanced SecurityFind and fix vulnerabilitieshttps://github.com/security/advanced-security
Code securitySecure your code as you buildhttps://github.com/security/advanced-security/code-security
Secret protectionStop leaks before they starthttps://github.com/security/advanced-security/secret-protection
Why GitHubhttps://github.com/why-github
Documentationhttps://docs.github.com
Bloghttps://github.blog
Changeloghttps://github.blog/changelog
Marketplacehttps://github.com/marketplace
View all featureshttps://github.com/features
Enterpriseshttps://github.com/enterprise
Small and medium teamshttps://github.com/team
Startupshttps://github.com/enterprise/startups
Nonprofitshttps://github.com/solutions/industry/nonprofits
App Modernizationhttps://github.com/solutions/use-case/app-modernization
DevSecOpshttps://github.com/solutions/use-case/devsecops
DevOpshttps://github.com/solutions/use-case/devops
CI/CDhttps://github.com/solutions/use-case/ci-cd
View all use caseshttps://github.com/solutions/use-case
Healthcarehttps://github.com/solutions/industry/healthcare
Financial serviceshttps://github.com/solutions/industry/financial-services
Manufacturinghttps://github.com/solutions/industry/manufacturing
Governmenthttps://github.com/solutions/industry/government
View all industrieshttps://github.com/solutions/industry
View all solutionshttps://github.com/solutions
AIhttps://github.com/resources/articles?topic=ai
Software Developmenthttps://github.com/resources/articles?topic=software-development
DevOpshttps://github.com/resources/articles?topic=devops
Securityhttps://github.com/resources/articles?topic=security
View all topicshttps://github.com/resources/articles
Customer storieshttps://github.com/customer-stories
Events & webinarshttps://github.com/resources/events
Ebooks & reportshttps://github.com/resources/whitepapers
Business insightshttps://github.com/solutions/executive-insights
GitHub Skillshttps://skills.github.com
Documentationhttps://docs.github.com
Customer supporthttps://support.github.com
Community forumhttps://github.com/orgs/community/discussions
Trust centerhttps://github.com/trust-center
Partnershttps://github.com/partners
GitHub SponsorsFund open source developershttps://github.com/sponsors
Security Labhttps://securitylab.github.com
Maintainer Communityhttps://maintainers.github.com
Acceleratorhttps://github.com/accelerator
Archive Programhttps://archiveprogram.github.com
Topicshttps://github.com/topics
Trendinghttps://github.com/trending
Collectionshttps://github.com/collections
Enterprise platformAI-powered developer platformhttps://github.com/enterprise
GitHub Advanced SecurityEnterprise-grade security featureshttps://github.com/security/advanced-security
Copilot for BusinessEnterprise-grade AI featureshttps://github.com/features/copilot/copilot-business
Premium SupportEnterprise-grade 24/7 supporthttps://github.com/premium-support
Pricinghttps://github.com/pricing
Search syntax tipshttps://docs.github.com/search-github/github-code-search/understanding-github-code-search-syntax
documentationhttps://docs.github.com/search-github/github-code-search/understanding-github-code-search-syntax
Sign in https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fpythonnet%2Fpythonnet%2Fissues%2F1407
Sign up https://github.com/signup?ref_cta=Sign+up&ref_loc=header+logged+out&ref_page=%2F%3Cuser-name%3E%2F%3Crepo-name%3E%2Fvoltron%2Fissues_fragments%2Fissue_layout&source=header-repo&source_repo=pythonnet%2Fpythonnet
Reloadhttps://github.com/pythonnet/pythonnet/issues/1407
Reloadhttps://github.com/pythonnet/pythonnet/issues/1407
Reloadhttps://github.com/pythonnet/pythonnet/issues/1407
pythonnet https://github.com/pythonnet
pythonnethttps://github.com/pythonnet/pythonnet
Notifications https://github.com/login?return_to=%2Fpythonnet%2Fpythonnet
Fork 770 https://github.com/login?return_to=%2Fpythonnet%2Fpythonnet
Star 5.4k https://github.com/login?return_to=%2Fpythonnet%2Fpythonnet
Code https://github.com/pythonnet/pythonnet
Issues 155 https://github.com/pythonnet/pythonnet/issues
Pull requests 18 https://github.com/pythonnet/pythonnet/pulls
Discussions https://github.com/pythonnet/pythonnet/discussions
Actions https://github.com/pythonnet/pythonnet/actions
Projects 0 https://github.com/pythonnet/pythonnet/projects
Wiki https://github.com/pythonnet/pythonnet/wiki
Security Uh oh! There was an error while loading. Please reload this page. https://github.com/pythonnet/pythonnet/security
Please reload this pagehttps://github.com/pythonnet/pythonnet/issues/1407
Insights https://github.com/pythonnet/pythonnet/pulse
Code https://github.com/pythonnet/pythonnet
Issues https://github.com/pythonnet/pythonnet/issues
Pull requests https://github.com/pythonnet/pythonnet/pulls
Discussions https://github.com/pythonnet/pythonnet/discussions
Actions https://github.com/pythonnet/pythonnet/actions
Projects https://github.com/pythonnet/pythonnet/projects
Wiki https://github.com/pythonnet/pythonnet/wiki
Security https://github.com/pythonnet/pythonnet/security
Insights https://github.com/pythonnet/pythonnet/pulse
New issuehttps://github.com/login?return_to=https://github.com/pythonnet/pythonnet/issues/1407
New issuehttps://github.com/login?return_to=https://github.com/pythonnet/pythonnet/issues/1407
ClassManager Cache Performance Impacthttps://github.com/pythonnet/pythonnet/issues/1407#top
https://github.com/C-SELLERS
https://github.com/C-SELLERS
C-SELLERShttps://github.com/C-SELLERS
on Mar 4, 2021https://github.com/pythonnet/pythonnet/issues/1407#issue-821672709
#1287https://github.com/pythonnet/pythonnet/pull/1287
Minimal, Complete, and Verifiable examplehttp://stackoverflow.com/help/mcve
https://user-images.githubusercontent.com/41277998/109896485-d4d13480-7c45-11eb-8031-065b0db03997.png
https://user-images.githubusercontent.com/41277998/109898073-7eb1c080-7c48-11eb-88a1-5613ef566e2c.png
https://user-images.githubusercontent.com/41277998/109897607-d4399d80-7c47-11eb-8539-bcd577843212.png
https://github.com
Termshttps://docs.github.com/site-policy/github-terms/github-terms-of-service
Privacyhttps://docs.github.com/site-policy/privacy-policies/github-privacy-statement
Securityhttps://github.com/security
Statushttps://www.githubstatus.com/
Communityhttps://github.community/
Docshttps://docs.github.com/
Contacthttps://support.github.com?tags=dotcom-footer

Viewport: width=device-width


URLs of crawlers that visited me.