简介:
在.NET框架中,C#语言的类(class)属于引用类型。
这意味着当你声明一个类的实例时,实际上是在堆上创建一个对象,并在栈上创建一个引用指向这个对象。
因此,当你将对象作为参数传递给函数时,实际上是传递了这个引用的副本,而不是对象本身。
这就是所谓的"传引用"或"按引用传递"。
让我们深入探讨一下这个问题,以标题和描述中给出的代码为例:```csharpclass Program{ static void Main(string[] args) { TestClass objA = new TestClass(); objA.Name = "I am ObjA"; Console.WriteLine(String.Format("In Main:{0}", objA.Name)); TestFun(objA); Console.WriteLine(String.Format("In Main:{0}", objA.Name)); Console.Read(); } static void TestFun(TestClass obj) { obj.Name = "I am be modified in TestFun"; Console.WriteLine(String.Format("In TestFun:{0}", obj.Name)); } public class TestClass { public string Name { get; set; } }}```在这个例子中,`TestFun`函数接收到`objA`的引用副本`obj`。
当在`TestFun`中修改`obj.Name`时,实际上是修改了`objA`引用的对象,因为它们都指向同一个堆上的实例。
因此,`Main`函数中再次打印`objA.Name`时,值已经被修改为"I am be modified in TestFun"。
然而,如果我们更改`TestFun`的实现:```csharpstatic void TestFun(TestClass obj){ TestClass objB = new TestClass(); obj = objB; obj.Name = "I am ObjB"; Console.WriteLine(String.Format("In TestFun:{0}", obj.Name));}```这里我们创建了一个新的`TestClass`实例`objB`,然后让`obj`引用`objB`。
虽然在`TestFun`内部`obj`的值改变了,但这不会影响`Main`函数中的`objA`,因为`objA`仍然指向原始的`TestClass`实例。
所以,`Main`函数中的`objA.Name`输出仍然是"I am ObjA",因为`objA`并没有被修改指向新创建的`objB`。
这个现象可以用内存模型来解释,就像描述中提到的那样。
在调用`TestFun`时,`objA`的地址被复制到`obj`,但是`objA`本身并未改变。
在`TestFun`中,`obj`被重新分配给`objB`的地址,但`objA`仍然指向原始对象,所以`Main`中的`objA`不会受到影响。
C#中的对象参数传递特性对于理解和调试代码非常重要。
理解这种行为可以帮助我们避免意外地修改了原本不想修改的对象,同时也能有效地利用引用传递来共享和修改数据。
在编写函数时,要清楚地知道参数是值类型(value type,如int、struct)还是引用类型(reference type,如class),因为这将直接影响到参数的处理方式和函数的行为。
2025/6/15 20:02:36 44KB
1
简介:
### Spring注解学习:构建简单Web应用#### 引言Spring框架自引入注解支持以来,极大地简化了Java开发中的依赖注入与配置管理过程。
本文将深入探讨如何利用Spring注解来构建一个简单的Web应用,从控制器(Controller)到数据访问对象(DAO),通过实例演示注解在不同层次的应用。
#### Spring注解概述Spring框架提供了多种注解来简化应用的配置和组件的定义。
以下是一些常用的Spring注解:- `@Component`:标记类为Spring的Bean,可以被Spring容器管理和注入到其他Bean中。
- `@Repository`:用于数据访问层,通常标记DAO类,提供额外的异常转换支持。
- `@Service`:用于业务逻辑层,表示服务层的Bean。
- `@Controller`:用于Web层,表示一个控制层的Bean,处理HTTP请求。
- `@Autowired`:自动装配Bean,用于字段或构造函数,无需手动设置依赖。
- `@RequestMapping`:映射Web请求到特定的方法上,用于控制器类或方法上。
- `@Transactional`:用于方法上,声明该方法需要在事务中执行。
#### 构建Web应用:关键步骤1. **项目搭建**:创建一个Web项目,并添加必要的Jar包,如Spring框架的各个模块、AOP联盟、日志库等。
文中提到的Jar包包括aopalliance-1.0.jar、commons-logging-1.1.1.jar、log4j-1.2.15.jar等,这些包对于Spring框架的正常运行至关重要。
2. **配置web.xml**:这是Web应用的部署描述符,用于配置Servlet、过滤器等。
在本例中,配置了Spring的上下文参数、Log4J的日志配置以及字符编码过滤器,确保应用能够正确读取配置并处理请求。
```xml <context-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/applicationContext.xml</param-value> </context-param> ``` 这段配置指定了Spring的配置文件位置,即`applicationContext.xml`。
3. **编写控制器**:使用`@Controller`注解定义控制器类,并使用`@RequestMapping`注解来指定URL映射。
例如: ```java @Controller public class HelloWorldController { @RequestMapping("/hello") public String helloWorld() { return "hello"; } } ```4. **数据访问层**:使用`@Repository`注解定义DAO类,负责数据的存取操作。
例如: ```java @Repository public class UserRepository { // 数据库操作方法 } ```5. **业务逻辑层**:使用`@Service`注解定义服务层,处理业务逻辑。
例如: ```java @Service public class UserService { @Autowired private UserRepository userRepository; // 业务逻辑方法 } ```6. **事务管理**:在业务逻辑中,可能需要使用`@Transactional`注解来确保数据的一致性和完整性。
7. **测试**:对应用进行单元测试和集成测试,确保各部分功能按预期工作。
#### 结论通过上述步骤,我们可以构建一个基于Spring注解的简单Web应用。
Spring注解的使用极大地简化了配置,提高了开发效率,使得开发者能够更加专注于业务逻辑的实现。
在未来的学习中,我们将更深入地探讨每一层的细节,以及如何利用Spring注解来优化和扩展应用的功能。
2025/6/15 19:51:25 187KB
1
简介:
### CAS单点登录服务器配置详解#### 一、CAS单点登录概述CAS(Central Authentication Service)是一种开放源代码的单点登录协议和服务框架,它为Web应用提供了一种简化了的身份验证流程。
通过CAS,用户只需要在一个地方完成登录过程,即可在多个应用间共享登录状态,无需重复登录。
#### 二、CAS服务器安装与配置##### 2.1 安装CAS服务端1. **下载CAS服务端**:首先从官方网址http://www.cas.org/下载最新的CAS服务端压缩包。
2. **部署WAR包**:将下载的WAR包复制到Tomcat的webapps目录下,并将其重命名为`cas.war`。
3. **启动Tomcat**:启动Tomcat服务器,自动解压WAR包,此时会在Tomcat的webapps目录下生成一个名为`cas`的文件夹。
4. **访问CAS**:通过浏览器访问`http://localhost:8896/cas`来测试CAS服务是否正常启动。
##### 2.2 配置CAS使用数据库验证为了实现更安全、更灵活的身份验证机制,我们可以配置CAS使用数据库进行用户身份验证。
具体步骤如下:1. **修改部署配置文件**:打开`cas-server-webapp\WEB-INF\deployerConfigContext.xml`文件,找到`SimpleTestUsernamePasswordAuthenticationHandler`配置项,将其替换为`QueryDatabaseAuthenticationHandler`。
```xml <bean id="authenticationHandler" class="org.jasig.cas.authentication.handler.QueryDatabaseAuthenticationHandler"> <!-- 数据库连接数据源 --> <property name="dataSource" ref="dataSource"/> <!-- 查询语句 --> <property name="sql" value="SELECT password FROM users WHERE username = ?"/> <!-- 密码加密方式 --> <property name="passwordEncoder" ref="passwordEncoder"/> </bean> ```2. **配置数据库连接**:在同一文件中添加一个新的`dataSource` bean来定义数据库连接信息。
```xml <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver"/> <property name="url" value="jdbc:mysql://localhost:3306/casdb"/> <property name="username" value="casuser"/> <property name="password" value="password"/> </bean> ```3. **配置密码加密方式**:继续在同一文件中添加`passwordEncoder` bean来指定密码加密方式,这里使用MD5作为示例。
```xml <bean id="passwordEncoder" class="org.springframework.security.crypto.password.StandardPasswordEncoder"> <constructor-arg value="MD5"/> </bean> ```4. **测试数据库验证**:重启Tomcat服务器,访问CAS服务器页面,使用数据库中的用户名和密码尝试登录,验证是否可以成功登录。
#### 三、CAS工作原理CAS的工作原理主要分为以下几个步骤:1. **用户访问服务**:用户首次访问受保护的资源时,CAS客户端会检测到HTTP请求中缺少ServiceTicket(简称ST),表明用户尚未经过身份验证。
2. **重定向至CAS服务器**:CAS客户端会将用户重定向到CAS服务器进行身份验证,并携带用户的请求URL作为参数(service参数)。
3. **用户认证**:CAS服务器接收到来自用户的认证请求后,引导用户进入登录页面。
用户输入用户名和密码进行登录,若身份验证成功,则CAS服务器通过HTTPS协议返回一个TGC(Ticket-Granting Cookie)给浏览器。
4. **发放ServiceTicket**:CAS服务器生成一个随机的ServiceTicket(简称ST),并将用户重定向回CAS客户端。
5. **验证ServiceTicket**:CAS客户端收到ST后,向CAS服务器验证ST的有效性。
如果验证通过,则允许用户访问受保护资源。
6. **传输用户信息**:CAS服务器验证ST通过后,将用户的相关认证信息发送给CAS客户端。
通过以上步骤,CAS实现了单点登录的功能,极大地提升了用户体验和系统的安全性。
### 四、CAS与HTTPS在配置CAS服务器时,可以选择使用HTTPS协议来增强通信的安全性。
如果选择HTTPS协议,则需要在服务器上配置CAS证书。
证书的创建和导入过程可以参考以下链接:[http://m.blog..net/zrk1000/article/details/51166603](http://m.blog..net/zrk1000/article/details/51166603)### 总结本文详细介绍了如何配置CAS单点登录服务,并重点讲解了如何利用Java代码实现CAS的配置,包括使用数据库进行登录验证的具体步骤。
同时,还阐述了CAS的基本工作原理,帮助读者更好地理解CAS的工作流程和技术细节。
2025/6/15 19:47:19 293KB
1
简单注释,通俗易懂,实现okhttp基本功能:1、Get请求(同步和异步);
2、POST请求表单(key-value);
3、POST请求提交(JSON/String等);
4、文件下载;
5、文件上传;
6、图片缓存加载
2025/6/8 5:03:28 4.02MB okhttp框架 上传 下载 post
1
如何理解深度强化学习基本概念:value-based,policy-based,off-policy,on-policy。
以及A3C算法
2025/6/7 7:14:30 2.64MB A3C DRL
1
做网站或某些基于网络的软件可能需要根据用户端IP地址查询其物理地址信息,那么就会用到IP数据库。
本文件是我根据2011年5月份的纯真IP数据库进行整合而成,合并了部分同一地区IP,将原来的记录由30多万缩减到17万多条,并将其中的IP数据格式转换成了整数形式,方便查询,提高了查询效率。
附:IP格式转换可参考下面两个VB.NET函数PublicSharedFunctionGetIpFromInt32(ByValvalueAsUInt32)AsStringReturn(value>>24).ToString()&"."&(value24).ToString()&"."&(value<>24).ToString()EndFunctionPublicSharedFunctionGetIpFromString(ByValvalueAsString)AsUInt32DimsAsString()=value.Split("."c)ReturnUInt32.Parse(s(0))<<24OrUInt32.Parse(s(1))<<16OrUInt32.Parse(s(2))<<8OrUInt32.Parse(s(3))EndFunction
2025/5/28 3:11:47 8.13MB IP数据库 IP地址
1
bitcoinj项目富含完整demo此项目使用maven构建,不会使用maven的同学,查看项目pom.xml文件,并在http://mvnrepository.com/下载相应的依赖jar包.demo:bitcoinj签名交易/***@paramunSpentBTCList未花费utxo集合*@paramfrom发送者地址*@paramto接收者地址*@paramprivateKey私钥*@paramvalue发送金额.单位:聪*@paramfee旷工费.单位:聪*@return签名之后未广播的原生交易字符串*@throwsException*/publicstaticStringsignBTCTransactionData(ListunSpentBTCList,Stringfrom,Stringto,StringprivateKey,longvalue,longfee)throwsException{NetworkParametersnetworkParameters=null;//networkParameters=MainNetParams.get();//测试网络networkParameters=TestNet3Params.get();Transactiontransaction=newTransaction(networkParameters);DumpedPrivateKeydumpedPrivateKey=DumpedPrivateKey.fromBase58(networkParameters,privateKey);ECKeyecKey=dumpedPrivateKey.getKey();longtotalMoney=0;Listutxos=newArrayList();//遍历未花费列表,组装合适的itemfor(UnSpentBTCus:unSpentBTCList){if(totalMoney>=(value+fee))break;UTXOutxo=newUTXO(Sha256Hash.wrap(us.getTxid()),us.getVout(),Coin.valueOf(us.getSatoshis()),us.getHeight(),false,newScript(Hex.decode(us.getScriptPubKey())));utxos.add(utxo);totalMoney+=us.getSatoshis();}transaction.addOutput(Coin.valueOf(value),Address.fromBase58(networkParameters,to));//transaction.//消费列表总金额-已经转账的金额-手续费就等于需要返回给自己的金额了longbalance=totalMoney-value-fee;//输出-转给自己if(balance>0){transaction.addOutput(Coin.valueOf(balance),Address.fromBase58(networkParameters,from));}//输入未消费列表项for(UTXOutxo:utxos){TransactionOutPointoutPoint=newTransactionOutPoint(networkParameters,utxo.getIndex(),utxo.getHash());
1
Flutter-Mqtts-esp8266物联网Flutter-mqtt客户端支持MQTTS双向加密通信,域名证书认证!Flutter-mqtt客户端支持:eps8266,nodemcu等.数据发送格式:{"SW":"A","onOff":"true"}SW代表开关,value值A,第一个,onOff代表开关状态,value值true是开,false是关.mqtt客户端,APP端订阅的是:topic='mqttMobile';//Notawildcardtopicesp8266客户端订阅的是:pubTopic='esp8266_client';
2025/5/5 3:40:57 332KB esp8266 mqtt mqtts 物联网
1
材料UI芯片输入该项目为提供了一个。
它受到启发。
如果您想亲自尝试该组件而不是观看gif,请转到进行实时演示!安装npmi--savematerial-ui-chip-input@next注意:这是Material-UI1.0.0或更高版本的版本。
如果您仍在使用Material-UI0.x,则可以使用我们的。
用法该组件支持受控或不受控制的输入模式。
如果使用受控模式(通过设置value属性),则不会调用onChange回调。
importChipInputfrom'material-ui-chip-input'//uncontrolledinputhandleChange(chips)}/>//controlledinputhandle
2025/5/2 3:23:05 474KB react material-design input material-ui
1
input{jdbc{#是否记录上次执行结果,如果为真,将会把上次执行到的tracking_column字段的值记录下来,保存到last_run_metadata_path指定的文件中record_last_run=>true#是否需要记录某个column的值,如果record_last_run为真,可以自定义我们需要track的column名称,此时该参数就要为true.否则默认track的是timestamp的值.use_column_value=>true#如果use_column_value为真,需配置此参数.track的数据库column名,该column必须是递增的.比如:ID.tracking_column=>MY_ID#指定文件,来记录上次执行到的tracking_column字段的值#我们只需要在SQL语句中WHEREMY_ID>:last_sql_value即可.其中:last_sql_value取得就是该文件中的值(10000).last_run_metadata_path=>"/etc/logstash/run_metadata.d/my_info"#是否清除last_run_metadata_path的记录,清除相当于从头开始查询所有的数据库记录clean_run=>false#是否将column名称转小写lowercase_column_names=>false#存放需要执行的SQL语句的文件位置statement_filepath=>"/etc/logstash/statement_file.d/my_info.sql"}}
2025/4/14 3:44:01 1KB logstash jdbc
1
共 92 条记录 首页 上一页 下一页 尾页
在日常工作中,钉钉打卡成了我生活中不可或缺的一部分。然而,有时候这个看似简单的任务却给我带来了不少烦恼。 每天早晚,我总是得牢记打开钉钉应用,点击"工作台",再找到"考勤打卡"进行签到。有时候因为工作忙碌,会忘记打卡,导致考勤异常,影响当月的工作评价。而且,由于我使用的是苹果手机,有时候系统更新后,钉钉的某些功能会出现异常,使得打卡变得更加麻烦。 另外,我的家人使用的是安卓手机,他们也经常抱怨钉钉打卡的繁琐。尤其是对于那些不太熟悉手机操作的长辈来说,每次打卡都是一次挑战。他们总是担心自己会操作失误,导致打卡失败。 为了解决这些烦恼,我开始思考是否可以通过编写一个全自动化脚本来实现钉钉打卡。经过一段时间的摸索和学习,我终于成功编写出了一个适用于苹果和安卓系统的钉钉打卡脚本。
2024-04-09 15:03 15KB 钉钉 钉钉打卡